30天学会Python编程:4. Python运算符与表达式
off999 2025-07-07 22:18 38 浏览 0 评论
4.1 运算符概述
4.1.1 运算符分类
Python运算符可分为以下几大类:
4.1.2 运算符优先级
表4-1 Python运算符优先级(从高到低)
运算符 | 描述 | 示例 |
** | 指数 | 2**3 → 8 |
~ + - | 按位取反/正负号 | ~5, +x, -y |
* / // % | 乘/除/整除/取模 | 10//3 → 3 |
+ - | 加/减 | 3+5 → 8 |
<< >> | 位移 | 4<<1 → 8 |
& | 按位与 | 5&3 → 1 |
^ | 按位异或 | 5^3 → 6 |
| | 按位或 | 5|3 → 7 |
< <= > >= | 比较运算 | 3<5 → True |
== != | 等于/不等于 | 3==3 → True |
is, is not | 身份运算 | x is None |
in, not in | 成员运算 | 2 in [1,2,3] |
not | 逻辑非 | not True → False |
and | 逻辑与 | True and False → False |
or | 逻辑或 | True or False → True |
4.2 算术运算符
4.2.1 基本算术运算
a, b = 10, 3
print(a + b) # 加法 13
print(a - b) # 减法 7
print(a * b) # 乘法 30
print(a / b) # 除法 3.333...
print(a // b) # 整除 3
print(a % b) # 取模 1
print(a ** b) # 指数 1000
4.2.2 增强算术运算
x = 5
x += 2 # 等价于 x = x + 2
x **= 3 # 等价于 x = x ** 3
print(x) # 343
4.2.3 特殊算术行为
# 除法总是返回浮点数
print(10 / 2) # 5.0
# 整数除法向下取整
print(-7 // 2) # -4
# 复数运算
c = 3 + 4j
print(c * 2) # (6+8j)
4.3 比较运算符
4.3.1 基本比较运算
a, b = 5, 7
print(a == b) # False
print(a != b) # True
print(a < b) # True
print(a >= b) # False
4.3.2 链式比较
x = 5
print(1 < x < 10) # True (等价于 1 < x and x < 10)
# 支持多个比较
print(1 < x <= 5 < 10) # True
4.3.3 特殊比较情况
# 浮点数比较
print(0.1 + 0.2 == 0.3) # False (精度问题)
print(abs((0.1 + 0.2) - 0.3) < 1e-10) # 正确比较方式
# 不同类型比较
print(10 == 10.0) # True (值相等)
print(10 is 10.0) # False (不是同一对象)
4.4 逻辑运算符
4.4.1 基本逻辑运算
x, y = True, False
print(x and y) # False
print(x or y) # True
print(not x) # False
4.4.2 短路求值特性
def check():
print("check()被调用")
return True
False and check() # check()不会被调用
True or check() # check()不会被调用
4.4.3 实际应用技巧
# 默认值设置
name = user_input or "匿名用户"
# 安全访问字典
value = my_dict.get("key") or default_value
# 条件执行
is_admin and admin_only_function()
4.5 位运算符
4.5.1 基本位运算
a, b = 0b1010, 0b1100 # 10, 12
print(bin(a & b)) # 按位与 0b1000 (8)
print(bin(a | b)) # 按位或 0b1110 (14)
print(bin(a ^ b)) # 按位异或 0b0110 (6)
print(bin(~a)) # 按位取反 -0b1011 (-11)
print(bin(a << 2)) # 左移 0b101000 (40)
print(bin(b >> 1)) # 右移 0b0110 (6)
4.5.2 位运算应用
# 快速乘除2的幂次
print(5 << 1) # 10 (5*2)
print(20 >> 2) # 5 (20/4)
# 判断奇偶
num = 7
is_odd = num & 1 # 1为奇,0为偶
# 交换两个数
a, b = 3, 4
a ^= b
b ^= a
a ^= b
print(a, b) # 4, 3
4.6 成员与身份运算符
4.6.1 成员运算符(in)
lst = [1, 2, 3]
dct = {"a": 1, "b": 2}
print(2 in lst) # True
print(4 not in lst) # True
print("a" in dct) # True (检查键)
print(1 in dct) # False
4.6.2 身份运算符(is)
x = [1, 2]
y = [1, 2]
z = x
print(x == y) # True (值相等)
print(x is y) # False (不同对象)
print(x is z) # True (同一对象)
# 小整数缓存
a = 256
b = 256
print(a is b) # True (Python缓存小整数)
c = 257
d = 257
print(c is d) # False (大整数不缓存)
4.7 海象运算符(:=)
4.7.1 基本用法
Python 3.8+引入的赋值表达式运算符:
# 传统写法
n = len("hello")
if n > 3:
print(n)
# 使用海象运算符
if (n := len("hello")) > 3:
print(n)
4.7.2 实用场景
# 循环中读取并判断
while (line := input()) != "quit":
print(f"输入: {line}")
# 列表推导式
numbers = [1, 2, 3, 4, 5]
squares = [y for x in numbers if (y := x**2) > 10]
4.8 运算符重载
4.8.1 类中运算符重载
class Vector:
def __init__(self, x, y):
self.x = x
self.y = y
def __add__(self, other):
return Vector(self.x + other.x, self.y + other.y)
def __str__(self):
return f"Vector({self.x}, {self.y})"
v1 = Vector(2, 3)
v2 = Vector(1, 4)
print(v1 + v2) # Vector(3, 7)
4.8.2 可重载运算符列表
表4-2 常用运算符重载方法
运算符 | 方法名 | 说明 |
+ | __add__ | 加法 |
- | __sub__ | 减法 |
* | __mul__ | 乘法 |
/ | __truediv__ | 真除法 |
// | __floordiv__ | 整除 |
% | __mod__ | 取模 |
** | __pow__ | 幂运算 |
<< | __lshift__ | 左移 |
>> | __rshift__ | 右移 |
& | __and__ | 按位与 |
| | __or__ | 按位或 |
^ | __xor__ | 按位异或 |
< | __lt__ | 小于 |
== | __eq__ | 等于 |
4.9 综合应用
案例1:简易计算器
def calculator():
"""支持加减乘除和指数的计算器"""
while True:
try:
expr = input("请输入表达式(或q退出): ").strip()
if expr.lower() == 'q':
break
# 使用eval实现简单计算(生产环境应使用更安全的方式)
result = eval(expr)
print(f"结果: {result:.2f}" if isinstance(result, float) else f"结果: {result}")
except (SyntaxError, NameError):
print("错误:无效表达式")
except ZeroDivisionError:
print("错误:不能除以零")
except Exception as e:
print(f"错误:{e}")
if __name__ == "__main__":
calculator()
案例2:权限控制系统
# 使用位运算管理权限
READ = 0b0001 # 1
WRITE = 0b0010 # 2
EXECUTE = 0b0100 # 4
ADMIN = 0b1000 # 8
class User:
def __init__(self, name, permissions=0):
self.name = name
self.permissions = permissions
def add_permission(self, perm):
self.permissions |= perm
def remove_permission(self, perm):
self.permissions &= ~perm
def has_permission(self, perm):
return self.permissions & perm == perm
def __str__(self):
perms = []
if self.has_permission(READ): perms.append("READ")
if self.has_permission(WRITE): perms.append("WRITE")
if self.has_permission(EXECUTE): perms.append("EXECUTE")
if self.has_permission(ADMIN): perms.append("ADMIN")
return f"{self.name} 权限: {', '.join(perms) or '无'}"
# 使用示例
user = User("Alice")
user.add_permission(READ | WRITE)
print(user) # Alice 权限: READ, WRITE
print("有写入权限:", user.has_permission(WRITE)) # True
print("有执行权限:", user.has_permission(EXECUTE)) # False
4.10 学习路线图
4.11 学习总结
- 核心要点:
- 掌握各类运算符的优先级和结合性
- 理解逻辑运算符的短路特性
- 熟练使用成员和身份运算符
- 了解位运算的实际应用场景
- 实践建议:
- 复杂表达式使用括号明确优先级
- 使用is比较单例对象(如None)
- 避免在复杂表达式中使用海象运算符
- 合理重载运算符增强类功能
- 进阶方向:
- 运算符的魔术方法实现
- 自定义异常处理运算错误
- 使用functools.total_ordering简化比较运算
- 元编程中的运算符控制
- 常见陷阱:
- 混淆==和is的用途
- 忽略浮点数比较的精度问题
- 误用可变对象的+=运算
- 忘记运算符优先级导致的逻辑错误
持续更新Python编程学习日志与技巧,敬请关注!
相关推荐
- Linux 网络协议栈_linux网络协议栈
-
前言;更多学习资料(包含视频、技术学习路线图谱、文档等)后台私信《资料》免费领取技术点包含了C/C++,Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,Z...
- 揭秘 BPF map 前生今世_bpfdm
-
1.前言众所周知,map可用于内核BPF程序和用户应用程序之间实现双向的数据交换,为BPF技术中的重要基础数据结构。在BPF程序中可以通过声明structbpf_map_def...
- 教你简单 提取fmpeg 视频,音频,字幕 方法
-
ffmpeg提取视频,音频,字幕方法(HowtoExtractVideo,Audio,SubtitlefromOriginalVideo?)1.提取视频(ExtractVi...
- Linux内核原理到代码详解《内核视频教程》
-
Linux内核原理-进程入门进程进程不仅仅是一段可执行程序的代码,通常进程还包括其他资源,比如打开的文件,挂起的信号,内核内部的数据结构,处理器状态,内存地址空间,或多个执行线程,存放全局变量的数据段...
- Linux C Socket UDP编程详解及实例分享
-
1、UDP网络编程主要流程UDP协议的程序设计框架,客户端和服务器之间的差别在于服务器必须使用bind()函数来绑定侦听的本地UDP端口,而客户端则可以不进行绑定,直接发送到服务器地址的某个端口地址。...
- libevent源码分析之bufferevent使用详解
-
libevent的bufferevent在event的基础上自己维护了一个buffer,这样的话,就不需要再自己管理一个buffer了。先看看structbufferevent这个结构体struct...
- 一次解决Linux内核内存泄漏实战全过程
-
什么是内存泄漏:程序向系统申请内存,使用完不需要之后,不释放内存还给系统回收,造成申请的内存被浪费.发现系统中内存使用量随着时间的流逝,消耗的越来越多,例如下图所示:接下来的排查思路是:1.监控系统中...
- 彻底搞清楚内存泄漏的原因,如何避免内存泄漏,如何定位内存泄漏
-
作为C/C++开发人员,内存泄漏是最容易遇到的问题之一,这是由C/C++语言的特性引起的。C/C++语言与其他语言不同,需要开发者去申请和释放内存,即需要开发者去管理内存,如果内存使用不当,就容易造成...
- linux网络编程常见API详解_linux网络编程视频教程
-
Linux网络编程API函数初步剖析今天我们来分析一下前几篇博文中提到的网络编程中几个核心的API,探究一下当我们调用每个API时,内核中具体做了哪些准备和初始化工作。1、socket(family...
- Linux下C++访问web—使用libcurl库调用http接口发送解析json数据
-
一、背景这两天由于一些原因研究了研究如何在客户端C++代码中调用web服务端接口,需要访问url,并传入json数据,拿到返回值,并解析。 现在的情形是远程服务端的接口参数和返回类型都是json的字符...
- 平衡感知调节:“系统如人” 视角下的架构设计与业务稳定之道
-
在今天这个到处都是数字化的时代,系统可不是一堆冷冰冰的代码。它就像一个活生生的“数字人”,没了它,业务根本转不起来。总说“技术要为业务服务”,但实际操作起来问题不少:系统怎么才能快速响应业务需求?...
- 谈谈分布式文件系统下的本地缓存_什么是分布式文件存储
-
在分布式文件系统中,为了提高系统的性能,常常会引入不同类型的缓存存储系统(算法优化所带来的的效果可能远远不如缓存带来的优化效果)。在软件中缓存存储系统一般可分为了两类:一、分布式缓存,例如:Memca...
- 进程间通信之信号量semaphore--linux内核剖析
-
什么是信号量信号量的使用主要是用来保护共享资源,使得资源在一个时刻只有一个进程(线程)所拥有。信号量的值为正的时候,说明它空闲。所测试的线程可以锁定而使用它。若为0,说明它被占用,测试的线程要进入睡眠...
- Qt编写推流程序/支持webrtc265/从此不用再转码/打开新世界的大门
-
一、前言在推流领域,尤其是监控行业,现在主流设备基本上都是265格式的视频流,想要在网页上直接显示监控流,之前的方案是,要么转成hls,要么魔改支持265格式的flv,要么265转成264,如果要追求...
- 30 分钟搞定 SpringBoot 视频推拉流!实战避坑指南
-
30分钟搞定SpringBoot视频推拉流!实战避坑指南在音视频开发领域,SpringBoot凭借其快速开发特性,成为很多开发者实现视频推拉流功能的首选框架。但实际开发中,从环境搭建到流处理优...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)