Python元類編程實(shí)現(xiàn)一個(gè)簡單的ORM
概述
什么是ORM?
ORM全稱“Object Relational Mapping”,即對象-關(guān)系映射,就是把關(guān)系數(shù)據(jù)庫的一行映射為一個(gè)對象,也就是一個(gè)類對應(yīng)一個(gè)表,這樣,寫代碼更簡單,不用直接操作SQL語句。
現(xiàn)在我們就要實(shí)現(xiàn)簡易版ORM。
效果
class Person(Model):
"""
定義類的屬性到列的映射
"""
pid = IntegerField('id')
names = StringField('username')
email = StringField('email')
password = StringField('password')
p = Person(pid=10086, names='曉明', email='10086@163.com', password='123456')
p.save()
通過執(zhí)行save()方法 動(dòng)態(tài)生成sql插入語句, 是不是很神奇, 那我們現(xiàn)在開始解析原理吧
步驟
首先我們要定義一個(gè) Field 類 它負(fù)責(zé)保存數(shù)據(jù)庫表的字段名和字段類型:
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)
在 Field 的基礎(chǔ)上,進(jìn)一步定義各種類型的 Field,比如 StringField,IntegerField 等等:
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')
下一步,就是編寫最復(fù)雜的 ModelMetaclass:
class ModelMetaclass(type):
def __new__(cls, name, bases, attrs):
if name == "Model":
return type.__new__(cls, name, bases, attrs)
mappings = dict()
print("Found class: %s" % name)
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["__table__"] = name # 表名和類名一致
attrs["__mappings__"] = mappings # 保存屬性和列的映射關(guān)系
return type.__new__(cls, name, bases, attrs)最后就是基類 Model:
class Model(metaclass=ModelMetaclass):
def __init__(self, **kwargs):
_setattr = setattr
if kwargs:
for k, v in kwargs.items():
_setattr(self, k, v)
super(Model, self).__init__()
def save(self):
fields = []
params = []
args = []
for k, v in self.__mappings__.items():
fields.append(k)
params.append("?")
args.append(getattr(self, k, None))
sql = "insert into %s (%s) values (%s)" % (self.__table__, ','.join(fields), ",".join(params))
print('插入語句: %s' % sql)
print('參數(shù): %s' % str(args))
def update(self):
fields = []
args = []
for k, v in self.__mappings__.items():
if getattr(self, k, None):
fields.append(k+"=?")
args.append(getattr(self, k, None))
sql = "update %s set %s" % (self.__table__, ','.join(fields))
print("更新語句: %s " % sql)
print("參數(shù): %s" % args)
def filter(self, *args):
pass
def delete(self):
pass當(dāng)用戶定義一個(gè) class Person(Model) 繼承父類時(shí),Python解釋器會(huì)在當(dāng)前類 Person 的定義中找 __metaclass__,如果沒有找到,就繼續(xù)到父類中找 __metaclass__,實(shí)在找不到就用默認(rèn) type 類。
我們在父類 Model 中定義了 __metaclass__ 的 ModelMetaclass 來創(chuàng)建 Person 類,所以 metaclass 隱式地繼承到子類。
在 ModelMetaclass 中,一共做了幾件事情:
- 排除掉對 Model 類的修改;
- 在當(dāng)前類(比如 Person )中查找定義的類的所有屬性,如果找到一個(gè) Field 屬性,就把它保存到一個(gè) __mappings__ 的dict中,同時(shí)從類屬性中刪除該Field屬性,否則,容易造成運(yùn)行時(shí)錯(cuò)誤;
- 把表名保存到 __table__ 中,這里簡化為表名默認(rèn)為類名。
在Model類中,就可以定義各種操作數(shù)據(jù)庫的方法,比如save(),delete(),find(),update() 等等。
我們實(shí)現(xiàn)了save(), update()方法,把一個(gè)實(shí)例保存到數(shù)據(jù)庫中。因?yàn)橛斜砻?,屬性到字段的映射和屬性值的集合,就可以?gòu)造出INSERT語句和UPDATE語句。
編寫代碼試試:
class UserInfo(Model):
"""
定義類的屬性到列的映射
"""
uid = IntegerField('uid')
name = StringField('username')
email = StringField('email')
password = StringField('password')
class Person(Model):
"""
定義類的屬性到列的映射
"""
pid = IntegerField('id')
names = StringField('username')
email = StringField('email')
password = StringField('password')
p = Person(pid=10086, names='曉明', email='10086@163.com', password='123456')
p.save()
u2 = UserInfo(password='123456')
u2.update()輸出
Found class: UserInfo
Found mapping: uid ==> <IntegerField:uid>
Found mapping: name ==> <StringField:username>
Found mapping: email ==> <StringField:email>
Found mapping: password ==> <StringField:password>
Found class: Person
Found mapping: pid ==> <IntegerField:id>
Found mapping: names ==> <StringField:username>
Found mapping: email ==> <StringField:email>
Found mapping: password ==> <StringField:password>
插入語句: insert into Person (pid,names,email,password) values (?,?,?,?)
參數(shù): [10086, '曉明', '10086@163.com', '123456']
更新語句: update UserInfo set password=?
參數(shù): ['123456']
結(jié)束語
就這樣一個(gè)小巧的ORM就這么完成了。是不是學(xué)到了很多呢 ?這里利用的是元編程,很多Python框架都運(yùn)用了元編程達(dá)到動(dòng)態(tài)操作類。
注:上述代碼列子 結(jié)合了廖雪峰的列子和少量的django ORM源碼。
完整代碼
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)
mappings = dict()
print("Found class: %s" % name)
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["__table__"] = name # 表名和類名一致
attrs["__mappings__"] = mappings # 保存屬性和列的映射關(guān)系
return type.__new__(cls, name, bases, attrs)
class Model(metaclass=ModelMetaclass):
def __init__(self, **kwargs):
_setattr = setattr
if kwargs:
for k, v in kwargs.items():
_setattr(self, k, v)
super(Model, self).__init__()
def save(self):
fields = []
params = []
args = []
for k, v in self.__mappings__.items():
fields.append(k)
params.append("?")
args.append(getattr(self, k, None))
sql = "insert into %s (%s) values (%s)" % (self.__table__, ','.join(fields), ",".join(params))
print('插入語句: %s' % sql)
print('參數(shù): %s' % str(args))
def update(self):
fields = []
args = []
for k, v in self.__mappings__.items():
if getattr(self, k, None):
fields.append(k+"=?")
args.append(getattr(self, k, None))
sql = "update %s set %s" % (self.__table__, ','.join(fields))
print("更新語句: %s " % sql)
print("參數(shù): %s" % args)
def filter(self, *args):
pass
def delete(self):
pass
class UserInfo(Model):
"""
定義類的屬性到列的映射
"""
uid = IntegerField('uid')
name = StringField('username')
email = StringField('email')
password = StringField('password')
class Person(Model):
"""
定義類的屬性到列的映射
"""
pid = IntegerField('id')
names = StringField('username')
email = StringField('email')
password = StringField('password')
p = Person(pid=10086, names='曉明', email='10086@163.com', password='123456')
p.save()
u2 = UserInfo(password='123456')
u2.update()到此這篇關(guān)于Python元類編程實(shí)現(xiàn)一個(gè)簡單的ORM的文章就介紹到這了,更多相關(guān)Python元類編程實(shí)現(xiàn)ORM內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python使用xlsxwriter實(shí)現(xiàn)有向無環(huán)圖到Excel的轉(zhuǎn)換
這篇文章主要為大家詳細(xì)介紹了python使用xlsxwriter實(shí)現(xiàn)有向無環(huán)圖到Excel的轉(zhuǎn)換,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12
python2.7無法使用pip的解決方法(安裝easy_install)
下面小編就為大家分享一篇python2.7無法使用pip的解決方法(安裝easy_install),具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-04-04
Python數(shù)據(jù)標(biāo)準(zhǔn)化的實(shí)例分析
在本篇文章里小編給大家整理了關(guān)于Python數(shù)據(jù)標(biāo)準(zhǔn)化的實(shí)例內(nèi)容,有需要的朋友們可以測試學(xué)習(xí)下。2021-08-08
python自動(dòng)化測試Data?Driven?Testing(DDT)用例解析
這篇文章主要為大家介紹了python自動(dòng)化測試Data?Driven?Testing(DDT)用例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09

