加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 编程开发 > Python > 正文

使用Python的web.py框架实现类似Django的ORM查询的教程

发布时间:2020-12-17 07:18:33 所属栏目:Python 来源:网络整理
导读:Django中的对象查询 Django框架自带了ORM,实现了一些比较强大而且方便的查询功能,这些功能和表无关。比如下面这个例子: class Question(models.Model): question_text = models.CharField(max_length=200) pub_date = models.DateTimeField('date publish

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,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,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,using=None,vars=None):
      db = _connect_to_db()
      try:
        with db.transaction():
          db.delete(self._table_name(),vars=vars)
      except (Exception) as e:
        raise ModelError(str(e))

    def save(self,vars=None,**kargs):
      db = _connect_to_db()
      try:
        with db.transaction():
          db.update(self._table_name(),vars,**kargs)
      except (Exception) as e:
        raise ModelError(str(e))

使用

首先定义表对应的类:

class Users(Model):
  ...

使用就和Django的方式一样:

>>> user_list = Users.objects.all()

 

您可能感兴趣的文章:

  • Django视图之ORM数据库查询操作API的实例
  • 利用Python的Django框架中的ORM建立查询API
  • 在Python的Django框架上部署ORM库的教程
  • Django ORM框架的定时任务如何使用详解
  • django 常用orm操作详解
  • Django中ORM表的创建和增删改查方法示例
  • Django学习笔记之ORM基础教程
  • python Django连接MySQL数据库做增删改查
  • 在Django的视图中使用数据库查询的方法
  • 在Python的Django框架中更新数据库数据的方法
  • 通过数据库对Django进行删除字段和删除模型的操作
  • Django基于ORM操作数据库的方法详解

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读