python学习笔记:第18天 面向对象04-反射
issubclass和isinstance
- issubclass:可以判断一个类是否另一个类的子类。
# issubclass
class A:
pass
class B(A):
pass
class C(B):
pass
print(issubclass(B, A))
print(issubclass(C, B))
print(issubclass(C, A))
# 结果:
# True
# True
# True
- type:然后我们来看type. type在前⾯的学习期间已经使⽤过了. type(obj) 表⽰查看obj是由哪个类创建的
- isinstance:判断一个对象是否是某个类的实例
class Foo:
pass
obj = Foo()
print(obj, type(obj)) # 查看obj是由那个类创建的
我们再来看一个加法函数函数:
# type和issubclass
# 定义一个函数,计算两个数的和
def cal(a, b):
if (type(a) == int or type(b) == float) and (type(b) == int or type(b) == float):
return a + b
else: # 这路先进行了类型的判断,如果不是int或者是float类型的则不进行计算
return '不能帮你计算'
print(cal(10, 20))
print(cal(10, '胡辣汤'))
# 结果:
# 30
# 不能帮你计算
像这种处理可以使用isinstance
来处理,isinstance
咳哟判断一个对象是否是一个类的实例或者是其父类的实例:
def cal (a, b):
if isinstance(a, (int, float)) and isinstance(b, (int, float)):
return a + b
else:
return '不能帮你计算'
print(cal(10.3, 20))
print(cal(10, '胡辣汤'))
# 结果:
# 30.3
# 不能帮你计算
总结一下:
内置函数 | 描述及使用场景 | 使用方法 |
---|---|---|
issubclass | 判断一个类是否是另一个类的子类 | issubclass(cls, class_or_tuple) |
type | 查看一个对象的具体类型,不会查找父类 | type(obj) |
isinstance | 判断一个对象是否是某个对象的实例,但是isinstance 没有type 那么精准,他会向上查找父类 |
isinstance(obj, class_or_tuple) |
区分函数和方法
from types import FunctionType, MethodType
class Car:
def run(self): # 实例方法
print("我是车, 我会跑")
@staticmethod
def cul():
print("我会计算")
@classmethod
def jump(cls):
print("我会jump")
# 实例方法:
# 1. 用对象.方法 方法
# 2. 类名.方法 函数
c = Car()
print(isinstance(c.run, FunctionType)) # False
print(isinstance(Car.run, FunctionType)) # True
print(isinstance(c.run, MethodType)) # True
print(isinstance(Car.run, MethodType)) # False
# 静态方法 都是函数
print(isinstance(c.cul, FunctionType)) # True
print(isinstance(Car.cul, FunctionType)) # True
print(isinstance(c.cul, MethodType)) # False
print(isinstance(Car.cul, MethodType)) # False
# 类方法都是方法
print(isinstance(c.jump, FunctionType)) # False
print(isinstance(Car.jump, FunctionType)) # False
print(isinstance(c.jump, MethodType)) # True
print(isinstance(Car.jump, MethodType)) # True
- 类⽅法. 不论任何情况, 都是⽅法.
- 静态⽅法, 不论任何情况. 都是函数.
- 实例⽅法, 如果是实例访问. 就是⽅法. 如果是类名访问就是函数.
反射
关于反射, 其实⼀共有4个函数:
hasattr(obj, str)判断obj中是否包含str成员
getattr(obj,str) 从obj中获取str成员
setattr(obj, str, value) 把obj中的str成员设置成value. 注意. 这⾥的value可以是值, 也可以是函数或者⽅法
delattr(obj, str) 把obj中的str成员删除掉
- hasattr:判断某个对象中是否存在指定的属性或者方法
- getattr:获取某个对象中指定的方法
hasattr(obj, name, /)
Return whether the object has an attribute with the given name.
This is done by calling getattr(obj, name) and catching AttributeError.
setattr(obj, name, value, /)
Sets the named attribute on the given object to the specified value.
setattr(x, 'y', v) is equivalent to ``x.y = v''
class Foo:
def func1(self):
print('我是func1函数')
def func2(self):
print('我是func2函数')
def func3(self):
print('我是func3函数')
f = Foo()
fn = input('请输入你要测试的函数:')
if hasattr(f, fn): # 先判断f中是否有fn属性
func = getattr(f, fn) # 确认有这个属性在从对象中获取这个属性
func()
else:
print('没有这个功能')
# 结果:
# 请输入你要测试的函数:func1
# 我是func1函数
- setattr:设置某个对象的属性(变量或者是方法)
setattr(f, 'func3', lambda:print('我是自定义函数')) # 设置自定义方法
f.func3()
setattr(f, 'name', 'zzc') # 设置自定义属性
print(f.name)
# 结果:
# 我是自定义函数
# zzc
- delattr:删除对象中指定的属性
delattr(Foo, 'func2')
f.func2() # 此时Foo中的func2方法已经被删除了,这里执行会抛出异常
---------------------------------------------------------------------------
AttributeError Traceback (most recent call last)
<ipython-input-5-dd69df692558> in <module>
----> 1 f.func2()
AttributeError: 'Foo' object has no attribute 'func2'