使用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 相关文章推荐
Python中使用logging模块代替print(logging简明指南)
Jul 09 Python
讲解Python中的递归函数
Apr 27 Python
使用Python编写一个最基础的代码解释器的要点解析
Jul 12 Python
windows上安装Anaconda和python的教程详解
Mar 28 Python
python利用lxml读写xml格式的文件
Aug 10 Python
Python编程实现双链表,栈,队列及二叉树的方法示例
Nov 01 Python
分数霸榜! python助你微信跳一跳拿高分
Jan 08 Python
解决Python logging模块无法正常输出日志的问题
Feb 21 Python
基于pycharm实现批量修改变量名
Jun 02 Python
Python实现查找数据库最接近的数据
Jun 08 Python
如何基于Python爬取隐秘的角落评论
Jul 02 Python
用python批量下载apk
Dec 29 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
基于mysql的论坛(4)
2006/10/09 PHP
php中Ctype函数用法详解
2014/12/09 PHP
你应该知道PHP浮点数知识
2015/05/13 PHP
Laravel实现ORM带条件搜索分页
2019/10/24 PHP
JS模拟多线程
2007/02/07 Javascript
如何简单地用YUI做JavaScript动画
2007/03/10 Javascript
javascript 动态数据下的锚点错位问题解决方法
2008/12/24 Javascript
jQuery+css+html实现页面遮罩弹出框
2013/03/21 Javascript
你的 mixin 真的兼容 ECMAScript 5 吗?
2013/04/11 Javascript
jQuery怎么解析Json字符串(Json格式/Json对象)
2013/08/09 Javascript
jQuery大于号(>)选择器的作用解释
2015/01/13 Javascript
Bootstrap幻灯片轮播图支持触屏左右手势滑动的实现方法
2016/10/13 Javascript
JavaScript Ajax实现异步通信
2016/12/14 Javascript
轻松理解Javascript变量的相关问题
2017/01/20 Javascript
Vue filters过滤器的使用方法
2017/07/14 Javascript
BackBone及其实例探究_动力节点Java学院整理
2017/07/14 Javascript
Angular js 实现添加用户、修改密码、敏感字、下拉菜单的综合操作方法
2017/10/24 Javascript
fastadmin中调用js的方法
2019/05/14 Javascript
详解nuxt 微信公众号支付遇到的问题与解决
2019/08/26 Javascript
React传值 组件传值 之间的关系详解
2019/08/26 Javascript
JavaScript遍历数组的方法代码实例
2020/01/14 Javascript
[03:07]【DOTA2亚洲邀请赛】我们,梦开始的地方
2017/03/07 DOTA
[52:40]完美世界DOTA2联赛PWL S2 Magma vs GXR 第一场 11.29
2020/12/02 DOTA
Python命令行参数解析模块getopt使用实例
2015/04/13 Python
Python学习小技巧之利用字典的默认行为
2017/05/20 Python
关于python列表增加元素的三种操作方法
2018/08/22 Python
python实现密码强度校验
2020/03/18 Python
django中ImageField的使用详解
2020/12/21 Python
Yves Rocher伊夫·黎雪美国官网:法国始创植物美肌1959
2019/01/09 全球购物
美国气象仪器、花园装饰和墙壁艺术商店:Wind & Weather
2019/05/29 全球购物
Nicole Miller官方网站:纽约女装品牌
2019/09/14 全球购物
5s标语大全
2014/06/23 职场文书
学校运动会广播稿范文
2014/10/02 职场文书
小学生必读成语故事大全:送给暑假的你们
2019/07/09 职场文书
浅谈Python基础之列表那些事儿
2021/05/11 Python
redis数据结构之压缩列表
2022/03/21 Redis