小白学python之使用元类_学习笔记

本文以廖雪峰的官方网站为参考来学习python的。其学习链接为廖雪峰小白学python教程

本文是学习到python的定制类。参考链接廖雪峰python使用元类

本学习笔记仅供参考。

type()

笔记

动态语言和静态语言最大的不同,就是函数和类的定义。不是编译时定义的,而是运行时动态创建的。

def fn(self, name='world'): #先定义函数
    print('Hello, %s' %name)

Hello = type('Hello',(object,),dict(hello=fn)) #创建Hello class
h = Hello()
h.hello()
print(type(Hello))
print(type(h))

运行结果:

Hello, world
<class 'type'>
<class '__main__.Hello'>

笔记:

要创建一个class对象,type()函数一次传入3个参数:

1. class的名称;

2.继承的父类集合,注意Python支持多重继承,如果只有一个父类,别忘了tuple的单元素写法;

3.class的方法名称与函数绑定,这里我们把函数fn绑定到方法名hello上。

笔记

通过type()函数创建的类和直接写class是完全一样的,因为Python解释器遇到class时,仅仅是扫描了一下class定义的语法,然后调用type()函数创建出class

笔记

正常情况下,我们都用class Xxx...来定义类。但是,type()函数也允许我们动态创建出类出来。动态语言本身支持运行期动态创建类,这和静态语言有非常大的不同。要在静态语言运行期创建类,必须构造源代码字符串再调用编译器。或者借助一些工具生成字节码实现。本质上都是动态编译,会非常复杂。

metaclass


class Field(object):

    def __init__(self,name,column_type):
        self.name = name
        self.column_type = column_type

    def __str__(self):
        return '<%s:%s>' % (self.__class__.__name__, self.name)

class StringField(Field):

    def __init__(self,name):
        super(StringField, self).__init__(name,'varchar(100)')

class IntegerField(Field):

    def __init__(self,name):
        super(IntegerField, self).__init__(name, 'bigint')

class ModelMetaclass(type):

    def __new__(cls, name, bases, attrs):
        if name=='Model':
            return type.__new__(cls,name,bases,attrs)
        print('Found model: %s' %name)
        mappings = dict()
        for k, v in attrs.items():
            if isinstance(v,Field):
                print('Found mapping: %s ==> %s' % (k,v))
                mappings[k] = v
        for k in mappings.keys():
            attrs.pop(k)
        attrs['__mapping__'] = mappings  #保存属性和列的映射关系
        attrs['__table__'] = name #假设表名和类名一致

class Model(dict,metaclass = ModelMetaclass):

    def __init__(self,**kw):
        super(Model, self).__init__(**kw)

    def __getattr__(self, key):
        try:
            return self[key]
        except KeyError:
            raise AttributeError(r"'Model' object has no attribute '%s'" % key)

    def __setattr__(self,key,value):
        self[key] = value

    def save(self):
        fields = []
        params = []
        artgs = []
        for k, v in self.__mappings__.items():
            fields.append(v.name)
            params.append('?')
            args.append(getattr(self,k,None))

        sql = 'insert into %s (%s) values (%s)' %(self.__table__,','.join(fields),','.join(params))
        print('SQL: %s' % sql)
        print('ARGS: %s' % str(args))

class User(Model):
    #定义类的属性到列的映射
    id = IntegerField('id')
    name = StringField('username')
    email = StringField('email')
    password = StringField('password')

#创建一个实例
u = User(id=12345, name='Michael', email='test@orm.org',password='my-pwd')
#保存到数据库
u.save()

运行结果,并且报错:

Traceback (most recent call last):
  File "*****************", line ***, in <module>
    u = User(id=12345, name='Michael', email='test@orm.org',password='my-pwd')
TypeError: 'NoneType' object is not callable

Found model: User
Found mapping: id ==> <IntegerField:id>
Found mapping: name ==> <StringField:username>
Found mapping: email ==> <StringField:email>
Found mapping: password ==> <StringField:password>

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

碰珺

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值