用Metaclass实现一个精简的ORM框架

存档:

复制代码
 1 # -*- coding: utf-8 -*-
 2 class Field(object):
 3 
 4     def __init__(self, name, column_type):
 5         self.name = name
 6         self.column_type = column_type
 7 
 8     def __str__(self):
 9         return '<%s:%s>' % (self.__class__.__name__, self.name)
10 class StringField(Field):
11 
12     def __init__(self, name):
13         super(StringField, self).__init__(name, 'varchar(100)')
14 
15 class IntegerField(Field):
16 
17     def __init__(self, name):
18         super(IntegerField, self).__init__(name, 'bigint')
19 class ModelMetaclass(type):
20 
21     def __new__(cls, name, bases, attrs):
22         if name=='Model':
23             return type.__new__(cls, name, bases, attrs)
24         print('Found model: %s' % name)
25         mappings = dict()
26         for k, v in attrs.items():
27             if isinstance(v, Field):
28                 print('Found mapping: %s ==> %s' % (k, v))
29                 mappings[k] = v
30         for k in mappings.keys():
31             attrs.pop(k)
32         attrs['__mappings__'] = mappings # 保存属性和列的映射关系
33         attrs['__table__'] = name # 假设表名和类名一致
34         return type.__new__(cls, name, bases, attrs)
35 class Model(dict, metaclass=ModelMetaclass):
36 
37     def __init__(self, **kw):
38         super(Model, self).__init__(**kw)
39 
40     def __getattr__(self, key):
41         try:
42             return self[key]
43         except KeyError:
44             raise AttributeError(r"'Model' object has no attribute '%s'" % key)
45 
46     def __setattr__(self, key, value):
47         self[key] = value
48 
49     def save(self):
50         fields = []
51         params = []
52         args = []
53         for k, v in self.__mappings__.items():
54             fields.append(v.name)
55             params.append('?')
56             args.append(getattr(self, k, None))
57         sql = 'insert into %s (%s) values (%s)' % (self.__table__, ','.join(fields), ','.join(params))
58         print('SQL: %s' % sql)
59         print('ARGS: %s' % str(args))
60 class User(Model):
61     # 定义类的属性到列的映射:
62     id = IntegerField('id')
63     name = StringField('username')
64     email = StringField('email')
65     password = StringField('password')
66 # 创建一个实例:
67 u = User(id=12345, name='Michael', email='test@orm.org', password='my-pwd')
68 # 保存到数据库:
69 u.save()
复制代码

结果如下:

Found model: User
Found mapping: id ==> <IntegerField:id>
Found mapping: name ==> <StringField:username>
Found mapping: email ==> <StringField:email>
Found mapping: password ==> <StringField:password>
SQL: insert into User (id,username,email,password) values (?,?,?,?)
ARGS: [12345, 'Michael', 'test@orm.org', 'my-pwd']
[Finished in 0.2s]

posted @   Angel_Kitty  阅读(340)  评论(0编辑  收藏  举报
编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· [AI/GPT/综述] AI Agent的设计模式综述
历史上的今天:
2017-02-08 深入理解树状数组
2017-02-08 Codeforces 712C Memory and De-Evolution
点击右上角即可分享
微信分享提示
哥伦布
14°
14:09发布
哥伦布
14:09发布
14°
大雨
南风
3级
空气质量
相对湿度
93%
今天
中雨
14°/19°
周日
中雨
5°/19°
周一
1°/11°