小白學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='[email protected]',password='my-pwd')
#保存到數據庫
u.save()

運行結果,並且報錯:

Traceback (most recent call last):
  File "*****************", line ***, in <module>
    u = User(id=12345, name='Michael', email='[email protected]',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>

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章