类属性在子类中被意外共享的几种典型写法错误
技术百科
冰川箭仙
发布时间:2026-01-20
浏览: 次 类属性被意外共享的本质是混淆类属性与实例属性的作用域;可变类属性(如list、dict)导致子类实例共享同一对象;子类动态修改父类属性易造成逻辑割裂;类方法硬编码父类名会绕过子类隔离;元类或装饰器注入类属性时未做子类隔离处理亦引发共享问题。
类属性在子类中被意外共享,本质是混淆了“类属性”和“实例属性”的作用域与生命周期。Python 中类属性属于类对象本身,所有实例(包括子类实例)默认共享同一份内存地址——除非显式覆盖或重新赋值。下面几种写法最容易导致隐蔽的共享问题。
用可变对象(如 list、dict)直接定义类属性
这是最常见也最危险的错误。类属性若为可变对象,子类未重定义时会沿用父类的引用,所有子类实例操作的其实是同一个对象。
-
错误示例:class Animal:
skills = [] # 可变类属性
class Dog(Animal): pass
class Cat(Animal): pass
Dog.skills.append("bark")
Cat.skills.append("meow")
print(Dog.skills) # ['bark', 'meow'] ← 意外共享! -
正确做法:把可变数据移到
__init__中作为实例属性;或在类内用None占位,首次访问时惰性初始化。
子类未重写类属性,却在运行时动态修改父类属性
子类没有定义同名类属性时,对 SubClass.attr 的读取会回溯到父类;但若执行 SubClass.attr = ...,则会在子类命名空间创建新属性——看似隔离,实则容易误判。
-
典型陷阱:开发者以为
Dog.name = "Dog"是给子类设名,但后续若又写Animal.name = "Animal",Dog 的name不受影响;可如果只改Dog.name而忘了其他子类,逻辑就割裂了。 - 建议:明确区分配置类属性(应设计为只读常量)和状态类属性(应避免,改用实例或单例管理)。
使用类方法修改类属性,却忽略子类继承链
类方法(@classmethod)中的 cls 指向实际调用者类,但如果类方法内部硬编码了父类名来修改属性,就会绕过子类隔离意图。
-
错误示例:class Base:
count = 0
@classmethod
def inc(cls):
Base.count += 1 # 硬编码父类名 → 所有子类共用 Base.count -
修复方式:用
cls.count += 1,并确保子类定义了自己的count(否则仍回溯);更稳妥的是在子类中显式初始化:count = 0。
元类或装饰器自动注入类属性,未做子类隔离处理
当通过元类、类装饰器批量添加属性(如注册表、缓存字典)时,若未按 cls 分别绑定,极易让所

-
风险场景:装饰器为每个类添加
_handlers = {},但实现时写成klass._handlers = global_dict,而非klass._handlers = {}。 -
检查要点:所有自动注入的类属性,必须保证每次赋值都是新对象(
[]、{}、set()),且不复用外部变量。
# 是在
# 就会
# 自己的
# 都是
# 这是
# python
# 首次
# app
# 注册表
# 对象
# class
# 编码
# 子类
# 类中
# 继承
# 作用域
# 命名空间
# count
# 常量
# 父类
# print
# append
# 类属
# 未做
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- php中常量能用::访问吗_类常量与作用域操作符使
- 如何使用Golang处理静态文件缓存_提高页面加载
- Windows 11如何查看系统激活密钥_Wind
- Win11怎么更改电脑名称_Windows 11修
- 如何使用Golang匿名函数_快速定义临时函数逻辑
- Win11怎么设置单手模式_Win11触控键盘布局
- MAC如何隐藏文件夹及文件_MAC终端命令隐藏与第
- Win10如何更改用户账户控制_Windows10
- Win11麦克风没声音怎么设置_Win11麦克风权
- 如何在Golang中使用闭包_封装变量与函数作用域
- Win10如何卸载预装Edge扩展_Win10卸载
- Python如何创建带属性的XML节点
- win11 OneDrive怎么彻底关闭 Win1
- c++的mutex和lock_guard如何使用
- c++中的std::conjunction和std
- Windows执行文件被SmartScreen拦截
- 如何使用Golang实现基本类型比较_Golang
- c# 如何深拷贝和浅拷贝
- Mac电脑进水了怎么办_MacBook进水后紧急处
- 如何在JavaScript中动态拼接PHP的bas
- Mac的“调度中心”与“空间”怎么用_Mac多桌面
- Win11怎么开启窗口对齐助手_Windows11
- Win10如何卸载WindowsDefender_
- 如何在Golang中编写异步函数测试_Golang
- php485能和物联网模块通信吗_php485对接
- 如何在Mac上搭建Golang开发环境_使用Hom
- Win11怎么关闭定位服务_保护Win11位置隐私
- 如何使用Golang反射创建map对象_动态生成键
- Windows10电脑怎么设置自动连接WiFi_W
- c# 在高并发下使用反射发射(Reflection
- Drupal 中 HTML 链接被重复转义导致渲染
- 如何使用正则表达式精确匹配最多含一个换行符的 st
- VSC怎么配置PHP的Xdebug_远程调试设置步
- 如何使用Golang包导出规则_控制函数和变量可见
- Windows怎样拦截WPS弹窗广告_Window
- 如何在Golang中写入XML文件_生成符合规范的
- 用lighttpd能运行php吗_lighttpd
- Mac如何彻底清理浏览器缓存?(Safari与Ch
- php订单日志怎么记录评价_php记录订单评价日志
- Win11怎么压缩文件 Win11自带压缩解压功能
- Win11怎么关闭键盘按键音_Win11禁用打字声
- 如何在 Go 应用中实现自动错误恢复与进程重启机制
- 如何在 Go 中高效缓存与分发网络视频流
- php能跑在stm32上吗_php在stm32微控
- 短链接还原php提示内存不足_调整PHP内存限制设
- 如何使用正则表达式批量替换重复的“-”模式为固定字
- Mac自带的词典App怎么用_Mac添加和使用多语
- 如何使用Golang实现函数指针_函数变量与回调示
- php嵌入式多设备通信怎么实现_php同时管理多个
- SAX解析器是什么,它与DOM在处理大型XML文件

QQ客服