我的编程空间,编程开发者的网络收藏夹
学习永远不晚

python元类编程的基本使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

python元类编程的基本使用

1.1.propety动态属性

在面向对象编程中,我们一般把名词性的东西映射成属性,动词性的东西映射成方法。在python中他们对应的分别是属性self.xxx和类方法。但有时我们需要的属性需要根据其他属性动态的计算,此时如果直接使用属性方法处理,会导致数据不同步。下面介绍@property方法来动态创建类属性。

from datetime import datetime,date
 
class User:
    def __init__(self,name,birthday):
        self.name = name
        self.birthday = birthday
        self._age = 0
 
    @property
    def age(self):
        return datetime.now().year - self.birthday.year
 
    @age.setter
    def age(self,value):
        self._age = value
 
if __name__ == '__main__':
    user = User("derek",date(year=1994,month=11,day=11))
    user.age = 23
    print(user._age)   # 23
    print(user.age)    # 24 ,动态计算出来的

1.2.__getattr__和__getattribute__的区别

object.__getattr__(self, name) 
找不到attribute的时候,会调用getattr,返回一个值或AttributeError异常。 

object.__getattribute__(self, name) 
无条件被调用,通过实例访问属性。如果class中定义了__getattr__(),则__getattr__()不会被调用(除非显示调用或引发AttributeError异常)

(1)调用一个不存在的属性

class User:
    def __init__(self,info={}):
        self.info = info
 
    # def __getattr__(self, item):
    #     return self.info[item]
 
if __name__ == '__main__':
    user = User(info={"name":"derek","age":24})
    print(user.name)

会报错

(2)加了__getattr__之后就可以调用了

class User:
    def __init__(self,info={}):
        self.info = info
 
    #__getattr__是在查找不到属性的时候调用
    def __getattr__(self, item):
        return self.info[item]
 
if __name__ == '__main__':
    user = User(info={"name":"derek","age":24})
    print(user.name)    #derek

 (3)__getattribute__

class User:
    def __init__(self,info={}):
        self.info = info
 
    #__getattr__是在查找不到属性的时候调用
    def __getattr__(self, item):
        return self.info[item]
 
    #__getattribute不管属性存不存在,都访问这个
    def __getattribute__(self, item):
        return "zhang_derek"
 
 
if __name__ == '__main__':
    user = User(info={"name":"derek","age":24})
    #不管属性存不存在,都走__getattribute__
    print(user.name)    #zhang_derek     #即使属性存在也走__getattribute__
    print(user.test)     #zhang_derek    #不存在的属性也能打印
    print(user.company)   #zhang_derek   #不存在的属性也能打印

1.3.属性描述符

验证赋值的时候是不是int类型

#属性描述符
 
import numbers
 
#只要一个类实现了下面三种魔法函数中的一种,这个类就是属性描述符
class IntField:
    def __get__(self, instance, owner):
        return self.value
    def __set__(self, instance, value):
        if not isinstance(value,numbers.Integral):
            raise ValueError("必须为int")
        self.value = value
    def __delete__(self, instance):
        pass
 
class User:
    age = IntField()
 
if __name__ == '__main__':
    user = User()
    user.age = 24
    print(user.age)

如果user.age=24,值是int,可以正常打印  

如果user.age='test',传一个字符串,则会报错

1.4.__new__和__init__的区别

(1)__new__方法如果不返回对象,不会执行init方法

class User:
    def __new__(cls, *args, **kwargs):
        print("in new")
 
    def __init__(self,name):
        print("in init")
        self.name = name
 
# new是用用来控制对象的生成过程,在对象生成之前
# init是用来完善对象的
# 如果new方法不返回对象,则不会调用init函数
if __name__ == '__main__':
    user = User("derek")

运行结果:没有调用init方法

 (2)返回对象就会执行init方法

class User:
    def __new__(cls, *args, **kwargs):
        print("in new")         #in new
        print(cls)              #cls是当前class对象    <class '__main__.User'>
        print(type(cls))        #<class 'type'>
        return super().__new__(cls)   #必须返回class对象,才会调用__init__方法
 
    def __init__(self,name):
        print("in init")        #in init
        print(self)             #self是class的实例对象      <__main__.User object at 0x00000000021B8780>
        print(type(self))       #<class '__main__.User'>
        self.name = name
 
# new是用用来控制对象的生成过程,在对象生成之前
# init是用来完善对象的
# 如果new方法不返回对象,则不会调用init函数
if __name__ == '__main__':
    user = User(name="derek")
 
#总结
# __new__ 用来创建实例,在返回的实例上执行__init__,如果不返回实例那么__init__将不会执行
# __init__ 用来初始化实例,设置属性什么的

1.5.自定义元类

(1)前戏:通过传入不同的字符串动态的创建不同的类

def create_class(name):
    if name == 'user':
        class User:
            def __str__(self):
                return "user"
        return User
 
    elif name == "company":
        class Company:
            def __str__(self):
                return "company"
        return Company
 
if __name__ == '__main__':
    Myclass = create_class("user")
    my_obj = Myclass()
    print(my_obj)    #user
    print(type(my_obj))     #<class '__main__.create_class.<locals>.User'>

(2)用type创建

虽然上面的方法能够创建,但很麻烦,下面是type创建类的一个简单实例

# 一个简单type创建类的例子
#type(object_or_name, bases, dict)
#type里面有三个参数,第一个类名,第二个基类名,第三个是属性
User = type("User",(),{"name":"derek"})
 
