python中set函数作用如何自己用代码实现_Python进阶开发之元类编程

Photo byJoyous From Lofter

本文目录

类是如何产生的

如何使用type创建类

理解什么是元类

使用元类的意义

元类实战:ORM

.1. 类是如何产生的

类是如何产生?这个问题肯定很傻。实则不然,很多人只知道使用继承的表面形式来创建一个类,却不知道其内部真正的创建是由type来创建的。

type?这不是判断对象类型的函数吗?

是的,type通常用法就是用来判断对象的类型。但除此之外,他最大的用途是用来动态创建类。当Python扫描到class的语法的时候,就会调用type函数进行类的创建。

.2. 如何使用type创建类

首先,type()需要接收三个参数

类的名称:若不指定,也要传入空字符串:""

父类:注意以tuple的形式传入,若没有父类也要传入空tuple:(),默认继承object

绑定的方法或属性:注意以dict的形式传入

来看个例子

1# 准备一个基类(父类)

2class BaseClass:

3    def talk(self):

4        print("i am people")

5

6# 准备一个方法

7def say(self):

8    print("hello")

9

10# 使用type来创建User类

11User = type("User", (BaseClass, ), {"name":"user", "say":say})

.3. 理解什么是元类

什么是类?可能谁都知道,类就是用来创建对象的「模板」。

那什么是元类呢?一句话通俗来说,元类就是创建类的「模板」。

为什么type能用来创建类?因为它本身是一个元类。使用元类创建类,那就合理了。

type是Python在背后用来创建所有类的元类,我们熟知的类的始祖object 也是由type创建的。更有甚者,连type自己也是由type自己创建的,这就过份了。

1>>> type(type)

2

