使用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中变量交换的例子
Aug 25 Python
python获取一组汉字拼音首字母的方法
Jul 01 Python
Python多线程下载文件的方法
Jul 10 Python
python编程之requests在网络请求中添加cookies参数方法详解
Oct 25 Python
python简单图片操作:打开\显示\保存图像方法介绍
Nov 23 Python
Python获取指定文件夹下的文件名的方法
Feb 06 Python
对python读取zip压缩文件里面的csv数据实例详解
Feb 08 Python
Python入门Anaconda和Pycharm的安装和配置详解
Jul 16 Python
django云端留言板实例详解
Jul 22 Python
Django admin.py 在修改/添加表单界面显示额外字段的方法
Aug 22 Python
django使用F方法更新一个对象多个对象字段的实现
Mar 28 Python
Python趣味入门教程之循环语句while
Aug 26 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
谏山创故乡大分县日田市水坝将设立《进击的巨人》立艾伦、三笠以及阿尔敏的铜像!
2020/03/06 日漫
PHP基础学习之流程控制的实现分析
2013/04/28 PHP
PHP处理大量表单字段的便捷方法
2015/02/07 PHP
微信公众号判断用户是否已关注php代码解析
2016/06/24 PHP
PHP中set_include_path()函数相关用法分析
2016/07/18 PHP
php中的依赖注入实例详解
2019/08/14 PHP
php抽象方法和普通方法的区别点总结
2019/10/13 PHP
自用js开发框架小成 学习js的朋友可以看看
2010/11/16 Javascript
JavaScript高级程序设计 DOM学习笔记
2011/09/10 Javascript
jquery 列表双向选择器之改进版
2013/08/09 Javascript
node.js中的events.emitter.removeListener方法使用说明
2014/12/10 Javascript
JavaScript检查弹出窗口是否被阻拦的方法技巧
2015/03/13 Javascript
如何提高Dom访问速度
2017/01/05 Javascript
javascript实现简易计算器
2017/02/01 Javascript
seajs下require书写约定实例分析
2018/05/16 Javascript
Angular5中调用第三方库及jQuery的添加的方法
2018/06/07 jQuery
学习React中ref的两个demo示例
2018/08/14 Javascript
js canvas画布实现高斯模糊效果
2018/11/27 Javascript
JS实现页面跳转与刷新的方法汇总
2019/08/30 Javascript
js实现点击图片在屏幕中间弹出放大效果
2019/09/11 Javascript
vue设置一开始进入的页面教程
2019/10/28 Javascript
[04:31]2016国际邀请赛中国区预选赛妖精采访
2016/06/27 DOTA
[47:26]完美世界DOTA2联赛 LBZS vs Forest 第二场 11.07
2020/11/09 DOTA
python 中的int()函数怎么用
2017/10/17 Python
Python探索之实现一个简单的HTTP服务器
2017/10/28 Python
python实现excel读写数据
2021/03/02 Python
Python爬虫实现全国失信被执行人名单查询功能示例
2018/05/03 Python
Django如何将URL映射到视图
2019/07/29 Python
使用phonegap克隆和删除联系人的实现方法
2017/03/31 HTML / CSS
AmazeUI 导航条的实现示例
2020/08/14 HTML / CSS
科茨沃尔德家居商店:Scotts of Stow
2018/06/29 全球购物
.NET概念性的面试题
2012/02/29 面试题
大学生优秀团员事迹材料
2014/01/30 职场文书
幼儿园中秋节活动反思
2014/02/16 职场文书
2014年统计工作总结
2014/11/21 职场文书
mysql多表查询-笔记七
2021/04/05 MySQL