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

目录
相关文章
|
8月前
|
存储 前端开发 Java
带你掌握框架的灵魂——反射技术
带你掌握框架的灵魂——反射技术
|
6天前
|
设计模式 Java
Java 设计模式:混合、装饰器与组合的编程实践
【4月更文挑战第27天】在面向对象编程中,混合(Mixins)、装饰器(Decorators)和组合(Composition)是三种强大的设计模式,用于增强和扩展类的功能。
11 1
|
5月前
|
设计模式 Java 数据库连接
设计模式与面向对象编程:举例说明在Java中应用工厂模式的场景,并编写一个简单的工厂模式实现。编写一个Java装饰器,用于添加日志记录功能到现有方法上。
设计模式与面向对象编程:举例说明在Java中应用工厂模式的场景,并编写一个简单的工厂模式实现。编写一个Java装饰器,用于添加日志记录功能到现有方法上。
23 0
|
6月前
|
Java
Java接口:实现多重继承,促进代码复用与扩展的强大工具
Java接口:实现多重继承,促进代码复用与扩展的强大工具
逆向工程的Example类用法
逆向工程的Example类用法
59 0
逆向工程的Example类用法
|
SQL XML Java
MyBatis框架:第七章:注解使用方式和参数传递及#{}和${}
MyBatis框架:第七章:注解使用方式和参数传递及#{}和${}
281 0
性能高、上手快,实体类转换工具 MapStruct 到底有多强大
1.什么是MapStruct 1.1 JavaBean 的困扰 对于代码中 JavaBean之间的转换, 一直是困扰我很久的事情。在开发的时候我看到业务代码之间有很多的 JavaBean 之间的相互转化, 非常的影响观感,却又不得不存在。我后来想的一个办法就是通过反射,或者自己写很多的转换器。 第一种通过反射的方法确实比较方便,但是现在无论是 BeanUtils, BeanCopier 等在使用反射的时候都会影响到性能。虽然我们可以进行反射信息的缓存来提高性能。但是像这种的话,需要类型和名称都一样才会进行映射,有很多时候,由于不同的团队之间使用的名词不一样,还是需要很多的手动 set/get
|
安全 Java 数据安全/隐私保护
Java反射(扩展)(五)
前几篇文章讲述Java反射核心功能与用法,基本的常用方法都已经囊括里面了,本篇针是对前几篇文章进行补充。
84 2
Java反射(扩展)(五)
【Groovy】MOP 元对象协议与元编程 ( Groovy 类内部和外部分别获取 metaClass | 分析获取 metaClass 操作的字节码 | HandleMetaClass 注入方法 )
【Groovy】MOP 元对象协议与元编程 ( Groovy 类内部和外部分别获取 metaClass | 分析获取 metaClass 操作的字节码 | HandleMetaClass 注入方法 )
166 0