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

简介: 存档: 1 # -*- coding: utf-8 -*- 2 class Field(object): 3 4 def __init__(self, name, column_type): 5 self.

存档:

 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]

您可以考虑给博主来个小小的打赏以资鼓励,您的肯定将是我最大的动力。thx.

微信打赏

微信账号 nzf6698

支付宝打赏

支付宝账号 18979406698


作  者: Angel_Kitty
出  处:http://www.cnblogs.com/ECJTUACM-873284962/
关于作者:潜心机器学习以及信息安全的综合研究。如有问题或建议,请多多赐教!
版权声明:本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。
特此声明:所有评论和私信都会在第一时间回复。也欢迎园子的大大们指正错误,共同进步。或者直接私信
声援博主:如果您觉得文章对您有帮助,可以点击右下角推荐推荐一下该博文。您的鼓励是作者坚持原创和持续写作的最大动力!

目录
相关文章
|
7月前
|
Java 编译器
【javaSE】面向对象程序三大特性之封装(一)
【javaSE】面向对象程序三大特性之封装(一)
|
7月前
|
存储 Java 编译器
【javaSE】面向对象程序三大特性之封装(二)
【javaSE】面向对象程序三大特性之封装(二)
|
7月前
|
Java 程序员
【javaSE】 面向对象程序三大特性之继承(一)
【javaSE】 面向对象程序三大特性之继承(一)
|
7月前
|
Java 编译器
【javaSE】 面向对象程序三大特性之继承(二)
【javaSE】 面向对象程序三大特性之继承(二)
|
7月前
|
存储 前端开发 Java
带你掌握框架的灵魂——反射技术
带你掌握框架的灵魂——反射技术
|
4月前
|
设计模式 Java 数据库连接
设计模式与面向对象编程:举例说明在Java中应用工厂模式的场景,并编写一个简单的工厂模式实现。编写一个Java装饰器,用于添加日志记录功能到现有方法上。
设计模式与面向对象编程:举例说明在Java中应用工厂模式的场景,并编写一个简单的工厂模式实现。编写一个Java装饰器,用于添加日志记录功能到现有方法上。
23 0
|
5月前
|
Java
Java接口:实现多重继承,促进代码复用与扩展的强大工具
Java接口:实现多重继承,促进代码复用与扩展的强大工具
逆向工程的Example类用法
逆向工程的Example类用法
59 0
逆向工程的Example类用法
|
设计模式 SQL 存储
Mybatis 使用的 9 种设计模式,太有用了~
Mybatis 使用的 9 种设计模式,太有用了~
183 0
Mybatis 使用的 9 种设计模式,太有用了~