首页 > 编程语言 >Python中动态类和动态方法的创建与调用

Python中动态类和动态方法的创建与调用

时间:2024-08-03 16:42:25浏览次数:17  
标签:__ 调用 name Python age print 动态 属性

借助于python的动态语言特性,很容易对对象进行添加方法或者属性,这也是python的灵活之一。

动态生成类的属性及其方法

在某些情况可能要根据不同的参数来动态生成不同的实例方法、静态方法、类方法。下面的例子中则展示了如何动态地向类中添加属性和方法。

import types

class Person():
	def __init__(self,name): 
		self.name = name 
    
li = Person('Lily') 
li.age = 20  # 实例属性添加,仅对当下实例有效
tom = Person('Tom')
print(tom.age) # 'Person' object has no attribute 'age'

Person.age = None  # 类属性添加 
print(tom.age)  # None

def eat(self): 
	print('%s正在吃东西。。'%self.name) 

li.eat = types.MethodType(eat, li) # 实际上python所有类都是type类的实例对象,动态添加了Person的实例方法
li.eat() 

@staticmethod  
def test():  
	print('这是一个静态方法。')

Person.test = test  # 动态添加动态方法
Person.test()  
 
@classmethod  
def test(cls): 
    print(cls.age) # None,访问动态创建的age
    print('这是一个类方法。')

Person.test = test # 动态添加类方法
Person.test() 
 
class test(object): 
	__slots__ = ('name','age') # 使用slots来将属性固定,不允许增删

动态地创建类

由于所有类都是type类的对象,所以也可以通过type动态地创建类。

Test = type('Test',(object,),{'num':0})  # 所有类都是type的对象,param1为类名,param2为继承对象,num为类属性,方法

class Test(object): # 与上述代码等效
    num = 0

如果需要添加属性方法,则在相应的传参字典中添加对应的方法,例如:

Test = type('Test',(object,),{'num':0, 'foo': fun})

动态访问类中的属性方法

动态地方法类中的属性方法,也是一种反射机制。python中的反射/自省的实现,是通过hasattr、getattr、setattr、delattr四个内置函数实现的,其实这四个内置函数不只可以用在类和对象中,也可以用在模块等其他地方,只是在类和对象中用的很多,所以单独提出来进行解释。

  • hasattr(key) # 返回的是一个bool值,判断某个成员或者属性在不在类或者对象中
  • getattr(key,default=xxx) # 获取类或者对象的成员或属性,如果不存在,则会抛出AttributeError异常,如果定义了default那么当没有属性的时候会返回默认值。
  • setattr(key,value)假如有这个属性,那么更新这个属性,如果没有就添加这个属性并赋值value
  • delattr(key)删除某个属性

其用法如下所示:

class Foo:
    def __init__(self,name,age):
        self.name=name
        self.age=age
    def show(self):
        return self.name,self.age

#学习中遇到问题没人解答?小编创建了一个Python学习交流群:531509025
obj=Foo("Tom",18)
print(getattr(obj,"name"))  # Tom
setattr(obj,"k1",eat)
print(obj.k1) # <function eat at 0x00000162CAD661F0>
print(hasattr(obj,"k1")) # True
delattr(obj,"k1")
show_fun=getattr(obj,"show") 
print(show_fun()) # ('Tom', 18)

动态访问普通全局函数

有时候需要通过函数名来动态访问全局的函数,那么依然有三种方法。

通过eval,不同由于考虑到安全因素,不能直接这样去写,可能会得到恶意代码

eval("print")("hello,world!")

通过建立字典,事先将需要调用的函数全部放入字典,缺点是每增加一个动态函数,就要更改字典:

dynamic_fun = {"print": print }
dynamic_fun["print"]("hello,world!")

通过调用global()来使用
global()维护了一个全局的变量列表,其实现过程和方式2类似,具体使用时通过global().get(fun_name)来完成。

类中其他内建属性方法(魔术方法)

__init__ # 构造初始化函数,__new__之后运行
__new__ # 创建实例所需的属性,类似于构造方法
__call__ # 可以使得类的实例通过function方式访问
__class__ # 实例所在的类,实例.__class__
__str__ # 实例的字符串表示,可读性高
__repr__ # 实例的字符串表示,准确性高
__del__ # 删除实例引用,类似于析构方法
__dict__ # 实力自定义属性,vars(实例.__dict__)
__doc__ # 类文档,help(类或者实例)
__bases__ #当前类的所有父类
__getattribute__ #属性访问拦截器

