Victor0321 发表于 2022-4-19 19:23:43

关于类属性和实例属性

各位大神,最近学到类这一块有点晕,概念性的东西有点转不太过弯来。下面三个例子,有三个问题:
1. 例子一和例子三对比: 例子三报错的原因是因为构造函数里暂时先给了形参x, 但是这个形参还需要与self 实例赋值,且self.x 必须在左边??

2. 例子一和例子二对比:为什么o集合前面必须要加Stack. ,这个o 不是已经是类属性了么。。。 还是因为我理解这一块有点类似闭包的nonlocal机制,在__init__里面需要修改类属性也必须要在类属性变量名前面加呼叫器类.

3. 总上,我的理解就是常见的大部分用得着的都是实例属性,这个类属性如果在方法里使用或者修改都需要加类. 去呼叫它,而实例属性用起来更加的容易修改也便于理解也跟实例有着类似一一对应的关系(可以重复)?

不知道这个理解对不对。。。

例子一:
class Stack:
    o = []
    def __init__(self,x):
      self.x = x
      Stack.o.append(self.x)

a = Stack('f')
b = Stack('u')
print(b.o)

例子二:
class Stack:
    o = []
    def __init__(self,x):
      self.x = x
      o.append(self.x)

a = Stack('f')
b = Stack('u')
print(b.o)

例子三:
class Stack:
    o = []
    def __init__(self,x):
      o.append(self.x)

a = Stack('f')
b = Stack('u')
print(b.o)

Stubborn 发表于 2022-4-19 19:44:14

本帖最后由 Stubborn 于 2022-4-19 19:52 编辑

在类中,self其实就和实例化后的变量一样。
class Stack:
    ....
    sefl.xx = xx
st = Stack()
st.xx

问题一。例三之所以报错,就像你没有声明一个变量,却要去用这个变量一样
问题二。在类里面,没有带self,类似于,共享一样的功能了(主要是容器类型数据),类,以及实例化类的操作都会同步
问题三,不是很理解
class Stack:
    o = []
    def __init__(self,x):
      self.x = x
      Stack.o.append(self.x)

if __name__ == '__main__':
    Stack.o.append(5)
    s = Stack(3)
    print(s.o)
    Stack.o.append(6)
    print(s.o)

Victor0321 发表于 2022-4-19 23:38:07

Stubborn 发表于 2022-4-19 19:44
在类中,self其实就和实例化后的变量一样。




emm其实我的意思就是

class Stack:
    o = []
    def __init__(self,x):
      self.x = x
      Stack.o.append(self.x)

if __name__ == '__main__':
    Stack.o.append(5) ##这个地方如果要相对类属性. 0 修改不能直接写o 需要Stack.o 对吗。
    s = Stack(3)
    print(s.o)
    Stack.o.append(6)
    print(s.o)

Stubborn 发表于 2022-4-20 00:03:55

Victor0321 发表于 2022-4-19 23:38
emm其实我的意思就是

class Stack:


类属性就相当于“母”空间,它的实例都会相互影响,共享的,针对容器类

Victor0321 发表于 2022-4-20 00:23:28

Stubborn 发表于 2022-4-20 00:03
类属性就相当于“母”空间,它的实例都会相互影响,共享的,针对容器类

明白了! 确认一下,就是如果下方这个例子如果里面的C.count += 1 都换成self.count += 1 那么它就变成了实例属性,这样它就不会共享了,是专属于实例的count ,a = C(), b=C() , c = C()这样的 abc.count都等于1 对吧!!!
class C:
      count = 0
      
      def __init__(self):
                C.count += 1

      def __del__(self):
                C.count -= 1

>>> a = C()
>>> b = C()
>>> c = C()

Victor0321 发表于 2022-4-20 00:27:10

Stubborn 发表于 2022-4-20 00:03
类属性就相当于“母”空间,它的实例都会相互影响,共享的,针对容器类

可是 我一开始还以为self.count += 1 随着实例增多它这个是递增的。。。 而且每个实例对应着不同的count...现在看来count 的确递增 但是所有实例对应的count都是一样的值

Stubborn 发表于 2022-4-20 11:35:18

Victor0321 发表于 2022-4-20 00:23
明白了! 确认一下,就是如果下方这个例子如果里面的C.count += 1 都换成self.count += 1 那么它就变成了 ...

对,你把self看成实例化后的变量名字就对了。self是约定成俗的写法
页: [1]
查看完整版本: 关于类属性和实例属性