3>>> type(object

4

5>>> type(int)

6

7>>> type(str)

8

如果要形象的来理解的话,就看下面这三行话。

1str:用来创建字符串对象的类。

2int:是用来创建整数对象的类。

3type:是用来创建类对象的类。

反过来看

1一个实例的类型,是类

2一个类的类型,是元类

3一个元类的类型,是type

来看下实例

1#Python3.7

2>>> class MetaPerson(type):

3...     pass

4...

5>>> class Person(metaclass=MetaPerson):

6...     pass

7...

8>>> Tom = Person()

9>>> print(type(Tom))

10

11>>> print(type(Tom.__class__))

12

13>>> print(type(Tom.__class__.__class__))

14

上面是一个简单的示例。

下面看一个稍微完整的

1# 注意要从type继承

2class BaseClass(type):

3    def __new__(cls, *args, **kwargs):

4        print("in BaseClass")

5        return super().__new__(cls, *args, **kwargs)

6

7class User(metaclass=BaseClass):

8    def __init__(self, name):

9        self.name = name

10

11user = User("wangbm")

.4. 使用元类的意义

正常情况下,我们都不会使用到元类。但是这并不意味着,它不重要。假如某一天,我们需要写一个框架,很有可能就需要用到元类。

但是,为什么要用它呢?不要它会怎样?

经过我的总结,元类的作用过程如下

拦截类的创建

拦截下后,进行修改

修改完后,返回修改后的类

很明显,使用元类,是要对类进行定制修改。使用元类来动态生成元类的实例,而99%的开发人员是不需要动态修改类的,因为这应该是框架才需要考虑的事。

但是,这样说,你一定不会服气,到底元类用来干什么?其实元类的作用就是创建API,一个最典型的应用是 Django ORM。

.5. 元类实战:ORM

使用过Django ORM的人都知道,有了ORM,使得我们操作数据库,变得异常简单。

ORM的一个类(User),就对应数据库中的一张表。id,name,email,password 就是字段。

1class User(BaseModel):

2    id = IntField('id')

3    name = StrField('username')

4    email = StrField('email')

5    password = StrField('password')

6

7    class Meta:

8        db_table = "user"

如果我们要插入一条数据,我们只需这样做

1# 实例化成一条记录

2u = User(id=20180424, name="xiaoming",

3         email="xiaoming@163.com", password="abc123")

4

5# 保存这条记录

6u.save()

通常用户层面,只需要懂应用,就像上面这样操作就可以了。

但是今天我并不是来教大家如何使用ORM,我们是用来探究ORM内部究竟是如何实现的。我们也可以自己写一个简易的ORM。

从上面的User类中,我们看到StrField和IntField,从字段意思上看,我们很容易看出这代表两个字段类型。字段名分别是id,username,email,password。

StrField和IntField在这里的用法,叫做属性描述符,如果对这个不了解的可以查看文章底部的参考文章,也是我写的。

简单来说呢,属性描述符可以实现对属性值的类型,范围等一切做约束,意思就是说变量id只能是int类型,变量name只能是str类型,否则将会抛出异常。

那如何实现这两个属性描述符呢?请看代码。

1import numbers

2

3class Field:

4    pass

5

6class IntField(Field):

7    def __init__(self, name):

8        self.name = name

9        self._value = None

10

11    def __get__(self, instance, owner):

12        return self._value

13

14    def __set__(self, instance, value):

15        if not isinstance(value, numbers.Integral):

16            raise ValueError("int value need")

17        self._value = value

18

19class StrField(Field):

20    def __init__(self, name):

21        self.name = name

22        self._value = None

23

24    def __get__(self, instance, owner):

25        return self._value

26

27    def __set__(self, instance, value):

28        if not isinstance(value, str):

29            raise ValueError("string value need")

30        self._value = value

我们看到User类继承自BaseModel,这个BaseModel里,定义了数据库操作的各种方法,譬如我们使用的save函数,也可以放在这里面的。所以我们就可以来写一下这个BaseModel类

1class BaseModel(metaclass=ModelMetaClass):

2    def __init__(self, *args, **kw):

3        for k,v in kw.items():

4            # 这里执行赋值操作,会进行数据描述符的__set__逻辑

5            setattr(self, k, v)

6        return super().__init__()

7

8    def save(self):

9        db_columns=[]

10        db_values=[]

11        for column, value in self.fields.items():

12            db_columns.append('`'+str(column)+'`')

13            _value=str(getattr(self, column))

14            db_values.append('\''+_value+'\'')

15        sql = "insert into `{table}` ({columns}) values({values})".format(

16                table=self.db_table,

17                columns=','.join(db_columns),

18                values=','.join(db_values))

19        # mysql_execute 函数可以自己写。调用驱动插入到数据库

20        # 查看完整代码请点击文章底部查看原文

21        mysql_execute(sql)

从BaseModel类中,save函数里面有几个新变量,

fields: 存放所有的字段属性

db_table:表名

注意:上面代码中class BaseModel(metaclass=ModelMetaClass)请替换成class BaseModel(object) 再阅读。这样更贴合思考顺序。

我们思考一下这个u实例的创建过程:

type -> object -> BaseModel -> User -> u

这里会有几个问题。

init的参数是User实例时传入的,所以传入的id是int类型,name是str类型。看起来没啥问题,若是这样,我上面的数据描述符就失效了,不能起约束作用。所以我们希望init接收到的id是IntField类型,name是StrField类型。

同时,我们希望这些字段属性,能够自动归类到fields变量中。因为,做为BaseModel,它可不是专门为User类服务的,它还要兼容各种各样的表。不同的表,表里有不同数量,不同属性的字段,这些都要能自动类别并归类整理到一起。这是一个ORM框架最基本的。

我们希望对表名有两种选择,一个是User中若指定Meta信息,比如表名,就以此为表名,若未指定就以类名的小写 做为表名。虽然BaseModel可以直接取到User的db_table属性,但是如果在数据库业务逻辑中,加入这段复杂的逻辑,显然是很不优雅的。

上面这几个问题,其实都可以通过元类的__new__函数来完成。

元类的__new__和普通类的可不一样,元类的__new__,可以获取到上层类的一切属性和方法,包括类名,魔法方法。

而普通类的__new__ 只能获取到实例化时外界传入的属性。

下面就来看看,如何用元类来解决这些问题呢?请看代码。

1class ModelMetaClass(type):

2    def __new__(cls, name, bases, attrs):

3        if name == "BaseModel":

4            # 第一次进入__new__是创建BaseModel类,name="BaseModel"

5            # 第二次进入__new__是创建User类及其实例,name="User"

6            return super().__new__(cls, name, bases, attrs)

7

8        # 根据属性类型,取出字段

9        fields = {k:v for k,v in attrs.items() if isinstance(v, Field)}

10

11        # 如果User中有指定Meta信息,比如表名,就以此为准

12        # 如果没有指定,就默认以 类名的小写 做为表名,比如User类,表名就是user

13        _meta = attrs.get("Meta", None)

14        db_table = name.lower()

15        if _meta is not None:

16            table = getattr(_meta, "db_table", None)

17            if table is not None:

18                db_table = table

19

20        # 注意原来由User传递过来的各项参数attrs,最好原模原样的返回,

21        # 如果不返回,有可能下面的数据描述符不起作用

22        # 除此之外,我们可以往里面添加我们自定义的参数

23        attrs["db_table"] = db_table

24        attrs["fields"] = fields

25        return super().__new__(cls, name, bases, attrs)

至此,我们的简易ORM就已经成型。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/489105.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《构建之法》 读书笔记(4)

敏捷流程 敏捷开发原则.尽早并持续地交付有价值的软件以满足顾客需求。敏捷流程欢迎需求的变化,并利用这种变化来提高用户的竞争优势。.经常发布可用的软件,发布间隔可以从几周到几个月,能短则短。.业务人员和开发人员在项目开发过程中应该每…

【周末阅读】10年内,19个关键技术将改变世界

来源:人民网 中国AI协会【导读】根据世界经济论坛全球议程理事会关于《未来软件与社会》的一份报告,到2025年前,很多新兴技术将会达到其临界点。该理事会调查走访了超过800名来自技术行业的企业高管和专家,并分享了他们关于这些黑…

linux mysql安装_LINUX 安装 MYSQL

安装成功的样子:[[email protected] mysql]# mysql -u root -pEnter password:Welcome to the MySQL monitor. Commands end with ; or \g.Your MySQL connection id is 18130Server version: 8.0.19 MySQL Community Server - GPLCopyright (c) 2000, 2020, Oracle…

上传文件实时显示网速怎么实现_担心PC程序吃内存占网速?有了它终于放心了...

自乔帮主发布iPhone至今,智能手机已经走过了13个年头,其系统、UI、操作逻辑都已经逐步完善且相当傻瓜化。所以总有许多人切换到PC上时,总感觉缺了那么点东西,Windows系统的生产力自然是毋庸置疑,但其繁琐、复杂且不直观…

Jmeter生成html格式测试报告

使用jmeter进行性能测试,运行完毕后生成html格式的测试报告,需要进行如下操作: 1.在C:\apache-jmeter-3.0\bin文件夹下的user.properties文本中添加如下信息: jmeter.save.saveservice.assertion_results_failure_messagetrue jme…

python字符串怎么用_零基础如何使用python处理字符串?

摘要:Python的普遍使用场景是自动化测试、爬取网页数据、科学分析之类,这其中都涉及到了对数据的处理,而数据的表现形式很多,今天我们来讲讲字符串的操作。字符串是作为任意一门编程语言的基础,在Python中的关键字叫做…

iOS NSString 与NSData转化

1. 字符串转Data NSString * str "str"; NSData *data [str dataUsingEncoding:NSUTF8StringEncoding]; 2.NSData 转NSString NSString * str [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]; 3.data 转char NSData *data; char * ha…

python eval函数格式_Python函数中eval函数知识点

python 的集合类型和 其他语言类似, 是一个无序不重复元素集,我在之前学过的其他的语言好像没有见过这个类型,基本功能包括关系测试和消除重复元素.集合对象还支持union(联合), intersection(交), difference(差)和sysmmetricdifference(对称差集)等数学运算,和我们…

为什么在python中整数的值没有限制_python-为什么math.log接受大整数值?

我终于钻入python math lib source code,发现了这一点:/* A decent logarithm is easy to compute even for huge ints, but libm cantdo that by itself -- loghelper can. func is log or log10, and name is"log" or "log10". Note that ove…

汽车芯片科普

来源:湖杉资本智能驾驶涉及人机交互、视觉处理、智能决策等,核心是 AI 算法和芯片。伴随汽车电子化提速,汽车半导体加速成长,2017 年全球市场规模 288 亿美元(26%),远高于整车销量增速&#xff…

python程序打包时出现lib not found_pyinstaller打包py脚本Warning:lib not found等相关问题...

小爬从使用Pyinstaller打包py为exe文件以来,一直都会碰到Warning:lib not found等相关问题,诸如:虽然大多数时候,您像我一样忽略这些warning,打包后的exe也能运行。网上有一堆的方法,有建议装wi…

hanoi塔java_Java实现hanoi塔

学习数据结构绕不过会遇到汉诺塔的问题,这个也比较经典,问题我也不详细描述了。简单来说三根木桩,初始状态下一根木桩上叠放着n个盘子,大盘在下小盘在上,任你操作,把这些盘子移动到第三根木桩,当…

信息太多,时间太少: 大脑如何区分重要和不重要的事?

来源:脑与心智毕生发展研究中心CLIMB我们总是会错过一些想看到、听到或感觉到的东西——尤其是当有大量信息争夺我们的注意时。是什么吸引了我们的注意?记住重要的事情需要多长时间?神经科学家(研究大脑和行为的科学家)发现,大脑的…

Java缓存框架使用EhCache结合Spring AOP

Java缓存框架使用EhCache结合Spring AOP 一.Ehcache简介 EhCache是一个纯Java的进程内缓存框架,具有如下特点: 1. 快速简单,非常容易和应用集成。 2.支持多种缓存策略 。 3. 缓存数据有两级:内存和磁盘,因…

python vba 秒出结果_为什么公司每天用20个人手工处理数据出报告,也不用5个会vba,python的人来制作??...

逼呼上无数人反复在说:先问是不是,再问为什么!不是没有道理啊!!!可能题主所在单位是这种情况,你看到处理数据出报告的人都不会VBA,也不会python,但是并不是所有单位都这样…

java 并发 set_高并发下的Java数据结构(List、Set、Map、Queue)

1.并发ListVector 或者 CopyOnWriteArrayList 是两个线程安全的List实现,ArrayList 不是线程安全的。因此,应该尽量避免在多线程环境中使用ArrayList。如果因为某些原因必须使用的,则需要使用Collections.synchronizedList(List list)进行包装…

知识图谱和图分析与可视化

来源:知链数据“知识图谱和图分析与可视化”这个题目看起来比较大,我尝试基于本人的一些图数据可视化与分析经验,对知识图谱和图分析与可视化之间的关系进行简单梳理,并分享一些以知识图谱为代表的图数据与图可视化、图分析结合进…

python 生成器装饰器_4.python迭代器生成器装饰器

基本概念1.容器(container)容器是一种把多个元素组织在一起的数据结构,容器中的元素可以逐个地迭代获取,可以用in, not in关键字判断元素是否包含在容器中。通常这类数据结构把所有的元素存储在内存中(也有一些特例,并不是所有的元素都放在内…

java开源对象池_JAVA 对象池

GenericObjectPool利用一个org.apache.commons.collections.CursorableLinkedList对象来保存对象池里的对象。这种对象池的特色是:可以设定最多能从池中借出多少个对象。可以设定池中最多能保存多少个对象。可以设定在池中已无对象可借的情况下,调用它的…

新一代人工智能专利分析

来源:三思派人工智能(Artificial Intelligence,AI)自诞生以来,已经过约60年的发展。2006年深度学习算法的重大突破带来了人工智能的第三次爆发。同时也引发专利申请的激增,2006年至2016年,十年的…