使用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+mysql实现简单的web程序
Sep 11 Python
Python实现的检测网站挂马程序
Nov 30 Python
Python获取邮件地址的方法
Jul 10 Python
11月编程语言排行榜 Python逆袭C#上升到第4
Nov 15 Python
python3读取excel文件只提取某些行某些列的值方法
Jul 10 Python
python八皇后问题的解决方法
Sep 27 Python
Python函数装饰器实现方法详解
Dec 22 Python
python接口自动化(十七)--Json 数据处理---一次爬坑记(详解)
Apr 18 Python
解决pycharm 工具栏Tool中找不到Run manager.py Task的问题
Jul 01 Python
解决Mac下使用python的坑
Aug 13 Python
Python自动化导出zabbix数据并发邮件脚本
Aug 16 Python
PyQt+socket实现远程操作服务器的方法示例
Aug 22 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实现倒计时功能
2020/11/16 PHP
Jquery 获取表单text,areatext,radio,checkbox,select值的代码
2009/11/12 Javascript
jquery $.ajax相关用法分享
2012/03/16 Javascript
JS过滤url参数特殊字符的实现方法
2013/12/24 Javascript
js左右弹性滚动对联广告代码分享
2014/02/19 Javascript
Javascript模块化编程详解
2014/12/01 Javascript
Bootstrap实现默认导航栏效果
2020/09/21 Javascript
JavaScript实现Base64编码转换
2016/04/23 Javascript
AngularJS 中的数据源的循环输出
2017/10/12 Javascript
详解开发react应用最好用的脚手架 create-react-app
2018/04/24 Javascript
vue-cli 引入jQuery,Bootstrap,popper的方法
2018/09/03 jQuery
使用taro开发微信小程序遇到的坑总结
2019/04/08 Javascript
Nautil 中使用双向数据绑定的实现
2019/10/02 Javascript
JavaScript实现Tab选项卡切换
2020/02/13 Javascript
Python+Opencv识别两张相似图片
2020/03/23 Python
Python编写登陆接口的方法
2017/07/10 Python
windows下python和pip安装教程
2018/05/25 Python
关于python中密码加盐的学习体会小结
2019/07/15 Python
python操作gitlab API过程解析
2019/12/27 Python
对Matlab中共轭、转置和共轭装置的区别说明
2020/05/11 Python
html5自定义video标签的海报与播放按钮功能
2019/12/04 HTML / CSS
英国豪华家具和家居用品购物网站:Teddy Beau
2020/10/12 全球购物
解决方案设计综合面试题
2015/08/31 面试题
活动总结结尾怎么写
2014/08/30 职场文书
查摆问题整改措施范文
2014/10/11 职场文书
2015国庆节66周年演讲稿
2015/03/20 职场文书
收入证明怎么写
2015/06/12 职场文书
2019年年中职场激励人心语录30条
2019/08/07 职场文书
创业计划书之少年玩具店
2019/09/05 职场文书
2021年pycharm的最新安装教程及基本使用图文详解
2021/04/03 Python
python字符串常规操作大全
2021/05/02 Python
MySQL系列之七 MySQL存储引擎
2021/07/02 MySQL
MySQL索引是啥?不懂就问
2021/07/21 MySQL
Java并发编程之原子性-Atomic的使用
2022/03/16 Java/Android
Vue+TypeScript中处理computed方式
2022/04/02 Vue.js
vscode内网访问服务器的方法
2022/06/28 Servers