my_obj = User()
print(my_obj.name)    #derek

(3)不但可以定义属性,还可以定义方法

def say(self):     #必须加self
    return "i am derek"
 
User = type("User",(),{"name":"derek","say":say})
 
my_obj = User()
print(my_obj.name)     #derek
print(my_obj.say())    #i am derek

(4)让type创建的类继承一个基类

def say(self):     #必须加self
    return "i am derek"
 
class BaseClass:
    def answer(self):
        return "i am baseclass"
 
#type里面有三个参数,第一个类名,第二个基类名,第三个是属性
User = type("User",(BaseClass,),{"name":"derek","say":say})
 
if __name__ == '__main__':
 
    my_obj = User()
    print(my_obj.name)          # derek
    print(my_obj.say())         # i am derek
    print(my_obj.answer())      # i am baseclass

1.6.什么是元类?

元类就是创建类的类,比如上面的type

在实际编码中,我们一般不直接用type去创建类,而是用元类的写法,自定义一个元类metaclass去创建

# 把User类创建的过程委托给元类去做,这样代码的分离性比较好
 
class MetaClass(type):
    def __new__(cls, *args, **kwargs):
        return super().__new__(cls,*args, **kwargs)
 
class User(metaclass=MetaClass):
    def __init__(self,name):
        self.name = name
 
    def __str__(self):
        return "test"
 
if __name__ == '__main__':
    #python中类的实例化过程,会首先寻找metaclass,通过metaclass去创建User类
    my_obj = User(name="derek")
    print(my_obj)    #test

到此这篇关于python元类编程的文章就介绍到这了,更多相关python元类编程内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

python元类编程的基本使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

python元类编程的基本使用

本文主要介绍了python元类编程的基本使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-02-22

python元类编程如何使用

本文小编为大家详细介绍“python元类编程如何使用”,内容详细,步骤清晰,细节处理妥当,希望这篇“python元类编程如何使用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。1.1.propety动态属性在面向对
2023-07-05

Python基本数据类型中元组的用法

本篇文章为大家展示了Python基本数据类型中元组的用法,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。1.元组的概念python中的元组是有序元素组成的集合,与列表的区别在于,元组是不可变的,一旦定
2023-06-02

详解Python编程中基本的数学计算使用

数 在 Python 中,对数的规定比较简单,基本在小学数学水平即可理解。 那么,做为零基础学习这,也就从计算小学数学题目开始吧。因为从这里开始,数学的基础知识列位肯定过关了。>>> 3 3 >>> 33333333333333333333
2022-06-04

怎么使用Python元类编程实现一个简单的ORM

这篇文章主要讲解了“怎么使用Python元类编程实现一个简单的ORM”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么使用Python元类编程实现一个简单的ORM”吧!概述什么是ORM?  
2023-07-05

Python使用SocketServer模块编写基本服务器程序的教程

SocketServer简化了网络服务器的编写。它有4个类:TCPServer,UDPServer,UnixStreamServer,UnixDatagramServer。这4个类是同步进行处理的,另外通过ForkingMixIn和Thre
2022-06-04

Shell编程控制结构的基本使用

目录if-then-else分支结构case分支结构for循环结构以常量作为值列表以变量作为值列表以命令运行结果作为值列表expr命令计算器while循环结构until循环结构if-then-else分支结构 if-then-else是一种
2022-06-04

使用Python怎么实现元编程

本篇文章为大家展示了使用Python怎么实现元编程,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。一、前言首先说,Python中一切皆对象,老生常谈。还有,Python提供了许多特殊方法、元类等等这样
2023-06-15

如何使用Python中的元编程技巧

如何使用Python中的元编程技巧导语:元编程是一种编程范式,指的是在运行时创建或修改代码的能力。Python作为一门动态语言,具备强大的元编程能力。本文将介绍Python中常用的元编程技巧,并给出具体的代码示例。一、使用元类元类是用于创建
2023-10-22

python-pptx的基本使用

1.引入python-pptxfrompptximportpresentation# 实例化Presentationprs= Presentation()2.ppt模板的选择a、使用ppt自带的模板prs= Presentation()pr
2023-01-31

Python元类编程实现一个简单的ORM

本文主要介绍了Python元类编程实现一个简单的ORM,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-03-06

Python编程基础的字典怎么使用

本篇内容介绍了“Python编程基础的字典怎么使用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、字典概念字典(dict)可能是最重要的P
2023-06-21

Shell AWK编程的基本介绍和使用详解

目录1、AWK介绍(1)AWK概述(2)printf格式化输出(3)printf命令说明2、AWK的基本使用(1)AWK命令说明(2)AWK命令使用1、AWK介绍 (1)AWK概述 AWK是一种处理文本文件的语言,是一个强大的文本分
2022-06-08

Python中JSON的基本使用

JSON (JavaScript Object Notation) 是一种轻量级的数据交换格式。Python3 中可以使用 json 模块来对 JSON 数据进行编解码,它主要提供了四个方法: dumps、dump、loads、load。d
2023-01-31

Python中csv模块的基本使用教程

1、csv简介CSV (Comma Separated Values) ,即逗号分隔值(也称字符分隔值,因为分隔符可以不是逗号),是一种常用的文本 格式,用以存储表格数据,包括数字或者字符。很多程序在处理数据时都会碰到csv这种格式的文件,
2022-06-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录