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

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

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

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

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

没有自己定义__getattr__的时候

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

我们自己定义一个__getattr__

class Foo:
def __init__(self,name):
self.name=name
def __getattr__(self, item):
print("你要找的属性%s不存在" %item)
f1=Foo("gouguoqi")
print(f1.name)
print(f1.age) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py gouguoqi 你要找的属性age不存在

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

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

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

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

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

class Foo:
def __init__(self,name):
self.name=name
def __setattr__(self, k, v):
print("执行setattr")
f1=Foo("gouguoqi")#实例化的过程就是运行init,init就有一个赋值name=gouguoqi所以就触发setaddr
f1.age=28
f1.gender="male"
print(f1.__dict__) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py 执行setattr 执行setattr 执行setattr {} 

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

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

class Foo:
def __init__(self,name):
self.name=name
def __setattr__(self, k, v):
print("执行setattr")
if type(v) is str:
print("开始设置")
else:
print("必须是字符串")
f1=Foo("gouguoqi")
f1.age=28
print(f1.__dict__) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py 执行setattr 开始设置 执行setattr 必须是字符串 {}

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

class Foo:
def __init__(self,name):
self.name=name
def __setattr__(self, k, v):
print("执行setattr")
if type(v) is str:
print("开始设置")
self.__dict__[k]=v
else:
print("必须是字符串")
f1=Foo("gouguoqi")
f1.age=28
print(f1.__dict__) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py 执行setattr 开始设置 执行setattr 必须是字符串 {'name': 'gouguoqi'}

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

class Foo:
def __init__(self,name):
self.name=name
def __setattr__(self, k, v):
print("执行setattr")
if type(v) is str:
print("开始设置")
self.__dict__[k]=v.upper()
else:
print("必须是字符串")
f1=Foo("gouguoqi")
f1.age=28
print(f1.__dict__) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py 执行setattr 开始设置 执行setattr 必须是字符串 {'name': 'GOUGUOQI'}

3、__delattr__  删除属性

class Foo:
def __init__(self,name):
self.name=name
def __delattr__(self, item):
print("执行delattr",item)
self.__dict__.pop(item)#操作底层的属性字典的key
f1=Foo("gouguoqi")
del f1.name#把f1传给self,把name传给item
print(f1.__dict__) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py 执行delattr name {}

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

class Foo:
def __init__(self,name):
self.name=name
def __delattr__(self, item):
print("%s属性不允许删除" %item)
f1=Foo("gouguoqi")
del f1.name
print(f1.__dict__) C:\python35\python3.exe D:/pyproject/day26/双下划线开头的attr方法.py name属性不允许删除 {'name': 'gouguoqi'} 

4、包装

class List(list):#定义个类,继承系统的list
pass l2=list("hello world")
print(l2,type(l2)) l1=List("hello world")
print(l1,type(l1)) C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py ['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'] <class 'list'> ['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'] <class '__main__.List'>

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

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

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

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

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

实现真正的append操作

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

类调用方法,推荐用super

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

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

class List(list):#定义个类,继承系统的list
def append(self, object):
if type(object) is str:
# list.append(self,object)#用父类的append方法
super().append(object)#super就是调用父类的方法,不用传self
else:
print("只能append字符串")
def show_middle(self):
mid_index=int(len(self)/2)#取到中间的索引
return self[mid_index]#返回列表的中间的索引的那个值
l1=List("helloworld")
l1.append("")
print(l1)
l1.append(666) C:\python35\python3.exe D:/pyproject/day26/包装标准类型.py ['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内核学习期末总结(网课)

    标签(空格分隔): 20135321余佳源 余佳源(原创作品转载请注明出处) <Linux内核分析> MOOC课程http://mooc.study.163.com/course/USTC ...

  2. java-过滤器、拦截器

    1.基础知识 1.1面向对象编程(OOP).面向切面编程(AOP) 面向对象编程: 将需求功能划分为不同的.相对独立的和封装良好的类,使他们有属于自己的行为,依靠继承和多态等来定义彼此的关系. 面向切 ...

  3. Junit4测试用例

    一.题目简介 测试一元一次方程的求解 二.源码的github链接 https://github.com/liujing1994/test1 三.所设计的模块测试用例.测试结果截图   一元一次方程测试 ...

  4. VirtualBox 导入虚拟机时的注意事项 VDI 与VMDK

    1. 建议不要勾选  as vdi   vmdk 最好不过了.. 长个经验教训 以后尽量不勾选 vmdk 有很多工具能进行转换 vdi的 要麻烦一些.

  5. 被辞退时N+1的说法

    “N+”,指在这家公司工作了N年,赔偿N个月的上年平均工资,再加上1个月“代通知金”. N的上限时12: 上年平均工资的上限是工作所在城市平均工资的三倍. StudyFrom知乎 所以很明显能够算出来 ...

  6. ubuntu安装命令

    sudo apt-get update  更新源sudo apt-get install package 安装包sudo apt-get remove package 删除包sudo apt-cach ...

  7. Delphi中封装ADO之我重学习记录

    delphi adodataset ctstatic 数据是缓存在服务器端还是客户端 答:客户端,开启本地缓存功能后,就能数据在本地批量修改后,再批量提交,减少了网络传送   原创,专业,图文 Del ...

  8. 我终于激活Windows Server2008 R2了!!

    经过我不懈的努力,在重装两次系统后,我终于实现了win2008的KMS激活.这个方法可以避免虚拟机架设KMS服务器的麻烦.现将激活方法发布如下. 首先要选择安装的操作系统.Windows Server ...

  9. ajax readystatue

    onReadyStatueChange存儲狀態處理函數,當readyStatue改變時,調用該函數. readyStatue的狀態: 0   請求未初始化 1   服務器連接已建立 2  請求已接收 ...

  10. Java之数组遍历

    package basic; //数组遍历方法 public class ForEach { public static void main(String[] args) { // 原始数组 Stri ...