__getattr__    当实例要调用的属性不存在的时候会触发

__setattr__    当实例要设置属性的时候会触发

__delattr__    当实例要删除属性的时候会触发

这三个方法是python内置的方法,如果不自己定义的话,就是默认的,自己定义的话就是用自己定义的

1、__getattr__           当要调用的属性不存在的时候会触发

没有自己定义__getattr__的时候

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. f1=Foo("gouguoqi")
  5. print(f1.name)
  6.  
  7. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  8.  
  9. gouguoqi

我们自己定义一个__getattr__

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __getattr__(self, item):
  5. print("你要找的属性%s不存在" %item)
  6. f1=Foo("gouguoqi")
  7. print(f1.name)
  8. print(f1.age)
  9.  
  10. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  11.  
  12. gouguoqi
  13.  
  14. 你要找的属性age不存在

2、__setaddr__    给属性设置值的时候会触发

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __setattr__(self, k, v):
  5. print("执行setattr")
  6. f1=Foo("gouguoqi")#实例化的过程就是运行initinit就有一个赋值name=gouguoqi所以就触发setaddr
  7.  
  8. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  9.  
  10. 执行setattr

2.1我们在没有自己定义__setattr__的时候,系统是帮我们把属性设置到属性字典里面

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. f1=Foo("gouguoqi")#实例化的过程就是运行initinit就有一个赋值name=gouguoqi所以就触发setaddr
  5. print(f1.__dict__)
  6.  
  7. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  8.  
  9. {'name': 'gouguoqi'}

2.2但是我们自己定制的那个setattr只是执行了一个打印的操作,并没有帮我们把值新增到属性字典里面去

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __setattr__(self, k, v):
  5. print("执行setattr")
  6. f1=Foo("gouguoqi")#实例化的过程就是运行initinit就有一个赋值name=gouguoqi所以就触发setaddr
  7. f1.age=28
  8. f1.gender="male"
  9. print(f1.__dict__)
  10.  
  11. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  12.  
  13. 执行setattr
  14.  
  15. 执行setattr
  16.  
  17. 执行setattr
  18.  
  19. {} 

2.3现在有一个需求,传入的值必须是字符串,应该怎么实现

先完成以下判断逻辑,然后在具体的去实现功能

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __setattr__(self, k, v):
  5. print("执行setattr")
  6. if type(v) is str:
  7. print("开始设置")
  8. else:
  9. print("必须是字符串")
  10. f1=Foo("gouguoqi")
  11. f1.age=28
  12. print(f1.__dict__)
  13.  
  14. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  15.  
  16. 执行setattr
  17.  
  18. 开始设置
  19.  
  20. 执行setattr
  21.  
  22. 必须是字符串
  23.  
  24. {}

实现设置功能,只能是字符串才能设置进去

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __setattr__(self, k, v):
  5. print("执行setattr")
  6. if type(v) is str:
  7. print("开始设置")
  8. self.__dict__[k]=v
  9. else:
  10. print("必须是字符串")
  11. f1=Foo("gouguoqi")
  12. f1.age=28
  13. print(f1.__dict__)
  14.  
  15. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  16.  
  17. 执行setattr
  18.  
  19. 开始设置
  20.  
  21. 执行setattr
  22.  
  23. 必须是字符串
  24.  
  25. {'name': 'gouguoqi'}

2.4还可以实现。如果是字符串的话,把字符串变为大写

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __setattr__(self, k, v):
  5. print("执行setattr")
  6. if type(v) is str:
  7. print("开始设置")
  8. self.__dict__[k]=v.upper()
  9. else:
  10. print("必须是字符串")
  11. f1=Foo("gouguoqi")
  12. f1.age=28
  13. print(f1.__dict__)
  14.  
  15. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  16.  
  17. 执行setattr
  18.  
  19. 开始设置
  20.  
  21. 执行setattr
  22.  
  23. 必须是字符串
  24.  
  25. {'name': 'GOUGUOQI'}

3、__delattr__  删除属性

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __delattr__(self, item):
  5. print("执行delattr",item)
  6. self.__dict__.pop(item)#操作底层的属性字典的key
  7. f1=Foo("gouguoqi")
  8. del f1.name#把f1传给self,把name传给item
  9. print(f1.__dict__)
  10.  
  11. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  12.  
  13. 执行delattr name
  14.  
  15. {}

