使用Python的web.py框架实现类似Django的ORM查询的教程


Posted in Python onMay 02, 2015

Django中的对象查询

Django框架自带了ORM,实现了一些比较强大而且方便的查询功能,这些功能和表无关。比如下面这个例子:

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)

从例子可以看出,objects.all和objects.get这些功能都不是在class Question中定义的,可能在其父类models.Model中定义,也可能不是。那么我们在web.py中如何实现这样的功能呢?(如果你选择使用SQLAlchemy就不需要自己实现了)。
实现
思路

我们注意到Question.objects.all()这样的调用是直接访问了类属性objects,并调用了objects属性的方法all()。这里objects可能是一个实例,也可能是一个类。我个人认为(我没看过Django的实现)这应该是一个实例,因为实例化的过程可以传递一些表的信息,使得类似all()这样的函数可以工作。经过分析之后,我们可以列出我们需要解决的问题:

  •     需要实现一个模型的父类Model,实际的表可以从这个父类继承以获得自己没有定义的功能。
  •     实际的模型类(比如Question类)定义后,不实例话的情况下就要具备objects.all()这样的查询效果。
  • 从上面的需求可以看出,我们需要在类定义的时候就实现这些功能,而不是等到类实例化的时候再实现这些功能。类定义的时候实现功能?这不就是metaclass(元类)做的事情嘛。因此实现过程大概是下面这样的:
  •     实现一个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()

 

Python 相关文章推荐
fastcgi文件读取漏洞之python扫描脚本
Apr 23 Python
浅谈python装饰器探究与参数的领取
Dec 01 Python
基于Python中numpy数组的合并实例讲解
Apr 04 Python
使用python读取txt文件的内容,并删除重复的行数方法
Apr 18 Python
django输出html内容的实例
May 27 Python
Flask之flask-script模块使用
Jul 26 Python
解决pycharm安装后代码区不能编辑的问题
Oct 28 Python
Python3实现爬取简书首页文章标题和文章链接的方法【测试可用】
Dec 11 Python
python web自制框架之接受url传递过来的参数实例
Dec 17 Python
python 实现矩阵上下/左右翻转,转置的示例
Jan 23 Python
django迁移数据库错误问题解决
Jul 29 Python
python的faker库用法
Nov 28 Python
在ironpython中利用装饰器执行SQL操作的例子
May 02 #Python
用Python编写简单的定时器的方法
May 02 #Python
用Python程序抓取网页的HTML信息的一个小实例
May 02 #Python
在Mac OS上部署Nginx和FastCGI以及Flask框架的教程
May 02 #Python
在Python的Django框架中用流响应生成CSV文件的教程
May 02 #Python
详细解读Python中的__init__()方法
May 02 #Python
举例讲解Python的Tornado框架实现数据可视化的教程
May 02 #Python
You might like
PHP 配置文件中open_basedir选项作用
2009/07/19 PHP
zf框架的Filter过滤器使用示例
2014/03/13 PHP
百度工程师讲PHP函数的实现原理及性能分析(一)
2015/05/13 PHP
实例说明js脚本语言和php脚本语言的区别
2019/04/04 PHP
jquery调用wcf并展示出数据的方法
2011/07/07 Javascript
推荐40个简单的 jQuery 导航插件和教程(下篇)
2012/09/14 Javascript
JS自定义功能函数实现动态添加网址参数修改网址参数值
2013/08/02 Javascript
Js制作简单弹出层DIV在页面居中 中间显示遮罩的具体方法
2013/08/08 Javascript
JavaScript中的Primitive对象封装介绍
2014/12/31 Javascript
详解jQuery向动态生成的内容添加事件响应jQuery live()方法
2015/11/02 Javascript
jQuery CSS3自定义美化Checkbox实现代码
2016/05/12 Javascript
JS表格组件BootstrapTable行内编辑解决方案x-editable
2016/09/01 Javascript
nodejs集成sqlite使用示例
2017/06/05 NodeJs
使用jQuery实现简单的tab框实例
2017/08/22 jQuery
Vue-router 类似Vuex实现组件化开发的示例
2017/09/15 Javascript
基于vue.js中事件修饰符.self的用法(详解)
2018/02/23 Javascript
vue单页应用在页面刷新时保留状态数据的方法
2018/09/21 Javascript
JavaScript原型对象原理与应用分析
2018/12/27 Javascript
JavaScript函数式编程(Functional Programming)纯函数用法分析
2019/05/22 Javascript
关于NodeJS中的循环引用详解
2019/07/23 NodeJs
jquery选择器和属性对象的操作实例分析
2020/01/10 jQuery
JavaScript实现沿五角星形线摆动的小圆实例详解
2020/07/28 Javascript
Python探索之创建二叉树
2017/10/25 Python
Python输出由1,2,3,4组成的互不相同且无重复的三位数
2018/02/01 Python
python内置函数sorted()用法深入分析
2019/10/08 Python
python爬取王者荣耀全皮肤的简单实现代码
2020/01/31 Python
用python批量移动文件
2021/01/14 Python
使用Python webdriver图书馆抢座自动预约的正确方法
2021/03/04 Python
kfc实习自我鉴定
2013/12/14 职场文书
会计电算化专业毕业生推荐信
2013/12/24 职场文书
初中化学教学反思
2014/01/23 职场文书
三八活动策划方案
2014/08/17 职场文书
租车协议书范本2014
2014/11/17 职场文书
2015世界地球日活动总结
2015/02/09 职场文书
创业计划书之校园超市
2019/09/12 职场文书
mybatis使用oracle进行添加数据的方法
2021/04/27 Oracle