使用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中设置变量访问权限的方法
Apr 27 Python
Python输出9*9乘法表的方法
May 25 Python
python写入已存在的excel数据实例
May 03 Python
python处理csv中的空值方法
Jun 22 Python
对Python发送带header的http请求方法详解
Jan 02 Python
解决Python对齐文本字符串问题
Aug 28 Python
python如何判断IP地址合法性
Apr 05 Python
使用keras实现非线性回归(两种加激活函数的方式)
Jul 05 Python
Numpy实现卷积神经网络(CNN)的示例
Oct 09 Python
Python加载数据的5种不同方式(收藏)
Nov 13 Python
教你怎么用PyCharm为同一服务器配置多个python解释器
May 31 Python
在Python中如何使用yield
Jun 07 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超级全局变量
2010/01/26 PHP
php header示例代码(推荐)
2010/09/08 PHP
php基于环形链表解决约瑟夫环问题示例
2017/11/07 PHP
laravel 模型查询按照whereIn排序的示例
2019/10/16 PHP
jQuery对象和DOM对象相互转化
2009/04/24 Javascript
JavaScript 编写匿名函数的几种方法
2010/02/21 Javascript
Javascript事件热键兼容ie|firefox
2010/12/30 Javascript
IE下使用cloneNode注意事项分享
2012/11/22 Javascript
javascript动态判断html元素并执行不同的操作
2014/06/16 Javascript
js在数组中删除重复的元素自保留一个(两种实现思路)
2014/08/22 Javascript
jQuery实现简单隔行变色的方法
2016/02/20 Javascript
javascript cookie的简单应用
2016/02/24 Javascript
Node.js插件安装图文教程
2016/05/06 Javascript
JS简单实现表格排序功能示例
2016/12/20 Javascript
解决OneThink中无法异步提交kindeditor文本框中修改后的内容方法
2017/05/05 Javascript
Mobile Web开发基础之四--处理手机设备的横竖屏问题
2017/08/11 Javascript
微信小程序实现MUI数字输入框效果
2018/01/31 Javascript
使用weixin-java-miniapp配置进行单个小程序的配置详解
2019/03/29 Javascript
优化Vue中date format的性能详解
2020/01/13 Javascript
整理 node-sass 安装失败的原因及解决办法(小结)
2020/02/19 Javascript
JavaScript 引用类型实例详解【数组、对象、严格模式等】
2020/05/13 Javascript
[35:44]2014 DOTA2华西杯精英邀请赛 5 24 iG VS VG
2014/05/26 DOTA
python中readline判断文件读取结束的方法
2014/11/08 Python
Python paramiko模块的使用示例
2018/04/11 Python
Python 实现某个功能每隔一段时间被执行一次的功能方法
2018/10/14 Python
Python实现分段线性插值
2018/12/17 Python
在pytorch中查看可训练参数的例子
2019/08/18 Python
tensorflow中tf.slice和tf.gather切片函数的使用
2020/01/19 Python
python Selenium 库的使用技巧
2020/10/16 Python
韩国三星旗下的一家超市连锁店:Home Plus
2016/07/30 全球购物
eBay澳大利亚站:eBay.com.au
2018/02/02 全球购物
幼儿园清明节活动总结
2014/07/04 职场文书
监考失职检讨书
2015/01/26 职场文书
公司文体活动总结
2015/05/07 职场文书
交通安全月活动总结
2015/05/08 职场文书
对讲机的最大通讯距离是多少
2022/02/18 无线电