使用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操作摄像头截图实现远程监控的例子
Mar 25 Python
Python ldap实现登录实例代码
Sep 30 Python
Python数据操作方法封装类实例
Jun 23 Python
轻量级的Web框架Flask 中模块化应用的实现
Sep 11 Python
Python实现读取txt文件并画三维图简单代码示例
Dec 09 Python
Python实现找出数组中第2大数字的方法示例
Mar 26 Python
Python模拟自动存取款机的查询、存取款、修改密码等操作
Sep 02 Python
Python3实现取图片中特定的像素替换指定的颜色示例
Jan 24 Python
OpenCV模板匹配matchTemplate的实现
Oct 18 Python
Python *args和**kwargs用法实例解析
Mar 02 Python
Python 线性回归分析以及评价指标详解
Apr 02 Python
在Pycharm中安装Pandas库方法(简单易懂)
Feb 20 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
一个查看session内容的函数
2006/10/09 PHP
PHP中使用匿名函数操作数据库的例子
2014/11/17 PHP
使用PHP连接多种数据库的实现代码(mysql,access,sqlserver,Oracle)
2016/12/21 PHP
javascript 打印内容方法小结
2009/11/04 Javascript
JQuery自定义事件的应用 JQuery最佳实践
2010/08/01 Javascript
开发插件的两个方法jquery.fn.extend与jquery.extend
2013/11/21 Javascript
JS创建类和对象的两种不同方式
2014/08/08 Javascript
Javascript中innerHTML用法实例分析
2015/01/12 Javascript
Bootstrap fileinput文件上传预览插件使用详解
2017/05/16 Javascript
深入理解vue2.0路由如何配置问题
2017/07/18 Javascript
解决vue中el-tab-pane切换的问题
2020/07/19 Javascript
Vue 使用typescript如何优雅的调用swagger API
2020/09/01 Javascript
原生js实现九宫格拖拽换位
2021/01/26 Javascript
[01:00:25]2018DOTA2亚洲邀请赛3月30日 小组赛A组 VG VS Liquid
2018/03/31 DOTA
[01:11:32]VG vs FNATIC 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/17 DOTA
在Apache服务器上同时运行多个Django程序的方法
2015/07/22 Python
浅谈Python由__dict__和dir()引发的一些思考
2017/10/30 Python
Python3实现的画图及加载图片动画效果示例
2018/01/19 Python
Python运维之获取系统CPU信息的实现方法
2018/06/11 Python
tensorflow: variable的值与variable.read_value()的值区别详解
2018/07/30 Python
解决python3捕获cx_oracle抛出的异常错误问题
2018/10/18 Python
pycharm 将python文件打包为exe格式的方法
2019/01/16 Python
python使用pipeline批量读写redis的方法
2019/02/18 Python
Python2比较当前图片跟图库哪个图片相似的方法示例
2019/09/28 Python
python 读写文件包含多种编码格式的解决方式
2019/12/20 Python
Python 实现平台类游戏添加跳跃功能
2020/03/27 Python
解决Python数据可视化中文部分显示方块问题
2020/05/16 Python
优秀团员个人事迹材料
2014/01/29 职场文书
幼儿园教学管理制度
2014/02/04 职场文书
毕业设计说明书
2014/05/07 职场文书
我们的节日春节活动方案
2014/08/22 职场文书
民间借贷协议书范本
2014/10/01 职场文书
网站文案策划岗位职责
2015/04/14 职场文书
2015大学迎新标语
2015/07/16 职场文书
Oracle笔记
2021/04/05 Oracle
CSS3实现的侧滑菜单
2021/04/27 HTML / CSS