
本文解析 python 链式赋值(如 `final_dict = start_dict["c"] = 7`)的执行逻辑,说明为何它不会将整个字典赋给变量,而是将右侧表达式的最终值(即 `7`)同时赋给左右两侧变量,并对比正确实现字典共享更新的写法。
在 Python 中,链式赋值语句 a = b = c 并非“先计算 c,再把 c 赋给 b,最后把 b 赋给 a”,而是等价于 从右向左依次执行赋值,且所有左侧目标都接收同一个右值(r-value)。关键在于:赋值表达式本身返回的是被赋的值,而非被赋值的对象。
以问题中的第一段代码为例:
starting_dictionary = {"a": 9, "b": 8}
final_dictionary = {"a": 9, "b": 8, "c": 7} # 此行创建的字典立即被后续覆盖,无实际作用
final_dictionary = starting_dictionary["c"] = 7该链式赋值 final_dictionary = starting_dictionary["c"] = 7 实际按如下顺序执行:
- 计算最右侧的 7(字面量);
- 执行 starting_dictionary["c"] = 7 → 修改原字典,添加键 "c",值为 7;该赋值表达式返回值为 7(Python 中所有赋值表达式均返回右值);
- 执行 final_dictionary = 7 → 将整数 7 赋给 final_dictionary,而非字典对象。
因此,运行后:
立即学习“Python免费学习笔记(深入)”;
- starting_dictionary 变为 {"a": 9, "b": 8, "c": 7} ✅(被成功修改)
- final_dictionary 是整数 7 ❌(不再是字典,更不是 starting_dictionary 的引用)
? 验证:print(type(final_dictionary)) 输出 ,print(final_dictionary["c"]) 会抛出 TypeError: 'int' object is not subscriptable。
而第二段代码是正确的做法:
starting_dictionary = {"a": 9, "b": 8}
# ...(此处初始化 final_dictionary 的代码实际未被使用,可省略)
starting_dictionary["c"] = 7 # 显式修改字典
final_dictionary = starting_dictionary # 将字典对象引用赋给 final_dictionary此时 final_dictionary 和 starting_dictionary 指向同一字典对象(内存地址相同),二者完全同步:
print(starting_dictionary) # {'a': 9, 'b': 8, 'c': 7}
print(final_dictionary) # {'a': 9, 'b': 8, 'c': 7}
print(starting_dictionary is final_dictionary) # True✅ 正确理解要点:
- 字典是可变对象,赋值操作(=)传递的是对象引用,而非拷贝;
- 链式赋值 a = b = expr 中,expr 只被求值一次,其结果被赋给所有左侧名称;
- 若想让 final_dictionary 成为 starting_dictionary 的别名,应先完成字典修改,再单独赋值引用;
- 若需独立副本(避免后续相互影响),应显式拷贝:final_dictionary = starting_dictionary.copy() 或 dict(starting_dictionary)。
? 总结:不要依赖链式赋值来“同时更新并赋值对象”,它只适合对同一值进行多变量绑定(如 x = y = 0)。涉及容器对象(字典、列表等)的结构修改与引用传递时,务必拆分为清晰的两步:先修改,再赋引用。










