鱼C论坛

 找回密码
 立即注册
查看: 1732|回复: 0

[技术交流] 《零基础入门学习Python》第046讲笔记:魔法方法:描述符

[复制链接]
发表于 2017-8-13 12:53:48 | 显示全部楼层 |阅读模式

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

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

x
描述符
  1. __get__(self, instance, owner)
  2. __set__(self, instance, value)
  3. __delete__(self, instance)
复制代码


MyDescriptor 就是描述符类
  1. class MyDescriptor:
  2.         def __get__(self, instance, owner):
  3.                 print("getting...", self, instance, owner)
  4.                
  5.         def __set__(self, instance, value):
  6.                 print("setting...", self, instance, value)
  7.                
  8.         def __delete__(self, instance):
  9.                 print("deleting...", self, instance)

  10. class Test:
  11.         x = MyDescriptor()
复制代码


MyProperty
  1. class MyProperty:
  2.         def __init__(self, fget=None, fset=None, fdel=None):
  3.                 self.fget = fget
  4.                 self.fset = fset
  5.                 self.fdel = fdel
  6.                
  7.         def __get__(self, instance, owner):
  8.                 return self.fget(instance)
  9.        
  10.         def __set__(self, instance, value):
  11.                 self.fset(instance, value)
  12.                
  13.         def __delete__(self, instance):
  14.                 self.fdel(instance)

  15. class C:
  16.         def __init__(self):
  17.                 self._x = None
  18.                
  19.         def getX(self):
  20.                 return self._x
  21.        
  22.         def setX(self, value):
  23.                 self._x = value
  24.                
  25.         def delX(self):
  26.                 del self._x
  27.                
  28.         x = MyProperty(getX, setX, delX)
复制代码


练习:定义一个温度类,定义两个描述符描述摄氏度和华氏度两个属性。要求:两个属性会自动转化。
  1. # 摄氏度 * 1.8 + 32 = 华氏度
  2. class Celsius:
  3.     def __init__(self, value = 26.0):
  4.         self.value = float(value)
  5.    
  6.     def __get__(self, instance, owner):
  7.         print('__get__被调用了!')
  8.         return self.value
  9.    
  10.     def __set__(self, instance, value):
  11.         print('__set__被调用了!')
  12.         self.value = float(value)

  13. class Fahrenheit:
  14.     def __get__(self, instance, owner):
  15.         return instance.cel * 1.8 + 32
  16.    
  17.     def __set__(self, instance, value):
  18.         instance.cel = (float(value) - 32) / 1.8
  19.         # 这里的instance就等于Temperature
  20. class Temperature:
  21.     cel = Celsius()
  22.     fah = Fahrenheit()
复制代码


效果:
  1. >>> temp = Temperature()

  2. >>> temp.cel
  3. 26.0
  4. >>> temp.fah
  5. 78.80000000000001

  6. >>> temp.fah = 100
  7. >>> temp.cel
  8. 37.77777777777778
复制代码


动动手
0
  1. class MyDis:
  2.         def __init__(self, initval = None, name = None):
  3.                 self.val = initval
  4.                 self.name = name

  5.         def __get__(self, instance, owner):
  6.                 print("正在获取变量:" self.name)
  7.                 return self.val

  8.         def __set__(self, instance, value):
  9.                 print("正在修改变量:" self.name)
  10.                 self.val = value

  11.         def __delete__(self, instance):
  12.                 print("正在删除变量:" self.name)
  13.                 print("这个变量无法删除~")
复制代码


1
  1. import time

  2. class Record:
  3.         def __init__(self, initval = None, name = None):
  4.                 self.val = initval
  5.                 self.name = name
  6.                 self.filename = "record.txt"

  7.         def __get__(self, instance, owner):
  8.                 with open(self.filename, 'a', encoding = 'utf-8') as f:
  9.                         f.write("%s 变量于北京时间 %s 被读取,%s = %s\n" % \
  10.                                 (self.name, time.ctime(), self.name, str(self.val)))
  11.                 return self.val
  12.        
  13.         def __set__(self, instance, value):
  14.                 filename = "%s_record.txt" % self.name
  15.                 with open(self.filename, encoding = 'utf-8') as f:
  16.                         f.write("%s 变量于北京时间 %s 被修改,%s = %s\n" % \
  17.                                 (self.name, time.ctime(), self.name, str(self.val)))
  18.                 self.val = value
复制代码


2
  1. import os
  2. import pickle

  3. class MyDes:
  4.         saved = []

  5.         def __init__(self, name = None):
  6.                 self.name = name
  7.                 self.filename = self.name + '.pkl'

  8.         def __get__(self, instance, owner):
  9.                 if self.name not in MyDes.saved:
  10.                         raise AttributeError("%s 属性还没有赋值" % self.name)

  11.                 with open (self.filename, 'rb') as f:
  12.                         value = pickle.load(f)

  13.                 return value

  14.         def __set__(self, instance, value):
  15.                 with open(self.filename, 'wb') as f:
  16.                         pickle.dump(value, f)
  17.                         MyDes.saved.append(self.name)

  18.         def __delete__(self, instance):
  19.                 os.remove(self.filename)
  20.                 MyDes.saved.remove(self.name)
复制代码


实现效果:及时生成x.pkl, y.pkl,并del时自动删除
  1. >>> class Test:
  2.         x = MyDes('x')
  3.         y = MyDes('y')
  4.         
  5. >>> test = Test()
  6. >>> test.x = 123
  7. >>> test.y = "I love FishC.com!"
  8. >>> test.x
  9. 123
  10. >>> test.y
  11. 'I love FishC.com!'
复制代码

评分

参与人数 1鱼币 +1 收起 理由
小甲鱼 + 1

查看全部评分

本帖被以下淘专辑推荐:

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

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

本版积分规则

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

GMT+8, 2024-4-28 11:57

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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