为什么这个程序不会打印m.x和m1.x的值?
class RevealAccess:def __init__(self,initval = None,name='var'):
self.val = initval
self.name = name
def __get__(self, instance, owner):
print("获取..",self.name)
return self.val
def __set__(self, instance, value):
print("设置值:",self.name)
self.val = value
class MyClass:
x = RevealAccess(10,"var 'x'")
y = 5
def __init__(self):
self.x = 'self x'
m = MyClass()
m.x
print("-----------------")
m1 = MyClass()
m1.x = 20
m1.x
程序执行结果:
设置值: var 'x'
获取.. var 'x'
-----------------
设置值: var 'x'
设置值: var 'x'
获取.. var 'x' 在shell里输入变量会回显变量,如果是跑程序,得print。
m = MyClass()
print(m.x)
print("-----------------")
m1 = MyClass()
m1.x = 20
print(m1.x) suchocolate 发表于 2021-8-14 15:29
在shell里输入变量会回显变量,如果是跑程序,得print。
OK,那还有第二个m1.x为什么会有两个设置值? 本帖最后由 suchocolate 于 2021-8-14 15:45 编辑
飞花落尽 发表于 2021-8-14 15:32
OK,那还有第二个m1.x为什么会有两个设置值?
初始对象时,为类变量x赋值调用一次描述符__set__,m1.x=20又调用一次__set__。 suchocolate 发表于 2021-8-14 15:44
初始对象时,为类变量x赋值调用一次描述符__set__,m1.x=20又调用一次__set__。
那实例化对象的赋值‘self x’不用调用吗? suchocolate 发表于 2021-8-14 15:44
初始对象时,为类变量x赋值调用一次描述符__set__,m1.x=20又调用一次__set__。
class RevealAccess:
def __init__(self,initval = None,name='var'):
self.val = initval
self.name = name
def __get__(self, instance, owner):
print("获取..",self.name)
return self.val
def __set__(self, instance, value):
print("设置值:",self.name)
self.val = value
def __delete__(self,instance):
del self.val
class MyClass:
x = RevealAccess(10,"var 'x'")
y = 5
def __init__(self):
self.x = 'self x'
m = MyClass()
print(m.x)
print("-----------------")
m1 = MyClass()
m1.x = 20
del m1.x
print(m1.x)
还有我删除实例化对象的x属性,类属性x是否会被保留?(运行结果AttributeError: 'RevealAccess' object has no attribute 'val')但我记得类属性好像不会被删除啊,希望大佬解释一下 飞花落尽 发表于 2021-8-14 16:12
那实例化对象的赋值‘self x’不用调用吗?
调用 suchocolate 发表于 2021-8-14 16:25
调用
那怎么不打印语句? 飞花落尽 发表于 2021-8-14 16:17
class RevealAccess:
def __init__(self,initval = None,name='var'):
self.val = initval ...
1.不是同一个东西,不影响。
2.可以删除,想不被删除用__开头。 本帖最后由 suchocolate 于 2021-8-14 16:57 编辑
飞花落尽 发表于 2021-8-14 16:31
那怎么不打印语句?
class RevealAccess:
def __init__(self,initval = None,name='var'): # 初始化运行,有设置值,自动调用__set__,会打印。
self.val = initval
self.name = name
def __get__(self, instance, owner):
print("获取..",self.name)
return self.val
def __set__(self, instance, value): # 对象变量被设置时自动调用,会打印。
print("设置值:",self.name)
self.val = value
def __delete__(self,instance):
del self.val
class MyClass:
x = RevealAccess(10,"var 'x'")# 初始化实例时运行,调用描述符类。
y = 5
def __init__(self): # 初始化实例时运行,没有打印的东西。
self.x = 'self x' suchocolate 发表于 2021-8-14 16:40
1.不是同一个东西,不影响。
2.可以删除,想不被删除用__开头。
我的意思是我用del这个函数删除的是类属性的x还是实例化属性的x?
抱歉,上面可能表达得不是很清楚 飞花落尽 发表于 2021-8-14 19:54
我的意思是我用del这个函数删除的是类属性的x还是实例化属性的x?
抱歉,上面可能表达得不是很清楚
del MyClass.x# 删除类属性
del m1.x# 删除实例属性 suchocolate 发表于 2021-8-14 20:09
del MyClass.x# 删除类属性
del m1.x# 删除实例属性
那删除了实例属性不也可以通过m1.x访问类属性吗?但是上面的结果报错了是为什么呢? 本帖最后由 suchocolate 于 2021-8-14 21:50 编辑
飞花落尽 发表于 2021-8-14 21:42
那删除了实例属性不也可以通过m1.x访问类属性吗?但是上面的结果报错了是为什么呢?
MyClass.x 类属性
m1.x 实例属性 suchocolate 发表于 2021-8-14 21:47
MyClass.x 类属性
m1.x 实例属性
啊这
举个例子吧
class Turtle:
color = 'green'
def __init__(self):
self.color = "black"
运行:
t= Turtle()
>>> t.color
'black'
>>> del t.color
>>> t.color
'green' suchocolate 发表于 2021-8-14 21:47
MyClass.x 类属性
m1.x 实例属性
这样就不会报错啊 飞花落尽 发表于 2021-8-14 22:01
这样就不会报错啊
print和shell是不同的,__str__ 控制printi行为 , __repr__控制shell回显行为,这两个函数不同,想避免报错就自定义这两个函数,不用默认的行为。
其次对象变量尽量不要和类变量同名。
suchocolate 发表于 2021-8-15 11:20
print和shell是不同的,__str__ 控制printi行为 , __repr__控制shell回显行为,这两个函数不同,想避免 ...
好的,谢谢
页:
[1]