标签:__,调用,name,Python,age,print,动态,属性
From: https://www.cnblogs.com/python1111/p/18340695

相关文章

  • Python学习中最常见的10个列表操作问题
    列表是Python中使用最多的一种数据结果,如何高效操作列表是提高代码运行效率的关键,这篇文章列出了10个常用的列表操作,希望对你有帮助。1、迭代列表时如何访问列表下标索引普通版:items=[8,23,45]forindexinrange(len(items)):print(index,"-->",items[index])​......
  • Python中定义(创建)、调用函数及返回值
    1.定义(创建)函数要调用一个函数,首先要定义它。在Python中使用关键字def来定义一个函数。函数通常由函数名、参数列表以及一系列语句组成的函数体构成的。函数定义的一般格式如下:def函数名(参数列表):函数体例如:defsayhello(): print('hello')最简单的函数:defm......
  • python用List的内建函数list.sort进行排序
    对List进行排序,Python提供了两个方法方法1用List的内建函数listsort进行排序listsort(func=None,key=None,reverse=False)Python实对List进行排序,Python提供了两个方法方法1.用List的内建函数list.sort进行排序list.sort(func=None,key=None,reverse=False)>>>list=......
  • 【笔记】动态规划选讲:凸优化技术大赏 2024.8.3
    如果您是搜索引擎搜进来的。很抱歉,没有您需要搜索的题目的题解。典题\(n\)个物品的背包,重量在\(1\sim4\)之间,价值在\(1\sim10^9\)之间。\(n\leq10^5\)。Minkowski和会遇到不连续的问题。不妨按照\(i\bmod12\)划分dp数组,每个剩余系都是凸的。枚举拿了\(......
  • 【JS】自调用函数怎么用?
    自调用函数定义自调用函数,也称为立即执行函数表达式(IIFE),是一种在定义后立即执行的函数(也就是说不用另外调用执行了)。它的主要目的是创建一个新的作用域,避免全局变量的污染。优势可以立即执行,不需要等待其他代码的执行。创建了新的作用域,可以保护内部的变量和函数不被外部......
  • 禁用 GIL 的 Python 3.13 非常慢
    我对python3.12.0与使用3.13.0b3标志编译的python--disable-gil进行了简单的性能测试。该程序使用ThreadPoolExecutor或ProcessPoolExecutor执行斐波那契数列的计算。引入禁用GIL的PEP文档表示,存在一些开销,主要是由于有偏差......
  • 有没有办法阻止 setUp() 为 python 测试用例中的每个测试方法启动浏览器?
    我正在练习编写Web自动化测试用例,并且编写了一些函数来测试登录、在用户主页中查找我的用户名以及测试GitHub的注销功能。然而,我通过经验和阅读了解到setUp()是在每个测试方法之前启动的,而我的问题是在每个测试方法之前它都会打开一个新的浏览器。我希望我的所有测......
  • 尝试使用Python抓取需要先登录的网站但没有成功
    我正在尝试抓取一个需要登录的网站(我的路由器GUI),但无论我做了什么,我都会反复返回登录站点的源代码,而不是成功登录后出现的页面。我做了一些阅读,并意识到我需要返回POST请求的答案。我想我找到了它们并返回了所需的值,但仍然-似乎没有任何效果。我使用https://curl.tri......
  • 给python初学者的一些建议
    写在开篇关于Python,可以这么说,这几年借着数据科学、机器学习与人工智能的东风,Python老树开新花,在风口浪尖上居高不下。Python之所以这么受大家的青睐,是因为它语言简洁,上手容易,让非计算机专业的人员也能快速上手,享受编程开发带来的便利和福利。但Python再简单,它也是一......
  • Python中15个递归函数经典案例解析
    1.阶乘计算阶乘是一个常见的递归应用,定义为n!=n*(n-1)*…*1。deffactorial(n):ifn==0:return1else:returnn*factorial(n-1)print(factorial(5))#输出:1202.斐波那契数列斐波那契数列的每一项都......