【Python】python 子类的类属性疑惑?

本人刚刚接触python不久, 在学习类继承的时候遇到了一点疑惑, 如下:

class A(object):

x = 1

class B(A):

B.x = 3

print(A.x) # 1

print(B.x) # 1

print(id(A.x)) # 4363216224

print(id(B.x)) # 4363216224

在子类中试图修改这个类属性失败

但是如果我这样改:

class A(object):

x = 1

class B(A):

# B.x = 3

x = 3

print(A.x) # 1

print(B.x) # 3

print(id(A.x)) # 4363216224

print(id(B.x)) # 4363216256

父类和子类中的类属性x就不一样了, 个人的疑惑如下:

  • 在第一种情况中, 似乎无法修改子类的类属性, 请问这是什么原因?
  • 第二种情况, 似乎是在B中重新创造了一个类属性x, 不知道是不是这么理解?

总体感觉很混淆, 没有一个清晰的概念来理清这两段代码, 希望能有各位前辈能指点一下, 不胜感激!



更新:

感谢各位前辈的回答, 我用的是3.6, 不过当时是用的jupyter notebook编辑代码的, 运行是没问题, 但是刚刚自己在别的ide上面试了一下, 确实会报错, 个人也很迷糊, 不知道是什么情况.

jupyter notebook截图如下:

【Python】python 子类的类属性疑惑?

所以现在我也很不理解, 有没有别的前辈出现过这样的情况?

回答

1: 这是因为这个是类属性,类属性可以通过类或者实例来访问,当你创建一个类的实例的时候,所有的类属性都会复制一份给这个实例,假设有
class A:

x = 1

a = A()
b = A()
此时a,b都从类那里得到了类属性x, 用实例也就是a, b都可以访问x, 也可以去修改它,但是它们互不影响
a.x = 2
b.x = 3
此时打印a.x 为2 b.x 为3,A.x 为 1
总之类属性要通过类来修改,创建的实例都有类属性的一份复制。

2:这里是你重写了类的属性,你在子类当中的修改不能影响到父类,要用父类来操作,不然就乱套了。

请问你的 Python 版本是什么,我尝试了使用 2.73.6 版本的 Python 执行你这段代码,但是都出现了错误:

Traceback (most recent call last):

File "Test.py", line 7, in <module>

class B(A):

File "Test.py", line 8, in B

B.x = 3

NameError: name 'B' is not defined

我这边python报错.
2.7

Traceback (most recent call last):

File "C:/work/python_proj/Odoo-erp_test/Src/ww.py", line 5, in <module>

class B(A):

File "C:/work/python_proj/Odoo-erp_test/Src/ww.py", line 6, in B

B.x = 3

NameError: name 'B' is not defined

==============================
更新
我觉得B.x = 3这句话的意思python解释器会理解为 B.B.x = 3 所以报错.

你的B.x = 3一般都会报错,可能你的Python版本比较特殊,我个人理解是在B类里创建了一个叫B.x的属性,与A中的x已经没有关系了,所以直接继承A中的x属性。第二种写法是重载了x属性,使得BAx属性不同。

以上是 【Python】python 子类的类属性疑惑? 的全部内容, 来源链接: utcz.com/a/78931.html

回到顶部