鱼C论坛

 找回密码
 立即注册
查看: 1107|回复: 13

[已解决]python中描述符的理解

[复制链接]
发表于 2019-10-11 18:35:55 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 zltzlt 于 2019-10-11 18:51 编辑

1.代码是很基础的,自定义一个描述符中执行过程不理解,望高手解答!!!

class MyProperty:
    def __init__(self,fget=None,fset=None,fdel=None):
        self.fget = fget
        self.fset = fset
        self.fdel = fdel

    def __get__(self,instance,owner):
        return self.fget(instance)
    def __set__(self,instance,value):
        self.fset(instance,value)            #对这个地方不理解
    def __delete__(self,instance):
        self.fdel(instance)


class C:
    def __init__(self):
        self._x = None

    def getX(self):
        return self._x
    def setX(self,value):
        self._x = value
    def delX(self):
        del self._x

    x=MyProperty(getX,setX,delX)

3.执行:

c=C()
c.x='X-man'
3.问题描述:

在执行c.x='X-man'后,此时self.fset = setX,则self.fset(instance,value)实际等同于setX(c,'X-man'),这样也能运行吗?不是应该是c.setX('X-man')吗?

4.问题补充,自己根据理解写的例子

class A:
    def qiu2(self,y):
        print('调用了qiu2')
        return  y*y
    def qiu1(self,x):
        print(x(self,5))         ##此处为疑惑点
执行

a=A()
a.qiu1(qiu2)
Traceback (most recent call last):
File "", line 1, in
a.qiu1(qiu2)
NameError: name 'qiu2' is not defined

#####按照描述符的类似的办法,x = qiu2 , 那x(self,5)应该也能调用成功的,可是会报错。然后将疑惑点改成print(self.x(5))运行仍然会出错,错误仍然是'qiu2'没有定义。为什么之前的代码能实现,而此处行不通呢?困扰几天了,希望大神能解答!!!!!!真是为了这个问题烦啊
最佳答案
2019-10-11 18:56:43
1. self.fset(instance, value) 表示调用自身 fset。在这里 fset 是 C.setX,C.setX 有两个参数 self 和 value,在这里给 C.setX 传递两个参数 instance(self 参数,是对象的实例))和 value。

2. 是相等的。c.setX('X-man') 会隐式地给 setX 传递 self 参数为 c。

3. 全局变量 qiu2 根本没定义,你是想这样吗:

  1. class A:
  2.     def qiu2(self,y):
  3.         print('调用了qiu2')
  4.         return  y*y
  5.     def qiu1(self,x):
  6.         print(x(self,5))

  7. a = A()
  8. a.qiu1(A.qiu2)
复制代码
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2019-10-11 18:56:43 | 显示全部楼层    本楼为最佳答案   
1. self.fset(instance, value) 表示调用自身 fset。在这里 fset 是 C.setX,C.setX 有两个参数 self 和 value,在这里给 C.setX 传递两个参数 instance(self 参数,是对象的实例))和 value。

2. 是相等的。c.setX('X-man') 会隐式地给 setX 传递 self 参数为 c。

3. 全局变量 qiu2 根本没定义,你是想这样吗:

  1. class A:
  2.     def qiu2(self,y):
  3.         print('调用了qiu2')
  4.         return  y*y
  5.     def qiu1(self,x):
  6.         print(x(self,5))

  7. a = A()
  8. a.qiu1(A.qiu2)
复制代码
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 1 反对 0

使用道具 举报

 楼主| 发表于 2019-10-11 19:43:03 | 显示全部楼层
zltzlt 发表于 2019-10-11 18:56
1. self.fset(instance, value) 表示调用自身 fset。在这里 fset 是 C.setX,C.setX 有两个参数 self 和 va ...

我是想和描述符一样,用qiu1调用qiu2
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-10-11 19:44:15 | 显示全部楼层
wangqiuqiu 发表于 2019-10-11 19:43
我是想和描述符一样,用qiu1调用qiu2

?全局变量 qiu2 都没定义,调用当然报错
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-10-11 19:47:11 | 显示全部楼层
zltzlt 发表于 2019-10-11 18:56
1. self.fset(instance, value) 表示调用自身 fset。在这里 fset 是 C.setX,C.setX 有两个参数 self 和 va ...


谢谢你的耐心解答,有个疑问,通过 x=MyProperty(getX,setX,delX)这条语句,针对您提的第一条,fset这里应该是setX,为什么是类的方法C.setX
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-10-11 19:47:59 | 显示全部楼层
wangqiuqiu 发表于 2019-10-11 19:47
谢谢你的耐心解答,有个疑问,通过 x=MyProperty(getX,setX,delX)这条语句,针对您提的第一条,fset这里 ...

setX 就是 C.setX 啊
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-10-11 19:49:28 | 显示全部楼层
zltzlt 发表于 2019-10-11 19:44
?全局变量 qiu2 都没定义,调用当然报错

qiu2是定义的方法名呀,和 x=MyProperty(getX,setX,delX)中方法名getX,setX,delX是一样的呀
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-10-11 19:50:46 | 显示全部楼层
wangqiuqiu 发表于 2019-10-11 19:49
qiu2是定义的方法名呀,和 x=MyProperty(getX,setX,delX)中方法名getX,setX,delX是一样的呀

你在全局用了 qiu2 变量但没定义。x=MyProperty(getX,setX,delX) 只是凑巧参数名和方法名一样罢了
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-10-11 19:51:47 | 显示全部楼层
zltzlt 发表于 2019-10-11 19:47
setX 就是 C.setX 啊

嗯?不认同,那为什么不直接写C.setX呢?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-10-11 19:52:58 | 显示全部楼层
wangqiuqiu 发表于 2019-10-11 19:51
嗯?不认同,那为什么不直接写C.setX呢?

恰好重名而已
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-10-11 20:07:45 | 显示全部楼层
zltzlt 发表于 2019-10-11 19:44
?全局变量 qiu2 都没定义,调用当然报错

这个想明白了!!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-10-11 20:11:32 | 显示全部楼层
wangqiuqiu 发表于 2019-10-11 20:07
这个想明白了!!

还有一个疑问,两段代码:
1.全局变量能被内部定义函数所访问:
c=10
def glass(x):
return x*c
>>> glass(5) #能访问全局变量
50
2.实例对象中的方法直接访问变量:
class A:
c=10
def glass(self,x):
return x*c #除非改成return x*self.c
>>> a=A()
>>> a.glass(5)
Traceback (most recent call last):
File "<pyshell#1>", line 1, in <module>
a.glass(5)
File "F:/new/new world/ex45/delete.all1.py", line 4, in glass
return x*c
NameError: name 'c' is not defined
>>>
为什么python需要做这样的处理呢?我之前理解所谓类无非是变量和函数的集合,方便编程。而从以上两段程序可以看出,类中的静态变量无法在方法中直接访问,而全局变量能在整个文件中都能访问,不太理解这种差异。望大神解答。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-10-11 20:13:12 | 显示全部楼层
wangqiuqiu 发表于 2019-10-11 20:11
还有一个疑问,两段代码:
1.全局变量能被内部定义函数所访问:
c=10

类的变量就是类的变量,可以通过 类名.属性名 或 实例名.属性名 访问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-10-12 21:21:59 | 显示全部楼层
zltzlt 发表于 2019-10-11 19:47
setX 就是 C.setX 啊

还是觉得不对,setX为什么就能认为是C.setX呢?如果按照这个逻辑,那我之前写的那个qiu2也可以认为是C.qiu2呀。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-5-22 01:17

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表