3.1那么系统默认的删除就是这样的啊,我们为什么要自己定义一个这个呢,那么又来一个新的需求,所有的属性都不能删除,改怎么做呢

  1. class Foo:
  2. def __init__(self,name):
  3. self.name=name
  4. def __delattr__(self, item):
  5. print("%s属性不允许删除" %item)
  6. f1=Foo("gouguoqi")
  7. del f1.name
  8. print(f1.__dict__)
  9.  
  10. C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
  11.  
  12. name属性不允许删除
  13.  
  14. {'name': 'gouguoqi'} 

4、包装

  1. class List(list):#定义个类,继承系统的list
  2. pass
  3.  
  4. l2=list("hello world")
  5. print(l2,type(l2))
  6.  
  7. l1=List("hello world")
  8. print(l1,type(l1))
  9.  
  10. C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py
  11.  
  12. ['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'] <class 'list'>
  13.  
  14. ['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'] <class '__main__.List'>

4.1现在想要实现的是 定制一个方法,显示出列表中间元素的那个元素内容

  1. class List(list):#定义个类,继承系统的list
  2. def show_middle(self):
  3. mid_index=int(len(self)/2)#取到中间的索引
  4. return self[mid_index]#返回列表的中间的索引的那个值
  5. l1=List("helloworld")
  6. print(l1,type(l1))
  7. print(l1.show_middle())
  8.  
  9. C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py
  10.  
  11. ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd'] <class '__main__.List'>
  12.  
  13. w

4.2我们自己定义一个apped的方法

结果就是append这个方法不能追加了。因为自己定义的是pass

  1. class List(list):#定义个类,继承系统的list
  2. def append(self, object):
  3. pass
  4. def show_middle(self):
  5. mid_index=int(len(self)/2)#取到中间的索引
  6. return self[mid_index]#返回列表的中间的索引的那个值
  7. l1=List("helloworld")
  8. l1.append(111)
  9. print(l1)
  10.  
  11. C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py
  12.  
  13. ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd']

实现真正的append操作

  1. class List(list):#定义个类,继承系统的list
  2. def append(self, object):
  3. if type(object) is str:
  4. list.append(self,object)#用父类的append方法
  5. def show_middle(self):
  6. mid_index=int(len(self)/2)#取到中间的索引
  7. return self[mid_index]#返回列表的中间的索引的那个值
  8. l1=List("helloworld")
  9. l1.append("")
  10. print(l1)
  11.  
  12. C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py
  13.  
  14. ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '']

类调用方法,推荐用super

  1. class List(list):#定义个类,继承系统的list
  2. def append(self, object):
  3. if type(object) is str:
  4. # list.append(self,object)#用父类的append方法
  5. super().append(object)#super就是调用父类的方法,不用传self
  6. def show_middle(self):
  7. mid_index=int(len(self)/2)#取到中间的索引
  8. return self[mid_index]#返回列表的中间的索引的那个值
  9. l1=List("helloworld")
  10. l1.append("")
  11. print(l1)
  12.  
  13. C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py
  14.  
  15. ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '']

4.3这样就实现了  append我们自己定制了一个,只能添加字符串,不能append数字

  1. class List(list):#定义个类,继承系统的list
  2. def append(self, object):
  3. if type(object) is str:
  4. # list.append(self,object)#用父类的append方法
  5. super().append(object)#super就是调用父类的方法,不用传self
  6. else:
  7. print("只能append字符串")
  8. def show_middle(self):
  9. mid_index=int(len(self)/2)#取到中间的索引
  10. return self[mid_index]#返回列表的中间的索引的那个值
  11. l1=List("helloworld")
  12. l1.append("")
  13. print(l1)
  14. l1.append(666)
  15.  
  16. C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py
  17.  
  18. ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '']

只能append字符串

