Django框架自带了ORM,实现了一些比较强大并且方便的查询功能,这些功能和表无关。好比下面这个例子:web
class Question(models.Model): question_text = models.CharField(max_length=200) pub_date = models.DateTimeField('date published')
Question.objects.all()
Question.objects.get(pk=1)sql从例子能够看出,
objects.all
和objects.get
这些功能都不是在class Question
中定义的,可能在其父类models.Model
中定义,也可能不是。那么咱们在web.py中如何实现这样的功能呢?(若是你选择使用SQLAlchemy就不须要本身实现了)。数据库
咱们注意到Question.objects.all()
这样的调用是直接访问了类属性objects
,并调用了objects
属性的方法all()
。这里objects
多是一个实例,也多是一个类。我我的认为(我没看过Django的实现)这应该是一个实例,由于实例化的过程能够传递一些表的信息,使得相似all()
这样的函数能够工做。通过分析以后,咱们能够列出咱们须要解决的问题:框架
须要实现一个模型的父类Model
,实际的表能够从这个父类继承以得到本身没有定义的功能。函数
实际的模型类(好比Question类)定义后,不实例话的状况下就要具有objects.all()
这样的查询效果。code
从上面的需求能够看出,咱们须要在类定义的时候就实现这些功能,而不是等到类实例化的时候再实现这些功能。类定义的时候实现功能?这不就是metaclass(元类)作的事情嘛。所以实现过程大概是下面这样的:sqlite
实现一个Model
类,其绑定方法和表的增、删、改有关。对象
修改Model
类的元类为ModelMetaClass
,该元类定义的过程当中为类增长一个objects
对象,该对象是一个ModelDefaultManager
类的实例,实现了表的查询功能。继承
都说不给代码就是耍流氓,我仍是给吧。说明下:使用的数据库操做都是web.py的db库中的接口。接口
# -*- coding: utf-8 -*- import web import config # 自定义的配置类,能够忽略 def _connect_to_db(): return web.database(dbn="sqlite", db=config.dbname) def init_db(): db = _connect_to_db() for statement in config.sql_statements: db.query(statement) class ModelError(Exception): """Exception raised by all models. Attributes: msg: Error message. """ def __init__(self, msg=""): self.msg = msg def __str__(self): return "ModelError: %s" % self.msg class ModelDefaultManager(object): """ModelManager implements query functions against a model. Attributes: cls: The class to be managed. """ def __init__(self, cls): self.cls = cls self._table_name = cls.__name__.lower() def all(self): db = _connect_to_db() results = db.select(self._table_name) return [self.cls(x) for x in results] def get(self, query_vars, where): results = self.filter(query_vars, where, limit=1) if len(results) > 0: return results[0] else: return None def filter(self, query_vars, where, limit=None): db = _connect_to_db() try: results = db.select(self._table_name, vars=query_vars, where=where, limit=limit) except (Exception) as e: raise ModelError(str(e)) return [self.cls(x) for x in results] class ModelMetaClass(type): def __new__(cls, classname, bases, attrs): new_class = super(ModelMetaClass, cls).__new__(cls, classname, bases, attrs) objects = ModelDefaultManager(new_class) setattr(new_class, "objects", objects) return new_class class Model(object): """Parent class of all models. """ __metaclass__ = ModelMetaClass def __init__(self): pass def _table_name(self): return self.__class__.__name__.lower() def insert(self, **kargs): db = _connect_to_db() try: with db.transaction(): db.insert(self._table_name(), **kargs) except (Exception) as e: raise ModelError(str(e)) def delete(self, where, using=None, vars=None): db = _connect_to_db() try: with db.transaction(): db.delete(self._table_name(), where, vars=vars) except (Exception) as e: raise ModelError(str(e)) def save(self, where, vars=None, **kargs): db = _connect_to_db() try: with db.transaction(): db.update(self._table_name(), where, vars, **kargs) except (Exception) as e: raise ModelError(str(e))
首先定义表对应的类:
class Users(Model): ...
使用就和Django的方式同样:
user_list = Users.objects.all()