成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

web.py中實現(xiàn)類似Django中的ORM的查詢效果

bladefury / 1772人閱讀

摘要:中的對象查詢框架自帶了,實現(xiàn)了一些比較強大而且方便的查詢功能,這些功能和表無關(guān)。實際的模型類比如類定義后,不實例話的情況下就要具備這樣的查詢效果。

Django中的對象查詢

Django框架自帶了ORM,實現(xiàn)了一些比較強大而且方便的查詢功能,這些功能和表無關(guān)。比如下面這個例子:

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.allobjects.get這些功能都不是在class Question中定義的,可能在其父類models.Model中定義,也可能不是。那么我們在web.py中如何實現(xiàn)這樣的功能呢?(如果你選擇使用SQLAlchemy就不需要自己實現(xiàn)了)。

實現(xiàn) 思路

我們注意到Question.objects.all()這樣的調(diào)用是直接訪問了類屬性objects,并調(diào)用了objects屬性的方法all()。這里objects可能是一個實例,也可能是一個類。我個人認為(我沒看過Django的實現(xiàn))這應該是一個實例,因為實例化的過程可以傳遞一些表的信息,使得類似all()這樣的函數(shù)可以工作。經(jīng)過分析之后,我們可以列出我們需要解決的問題:

需要實現(xiàn)一個模型的父類Model,實際的表可以從這個父類繼承以獲得自己沒有定義的功能。

實際的模型類(比如Question類)定義后,不實例話的情況下就要具備objects.all()這樣的查詢效果。

從上面的需求可以看出,我們需要在類定義的時候就實現(xiàn)這些功能,而不是等到類實例化的時候再實現(xiàn)這些功能。類定義的時候?qū)崿F(xiàn)功能?這不就是metaclass(元類)做的事情嘛。因此實現(xiàn)過程大概是下面這樣的:

實現(xiàn)一個Model類,其綁定方法和表的增、刪、改有關(guān)。

修改Model類的元類為ModelMetaClass,該元類定義的過程中為類增加一個objects對象,該對象是一個ModelDefaultManager類的實例,實現(xiàn)了表的查詢功能。

代碼

都說不給代碼就是耍流氓,我還是給吧。說明下:使用的數(shù)據(jù)庫操作都是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()

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/37497.html

相關(guān)文章

  • Lunar, 一個Python網(wǎng)絡框架現(xiàn)

    摘要:核心的幾個組件模板引擎,框架,請求和應答的處理還是有一些難度,但是經(jīng)過一步步的分析和編碼還是能夠完成功能。模板引擎模板引擎是另外一個比較大和的模塊。 前前后后,大概兩個月的時間,lunar這個項目終于達到了一個很高的完整度。 Lunar是一個Python語言的網(wǎng)絡框架,類似于Django,F(xiàn)lask,Tornado等當下流行的web framework。最初有這個想法是在大二下學期,...

    邱勇 評論0 收藏0
  • Python_Django

    摘要:為了將代碼規(guī)范,約定俗成將視圖放置在項目或應用程序目錄中命名為文件中。必須接受字段表示字符串長度不能超過該值,默認的標簽最常用的。例,自動添加發(fā)布時間。以字符串的形式存在,默認最大長度,可以通過參數(shù)設置。用于保存通用唯一識別碼的字段。 MVC 大部分開發(fā)語言中都有MVC框架 MVC框架的核心思想是:解耦 降低各功能模塊之間的耦合性,方便變更,更容易重構(gòu)代碼,最大程度上實現(xiàn)代碼的重...

    BearyChat 評論0 收藏0
  • 通過demo學習OpenStack開發(fā)所需基礎知識 -- 數(shù)據(jù)庫(1)

    摘要:另外,項目在單元測試中使用的是的內(nèi)存數(shù)據(jù)庫,這樣開發(fā)者運行單元測試的時候不需要安裝和配置復雜的數(shù)據(jù)庫,只要安裝好就可以了。而且,數(shù)據(jù)庫是保存在內(nèi)存中的,會提高單元測試的速度。是實現(xiàn)層的基礎。項目一般會使用數(shù)據(jù)庫來運行單元測試。 OpenStack中的關(guān)系型數(shù)據(jù)庫應用 OpenStack中的數(shù)據(jù)庫應用主要是關(guān)系型數(shù)據(jù)庫,主要使用的是MySQL數(shù)據(jù)庫。當然也有一些NoSQL的應用,比如Ce...

    warnerwu 評論0 收藏0
  • Awesome Python

    摘要:漢字拼音 Awesome Python A curated list of awesome Python frameworks, libraries and software. Inspired by awesome-php. Awesome Python Environment Management Package Management Package Repositorie...

    fizz 評論0 收藏0
  • 個人博客三|首頁后臺開發(fā)

    摘要:聲明本渣渣部分代碼參考自其實有很多代碼是不需要自己一行行碼出來,生產(chǎn)力是第一位。只有研究型人才需要生產(chǎn)代碼,作為一名渣渣拿來用是最高效的做法。程序員都有一個開源的精神,碼出來的代碼本身是希望更多的人用到,應用到生產(chǎn)中。 聲明:本渣渣部分代碼參考自TendCode其實有很多代碼是不需要自己一行行碼出來,生產(chǎn)力是第一位。只有研究型人才需要生產(chǎn)代碼,作為一名渣渣拿來用是最高效的做法。程序員都...

    zorpan 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<