Python编程需注意缩进强制性、赋值即绑定、默认参数陷阱、is/==区别、for/else语义、模块导入规则及f-string特性等易忽略细节,这些决定代码健壮性与可维护性。

Python看似简单,但很多语法细节在实际编码中容易被忽略,导致隐性bug、可读性下降或性能问题。掌握这些“不写错但写不好”的规则,比死记硬背语法更重要。
缩进不是风格问题,是语法强制要求
Python用缩进来定义代码块,而非大括号。同一代码块内必须使用**完全一致的缩进方式**(空格或Tab),混合使用会直接报 IndentationError。PEP 8 明确推荐用4个空格,且禁止Tab与空格混用。
- 函数体、if/for/while/try等后必须跟冒号,下一行开始缩进
- 空代码块不能留空行,需用
pass占位 - 多行语句(如长列表、字典)内部缩进不影响逻辑块,但建议对齐提升可读性
赋值即绑定:变量不是盒子,而是标签
Python中赋值操作(=)本质是将名称绑定到对象,不复制数据。对可变对象(如list、dict)修改其内容,所有引用该对象的变量都会看到变化。
-
a = [1, 2]; b = a→ 修改b.append(3)后,a也变成[1, 2, 3] - 需要独立副本时,用
b = a.copy()(浅拷贝)或import copy; b = copy.deepcopy(a) - 不可变对象(str、int、tuple)赋值后修改会创建新对象,原绑定不受影响
默认参数陷阱:只在定义时求值一次
函数定义时,如果默认参数是可变对象(如列表、字典),它会在函数首次加载时创建并复用,后续调用若修改该对象,会影响下一次调用。
立即学习“Python免费学习笔记(深入)”;
- 错误写法:
def add_item(item, lst=[]): lst.append(item); return lst→ 多次调用会累积元素 - 正确写法:
def add_item(item, lst=None): if lst is None: lst = []; lst.append(item); return lst - 本质原因:函数对象的
__defaults__属性存储默认值,且只初始化一次
is 和 == 的本质区别:身份 vs 相等
== 调用对象的 __eq__ 方法判断逻辑相等;is 判断两个变量是否指向**同一内存地址**(即是否为同一个对象)。
- 小整数(-5 到 256)和短字符串会被Python缓存,此时
is偶尔成立,但属实现细节,不可依赖 - 检查是否为
None必须用is None,这是PEP 8明确规定的惯用法 - 自定义类中若重写了
__eq__,==行为可变,而is始终不变
for/else 和 try/else 的真实含义
这两个 else 子句都表示“未发生中断时执行”,而非“条件不满足时执行”。
-
for x in seq:循环正常结束(没遇到break)才执行else块 -
try:中没抛出异常才执行else块(常用于避免把不应捕获的代码放进try) -
else不是if的配套,而是对应for或try的“成功路径”
模块导入的查找顺序与相对导入限制
Python按 sys.path 列表顺序查找模块:脚本所在目录 → PYTHONPATH → 标准库路径 → site-packages。相对导入(from . import xxx)只能在包内使用,且仅限于 from 语句中。
- 直接运行包内模块(
python mypackage/mymodule.py)会导致相对导入失败,因模块被视为__main__而非包的一部分 - 应使用
python -m mypackage.mymodule运行,确保包上下文完整 - 绝对导入更清晰可靠,优先使用
from mypackage import submodule
字符串格式化:f-string 不只是语法糖
f-string(f"...")在运行时求值表达式,支持嵌套、调用函数、甚至带格式说明符(如 f"{x:.2f}"),且性能最优。
- 表达式内不能有反斜杠、注释或未闭合的括号
- 调试时可用
f"{x=}"(Python 3.8+)自动输出变量名和值 - 不要在f-string中拼接敏感信息(如密码),因为表达式在编译期已确定,可能被静态分析工具误判
不复杂但容易忽略。真正写好Python,不在“会不会”,而在“知不知为什么这样”。










