python基础day34 魔术方法和反射
魔术方法(内置方法)
类里面内置的双下划线开头的一线方法,他们具有特殊的功能,我们称之为是魔术方法,简称魔法。
eg:__init__
魔术方法的学习之需要掌握每个方法什么时候触发或者执行
1. __str__, __repr__方法
1. 打印对象的时候会自动触发这两个方法
2. 如果有返回值,类型必须是字符串形式的
3. 如果有返回值,那么,打印的结果就是它的返回值
4. 如果这两个方法同时存在,优先使用__str__方法,__str__的优先级高
5. print打印会执行这两个任意一个,str()会执行__str__, repr()会执行__repr__
class Student(): def __init__(self, name, age, gender): self.name = name self.age = age self.gender = gender def __str__(self): print("我执行了") # return None """__str__方法返回值必须是字符串类型""" # return "123" return "name:%s" % self.name # return 123 def __repr__(self): return "name1:%s" % self.name stu = Student("kevin", 19, 'male') """也就是说,__str__方法内部返回什么结果,打印的就是返回的结果,""" # print(stu) # 当打印或者输出对象的时候,会自动触发__str__的执行 print('from repr: ',repr(stu)) # __next__ __len__ print('from str: ',str(stu)) """__repr__:它的功能和__str__一样,只不过,只写了他们两个其中之一,都会执行,但是,如果两个都写了,__str__它的优先级高于_repr__""" # 我们一般用的是__Str__方法
2. __del__方法
1. 当删除对象的时候,会自动触发
2. 当程序执行结束的时候也会触发方法的执行
3. 它的作用一般用来清理数据
####__del__方法 class Student(): def __init__(self, name, age, gender): self.name = name self.age = age self.gender = gender self.f = open("a.txt", "w", encoding="utf8") # 我执行了: # 1. 当删除对象的时候,会自动触发函数的执行 # 2. 当程序结束的时候,也会自动触发执行 def __del__(self): print("我执行了") """可以做一些清理垃圾的操作""" self.f.close() stu = Student("kevin", 19, 'male') # del stu print("123")
isinstance(obj,cls)和issubclass(sub,super)
class People(): pass class Student(People): pass # stu = Student() # print(isinstance(stu, Student)) # True # print(isinstance("hellowrold", str)) # res = 'helloworld' # <class 'str'> # # res1 = int # # print(type(res)) # # # res = str('helloworld') print(issubclass(Student, People)) class Foo: """ author:kevin date:2023-01-01 email:liyangqit@163.com 这是注释""" pass class Bar(Foo): """这是Bar""" pass # document:可以查看出类内部的详细信息,其实就是注释里面的内容 # print(Foo.__doc__) print(Bar.__doc__) # 这个特性不能够继承到父类
__doc__
# 它是查看类内部的详细信息,类内部的注释内容 class Foo(): '''注释''' pass Foo.__doc__
3. __enter__和__exit__
1. 当出现with语句的时候,会触发__enter__方法
2. 如果__enter__方法中有返回值,则就赋值给as后面的变量名
3. 开始执行with代码块,当with代码块执行完毕之后,在执行__exit__方法
4. with代码块中出现了异常,则在__exit__方法中会拿到异常的信息(异常类型、异常值、追溯信息)
5. 如果在__exit__方法中return True,忽略异常,相当于什么都没发生
举例: with打开文件
class Open: def __init__(self,name): self.name=name def __enter__(self): print('出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量') # return self def __exit__(self, exc_type, exc_val, exc_tb): print('with中代码块执行完毕时执行我啊') with Open('a.txt') as f: print('=====>执行代码块') print('=====>执行代码块') print('=====>执行代码块') print('=====>执行代码块') print('=====>执行代码块') print('=====>执行代码块') class Open: def __init__(self,name): self.name=name def __enter__(self): print('出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量') def __exit__(self, exc_type, exc_val, exc_tb): print('with中代码块执行完毕时执行我啊') print(exc_type) # 异常类型 print(exc_val) # 异常值 print(exc_tb) # 追溯信息 # return True with Open('a.txt') as f: print('=====>执行代码块') raise AttributeError('***着火啦,救火啊***') print('0'*100) #------------------------------->不会执行 class Open: def __init__(self,name): self.name=name def __enter__(self): print('出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量') def __exit__(self, exc_type, exc_val, exc_tb): print('with中代码块执行完毕时执行我啊') print(exc_type) # 异常类型 print(exc_val) # 异常值 print(exc_tb) # 追溯信息 # return True with Open('a.txt') as f: print('=====>执行代码块') raise AttributeError('***着火啦,救火啊***') print('0'*100) #------------------------------->不会执行 """面试题:探探你对with上下文管理协议的理解?"""
4. __setattr__,__delattr__,__getattr__
前提:必须是使用点语法(.) obj.x
__getattr__: 获取一个不存在的属性名的时候,会自动触发__getattr__的执行,并且有个参数:属性名
__setattr__:设置一个不存在的属性的时候,会自动触发__setattr__的执行,并且有两个参数:key,value, self.__dict__[key] = value =====> self.key = value=====> 无限递归
__delattr__:删除一个属性的时候,触发的函数
class Foo: x = 1 def __init__(self, y): self.y = y def __getattr__(self, item): print('----> from getattr:你找的属性不存在') def __setattr__(self, key, value): print('----> from setattr') # print(key, value) # z 1 # self.z = 1 # 你使用点语法,设置一个不存在的属性值就会自动触发__setattr__ # self.key=value # 这就无限递归了,你好好想想 # self.__dict__[key] = value # self.__dict__[key]=value #应该使用它 # super(Foo, self).__setattr__() def __delattr__(self, item): # z print('----> from delattr') # del self.item #无限递归了 self.__dict__.pop(item) obj = Foo(10) obj.z # obj.z = 1 # ----> from setattr # print(obj.z) # None del obj.z
5. __setitem__,__getitem,__delitem__
前提:必须是使用中括号([]) obj['x']
__getitem__: 获取一个不存在的属性名的时候,会自动触发__getitem__的执行,并且有个参数:属性名
__setitem__:设置一个不存在的属性的时候,会自动触发__setitem__的执行,并且有两个参数:key,value, self.__dict__[key] = value =====> self.key = value=====> 无限递归
__delitem__:删除一个属性的时候,触发的函数
class Foo(object): def __init__(self, name): self.name = name def __getitem__(self, item): print(item) print("我执行了") print(self.__dict__[item]) def __setitem__(self, key, value): print("我也执行了") print(key, value) self.__dict__[key] = value def __delitem__(self, key): print('del obj[key]时,我执行') self.__dict__.pop(key) def __delattr__(self, item): print('del obj.key时,我执行') self.__dict__.pop(item) obj = Foo('kevin') # obj['name'] obj['age'] = 19 print(obj['age']) del obj['age']
__call__
# __call__ class Foo: def __init__(self): pass def __call__(self, *args, **kwargs): print('__call__') return '123' obj = Foo() # 123 print(obj()) # 对象加括号 # __call__
反射
反射其实就是通过字符串的形式来操作对象的属性
stu.z = 1 class Student(): school ='SH' def __init__(self, name, age ): self.name = name self.age = age def index(self): print('Student.index') stu = Student('jason', 20) # print(stu.'name')] # stu.name
getattr() setattr() delattr() hasattr()
# 1. getattr() res=getattr(stu, 'name') # attribute 属性 attr # jason res1=getattr(stu, 'age1', 666) # attribute 属性 attr # jason res1=getattr(stu, 'age1', 666) # attribute 属性 attr # jason print(res) print(res1) res = getattr(stu, 'index') # index函数的内存地址 print(res) res()
# 2. setattr() setattr(stu, 'x', 1) setattr(stu, 'name', '666') # 如果属性名存在就是修改,不存在就是增加 print(stu.__dict__)
# 3. delattr() delattr(stu, 'name') print(stu.__dict__)
# 4. hasattr() print(hasattr(stu, 'index')) # 返回布尔值,只是一个判断是否存在
# 补充 import time time.sleep(3) res=getattr(time, 'sleep') # built-in print(res) res(3) # time.sleep(3) getattr(time, 'sleep')(3) import time time = __import__("time") # 以字符串的形式导入模块 time.sleep(3) random = __import__("random") res=random.randint(0,9) print(res)
反射的案例
class FtpServer: def serve_forever(self): while True: inp = input('input your cmd>>: ').strip() # get a.txt put a.txt cmd, file = inp.split() # get a.txt ['get', 'a.txt'] # hasattr(self, 'get') # hasattr(self, 'put') if hasattr(self, cmd): # 根据用户输入的cmd,判断对象self有无对应的方法属性 # getattr(self, ’get‘) get或者put的内存地址 func = getattr(self, cmd) # 根据字符串cmd,获取对象self对应的方法属性 func(file) def get(self, file): print('Downloading %s...' % file) def put(self, file): print('Uploading %s...' % file) server = FtpServer() server.serve_forever()
异常的剩余内容
1. 异常就是错误发生的信息,我们必须要做处理,否则,后续代码无法正常运行 2. 如何捕捉异常 try: 被检测的代码 except 异常类型: pass else: print() finally: print() """以上结构是我们为了捕捉代码中出现的异常""" 3. 我们自己如何抛出异常:raise class Animal(): # @abc.abstractmethod def speak(self): raise Exception("请先实现speak方法") class People(Animal): def speak(self): pass # pass """主动抛出异常""" stu = People() stu.speak() 4. 自定义异常 class MyException(BaseException): def __init__(self, msg): self.msg = msg def __str__(self): return self.msg raise MyException("这是异常信息")