web.py中实现类似Django中的ORM的查询效果

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() 这样的函数可以工作。经过分析之后,我们可以列出我们需要解决的问题:

  1. 需要实现一个模型的父类 Model ,实际的表可以从这个父类继承以获得自己没有定义的功能。
  2. 实际的模型类(比如 Question 类)定义后,不实例话的情况下就要具备 objects.all() 这样的查询效果。

从上面的需求可以看出,我们需要在类定义的时候就实现这些功能,而不是等到类实例化的时候再实现这些功能。 类定义的时候实现功能 ?这不就是 metaclass (元类)做的事情嘛。因此实现过程大概是下面这样的:

  1. 实现一个 Model 类,其绑定方法和表的增、删、改有关。
  2. 修改 Model 类的元类为 ModelMetaClass ,该元类定义的过程中为类增加一个 objects 对象,该对象是一个 ModelDefaultManager 类的实例,实现了表的查询功能。

代码

都说不给代码就是耍流氓,我还是给吧。说明下:使用的数据库操作都是web.py的db库中的接口。

Python	# -*- 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()
时间: 2024-09-27 05:50:10

web.py中实现类似Django中的ORM的查询效果的相关文章

自定义模板方法(类似django中的simple_tag) | Tornado

# 自定义模板方法|相当于django中的simple_tag以及filter: # 1.uimethods.py def tag_one(self): print(self) return 'tags' # 2.uimodules.py from tornado.web import UIModule from tornado import escape class Custom(UIModule): def embedded_css(self): # 页面添加css样式 return 'bo

SQLServer 中实现类似MySQL中的group_concat函数的功能

SQLServer中没有MySQL中的group_concat函数,可以把分组的数据连接在一起. 后在网上查找,找到了可以实现此功能的方法,特此记录下. SELECT a, stuff((SELECT ',' + b FROM #tb WHERE a = t.a FOR xml path('')), 1, 1, '' )AS b from  # tb AS t GROUP BY a; 先对a列进行分组,对分组中的b以Xml形式输出,再使用stuff将开关多出的,删掉. 具体实现参考:http:/

Django中的会话技术(Cookie,Session,Token)

一.Cookie 客户端技术,将数据信息存储到浏览器中,存储的结构是字典结构,即key-value. Cookie是服务端创建,但保存于客户端,客户端每次发送请求时都会将Cookie信息发送到服务器(因为Cookie是请求头信息的一部分) Cookie不支持中文,不能跨浏览器,不能跨域名 1.设置cookie HttpResponse.set_cookie() response.set_signed_cookie("Cookie名称","Cookie值",salt=

go语言如何实现类似c++中的多态功能

go语言作为编程语言中的后起之秀,在博采众长的同时又不失个性,在注重运行效率的同时又重视开发效率,不失为一种好的开发语言.在go语言中,没有类的概念,但是仍然可以用struct+interface来实现类的功能,下面的这个简单的例子演示了如何用go来模拟c++中的多态的行为. package main import "os" import "fmt" type Human interface { sayHello() } type Chinese struct {

django中怎么使用mysql数据库的事务

Mysql数据库事务: 在进行后端业务开始操作修改数据库时,可能会涉及到多张表的数据修改,对这些数据的修改应该是一个整体事务,即要么一起成功,要么一起失败. Django中对于数据库的事务,默认每执行一句数据库操作,便会自动提交.我们需要在保存数据库操作中自己控制数据库事务的执行流程. 在Django中可以通过django.db.transaction模块提供的atomic来定义一个事务,atomic提供两种用法: 装饰器用法 from django.db import transaction

使用web.py 搭建服务器

有很多python的web框架,web.py是一个轻量级Python web框架.她并不是使用很多的一个. 但并不妨碍她的简单实用.搭建一个嵌入式web服务器最好不过. 下面把笔者搭建过程做一个介绍: 1. 下载python 2.7.8 ,做交叉编译,需要依据自己的平台做修改: ./configure make python Parser/pgen mv python python_for_build;mv Parser/pgen Parser/pgen_for_build make distc

django框架基础-ORM操作-长期维护-20191213

###############    ORM介绍和使用mysql的基本配置    ################ # ORM简介 # O是object,对象 # R是relation,关系,这是关系数据库中的表 # M是mapping,映射 # 在django里面主要是在models.py文件里面设计模型类, ########################### # ORM另一个作用:根据设计的类生成数据库中的表 # django中使用ORM创建表 # 这种可以使用写Python语句,然后自

django中settings.py中变量的全局引用

在settings.py中添加自定义变量,可以通过setting.(点)变量名的方式访问,如: from django.conf import settings site_name = settings.SITE_NAME site_desc = settings.SITE_DESC 但是,如果遇到了一些频繁访问的变量,如:邮箱,网站标题,网站的描述,这样访问就很不方便,解决方法: 1.首先在settings.py中添加对应的变量: #网站信息 SITE_NAME="hupeng的个人博客&qu

CentOS6.5环境中配置Python + Web.py + Apache部署环境

1. 安装apache:    yum install -y httpd httpd-devel     修改/etc/httpd/conf/httpd.conf中的servername等配置,使http://*:*/ 能正常访问 2. 安装python:    wget https://www.python.org/ftp/python/2.7.8/Python-2.7.8.tgz        tar -xzvf Python-2.7.8.tgz    cd Python-2.7.8