继承的方式完成包装__attr__的更多相关文章

  1. javascript 中继承实现方式归纳

    转载自:http://sentsin.com/web/1109.html 不同于基于类的编程语言,如 C++ 和 Java,javascript 中的继承方式是基于原型的.同时由于 javascrip ...

  2. javascript继承(七)—用继承的方式实现照片墙功能

    照片墙DEMO下载 注意:图片有四种类型:1可放大:2可拖动:3既可放大也可拖动:4都不行.由于每个图片的构造函数不同而不同(目前在火狐上调试的,其它的浏览器可能不行,请见谅,主要讲继承的思想.以后会 ...

  3. JS类继承常用方式发展史

    JS类继承常用方式发展史 涉及知识点 构造函数方式继承 1-继承单个对象 1.1 多步走初始版 1.2 多步走优化版 1.3 Object.create()方式 2-继承多个对象 2.1 遍历 Obj ...

  4. Java基础 继承的方式创建多线程 / 线程模拟模拟火车站开启三个窗口售票

    继承的方式创建多线程 笔记: /**继承的方式创建多线程 * 线程的创建方法: * 1.创建一个继承于Thread 的子类 * 2.重写Thread类的run()方法 ,方法内实现此子线程 要完成的功 ...

  5. JavaScript实现继承的方式和各自的优缺点

    ECMAscript只支持实现继承,主要是依靠原型链来实现的. JavaScript实现继承的方式: 类式继承 构造函数继承 组合继承 寄生组合式继承 1.类式继承 //类式继承 //声明父类 fun ...

  6. JavaScript的继承实现方式

    1.使用call或apply方法,将父对象的构造函数绑定在子对象上 function A(){ this.name = 'json'; } function B(){ A.call(this); } ...

  7. js实现继承的方式总结

    js实现继承的5种方式 以下 均为 ES5 的写法: js是门灵活的语言,实现一种功能往往有多种做法,ECMAScript没有明确的继承机制,而是通过模仿实现的,根据js语言的本身的特性,js实现继承 ...

  8. Javascript学习笔记:6种实现继承的方式

    ①原型链式继承 function Person(name){ this.name=name; } Person.prototype.getName=function(){ return this.na ...

  9. javascript实现继承的方式

    this this表示当前对象,如果在全局作用范围内使用this,则指代当前页面对象window: 如果在函数中使用this,则this指代什么是根据运行时此函数在什么对象上被调用. 我们还可以使用a ...

随机推荐

  1. Linux内核分析— —操作系统是如何工作的(20135213林涵锦)

    mykernel实验指导(操作系统是如何工作的) 实验要求 运行并分析一个精简的操作系统内核,理解操作系统是如何工作的 使用实验楼的虚拟机打开shell cd LinuxKernel/linux-3. ...

  2. 使用thinkphp框架实现Excel导入数据库

    之前讲过php实现Excel导出数据库的随笔,链接:https://www.cnblogs.com/nuanai/p/6727711.html 之前的项目用到较多的就是Excel导出,现在用到了Exc ...

  3. Beta 总结

    前言 作业发布 组长 成员 贡献分 ★ 530 雨勤 14 311 旭 15 403 俊 16 223 元 14 437 海辉 17 7天 Beta 冲刺站立会议博客链接汇总 Beta 冲刺 (1/7 ...

  4. 对TCP重传的进一步认识

    http://blog.sina.com.cn/s/blog_4d276ac901011ee7.html ——TCM项目所得 一.看图说话 1.基于套接字的TCP服务器/客户端程序流程 2.TCP三次 ...

  5. CAS的应用场景

    国外应用(需FQ尝试): 来自CAS官网推荐的Demo http://casserver.herokuapp.com/cas/login https://casserver.herokuapp.com ...

  6. 【Java集合的详细研究4】Java中如何遍历Map对象的4种方法

    方法一 通过Map.entrySet遍历key和value,在for-each循环中使用entries来遍历.推荐,尤其是容量大时 这是最常见的并且在大多数情况下也是最可取的遍历方式.在键值都需要时使 ...

  7. SQLSERVER case when 的学习

    sqlserver 查询时的CASE WHEN学习记录 ) as '任务数', RPATask_State as id, case RPATask_State when then '已接收' when ...

  8. [日常工作]GS使用消息队列进行凭证实时记账 提高性能配置方法

    1. 安装消息队列服务 使用平台技术部的一键安装工具,安装. 自带jdk以及activeMQ 自动注册服务. 比较方便. 2. 修改/gsp/config下面的MQ配置文件,将消息队列服务修改为当前虚 ...

  9. codeforces580C

    Kefa and Park CodeForces - 580C 一棵以1为根的树,树上有些点是红的.一个叶子是合法的当且仅当从根到它的路径上出现的连续红点个数不超过m.求有多少个叶子是合法的.Inpu ...

  10. 国产首款5G手机抢先亮相:如此给力的说!

    5G网络是接下来移动互联网发展的主旋律,各家都在努力跟进,目前最积极的当属手机厂商,而2019年我们就能看到多款5G手机降临了. 在11月27日的未来信息通信技术国际研讨会上,vivo展示了他们正在研 ...