我试图了解他们在self
和cls
之间的差异,但是我很挣扎,即使它存在很多话题。例如:
class maclass():
A = "class method"
def __init__(self):
self.B = "instance method"
def getA_s(self):
print(self.A)
def getA_c(cls):
print(cls.A)
def getB_s(self):
print(self.B)
def getB_c(cls):
print(cls.B)
C = maclass()
C.getA_s()
C.getA_c()
C.getB_s()
C.getB_c()
给我:
class method
class method
instance method
instance method
所以无论我使用self
还是cls
它都会引用相同的变量。当我在self.A
中添加Init__
时,cls.A就会被替换
def __init__(self):
self.B = "instance method"
self.A = "new instance method"
我得到了:
new instance method
new instance method
instance method
instance method
如果他们是相同的话,我不明白有两种方式来打电话给班级成员吗?我知道这是这个论坛上的一个常见问题,但我真的不明白为什么我们可以使用不同的词来指代同一个词(我们甚至可以使用任何单词而不是self
或{{ 1}})
在以下情况中:
cls
我明白了:
class maclass():
A = "class method, "
def __init__(self):
self.A = "instance method, "
def getA_s(self):
print(self.A) #give me "instance method, "
@classmethod
def getA_c(cls):
print(cls.A) #give me "class method, "
C = maclass()
C.getA_s()
C.getA_c()
print(' ')
print(C.A) #give me "instance method, "
因此,在这种情况下,instance method,
class method,
instance method,
maclass
或cls.A
不会引用相同的变量。
答案 0 :(得分:14)
所有您的方法都是实例方法。它们都不是类方法。
方法的第一个参数仅按惯例命名为self
。您可以将其命名为任何名称,并将其命名为cls
,而不是将其命名为类。第一个参数绑定到实例是由于方法查找的工作方式(访问C.getA_s
生成绑定方法对象,并且调用该对象导致C
传递到原始函数getA_s
),参数的名称不起作用。
在您的方法中,您只是引用实例属性。 A
属性最终只在类上定义并不重要,您仍然通过C.A
访问该属性(其中C
是您创建的实例),而不是{{1 }}。如果没有实例属性遮蔽它,则在实例上查找属性也会找到在类上定义的属性。
要使方法成为类方法,请使用@classmethod
decorator:
maclass.A
现在@classmethod
def getA_c(cls):
print(cls.A)
将始终是对类的引用,而不是对实例的引用。我需要再次强调,Python为我的第一个参数选择了什么名称并不重要,但是cls
是这里的约定,因为它更容易提醒读者这个方法绑定到类宾语。
请注意,如果您对cls
方法执行此操作,则尝试访问方法中的getB_c()
将失败,因为cls.B
类上没有B
属性对象
那是因为maclass
将函数包装在descriptor object中,该函数会覆盖正常的函数绑定行为。描述符 protocol 导致方法在作为实例上的属性访问时绑定到实例,classmethod
对象重定向该绑定过程。
这是一个带内联注释的简短演示,我使用Python转换来命名类(使用CamelCase),以及实例,属性,函数和方法(使用snake_case):
classmethod
请注意,即使我们在实例上设置了一个具有相同名称的属性,class方法也会访问class属性。
接下来是绑定行为:
>>> class MyClass():
... class_attribute = "String attribute on the class"
... def __init__(self):
... self.instance_attribute = "String attribute on the instance"
... @classmethod
... def get_class_attribute(cls):
... return cls.class_attribute
... def get_instance_attribute(self):
... return self.instance_attribute
... @classmethod
... def get_instance_attribute_on_class(cls):
... return cls.instance_attribute
...
>>> instance = MyClass()
>>> instance.class_attribute # class attributes are visible on the instance
'String attribute on the class'
>>> MyClass.class_attribute # class attributes are also visible on the class
'String attribute on the class'
>>> instance.get_class_attribute() # bound to the class, but that doesn't matter here
'String attribute on the class'
>>> instance.class_attribute = "String attribute value overriding the class attribute"
>>> instance.get_class_attribute() # bound to the class, so the class attribute is found
'String attribute on the class'
>>> MyClass.get_instance_attribute_on_class() # fails, there is instance_attribute on the class
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 12, in get_instance_attribute_on_class
AttributeError: type object 'MyClass' has no attribute 'instance_attribute'
绑定方法告诉您它们绑定到到的内容,调用该方法将该绑定对象作为第一个参数传递。通过查看绑定方法的>>> MyClass.get_instance_attribute # accessing the method on the class gives you the function
<function MyClass.get_instance_attribute at 0x10f94f268>
>>> instance.get_instance_attribute # accessing the method on the instance gives you the bound method
<bound method MyClass.get_instance_attribute of <__main__.MyClass object at 0x10f92b5f8>>
>>> MyClass.get_class_attribute # class methods are always bound, to the class
<bound method MyClass.get_class_attribute of <class '__main__.MyClass'>>
>>> instance.get_class_attribute # class methods are always bound, to the class
<bound method MyClass.get_class_attribute of <class '__main__.MyClass'>>
属性,也可以对该对象进行反省:
__self__