Python使用描述器實現ORM模型的方法詳解

訪問或者修改描述器對象的屬性時無法觸發__setattr__等方法,隻會觸發描述器類內部的__set__,__get__,__delete__方法.

ORM模型:類名對應表名,對象對應的數據行,類屬性對應數據行的各字段,有幾個表字段,就綁定幾個類屬性;往表中增加數據就是創建對象,每創建一個對象,就是增加一行數據記錄。

ORM框架的功能:

1.建立模型類和表之間的對應關系,允許我們通過面向對象的方式操作數據庫。

2.根據設計的模型類生成數據庫中的表格。

3.通過方便的配置就可以進行數據庫的切換。

MySql的常用數據類型:

1.整數:int,bit

2.小數:decimal(decimal表示浮點數,decimal(5,2)表示共計5位數,小數占2位)

3.字符串:varchar(可變長度),char(不可變長度)

4.日期時間:date,time,datetime

5.枚舉類型:enum

模型類案例:

"""
django的ORM模型字段
BooleanField:   佈爾字段,True或False
CharField(max_length=最大長度):     字符串,參數max_length表示最大字符個數
IntegerField:   整數
"""
class TestReport(BaseTable):
    class Meat:
        verbose_name = '測試報告'
        db_table = "TestReport"
    report_name = models.CharField(max_length=40, null=False)
    start_at = models.CharField(max_length=40, null=True)
    status = models.BooleanField()
    testRun = models.IntegerField()
    successes = models.IntegerField()
    reports = models.TextField()

1.__set__方法:設置屬性 

"""
隻要一個類中出現瞭:
    __get__(self,instance,owner)
    __set__(self,instance,value)
    __delete__(self,instance)
    __set_name__(self,owner,name)
    中的任意一個方法,這個類就不是一個普通的類瞭,
    應被稱為:
        描述器類(應用在ORM模型中)
"""
class Field(object):
    """
    隻要一個類中出現瞭以下任意一個方法,這個類就是一個描述器類
    """
    def __get__(self, instance, owner):
        pass
    def __set__(self, instance, value):
        print('---set---方法被觸發瞭')
    def __delete__(self, instance):
        pass
class Model(object):
    attr = Field()  # attr是一個描述器對象,修改時無法觸發__setattr__等方法,
                    # 隻能觸發描述器類內部的__set__方法
if __name__ == '__main__':
    m = Model()
    m.attr = 666    # 嘗試修改attr屬性
    print(m.attr)  
    # 輸出:
    ---set - --方法被觸發瞭
    None
 
class Field(object):
    """
    隻要一個類中出現瞭以下任意一個方法,這個類就是一個描述器類
    """
    def __get__(self, instance, owner):
        pass
    def __set__(self, instance, value):
        """
        :param instance: 修改的對象
        :param value: 修改的值
        :return:
        """
        print('---set---方法被觸發瞭')
        self.value = value
        print(self)      # <__main__.Field object at 0x7fecc01f8a30>
        print(instance)  # <__main__.Model object at 0x7fecc01f8a00>
        print(value)     # 666
    def __delete__(self, instance):
        pass
class Model(object):
    attr = Field()  # attr是一個描述器對象,修改時無法觸發__setattr__等方法,
                    # 隻能觸發描述器類內部的__set__方法
if __name__ == '__main__':
    m = Model()
    print(m)      # <__main__.Model object at 0x7fecc01f8a00>,與instance相同
    m.attr = 666  # 嘗試修改attr屬性
 

2.__get__方法:訪問屬性

 
"""
隻要一個類中出現瞭:
    __get__(self,instance,owner)
    __set__(self,instance,value)
    __delete__(self,instance)
    __set_name__(self,owner,name)
    中的任意一個方法,這個類就不是一個普通的類瞭,
    應被稱為:
        描述器類(應用在ORM模型中)
"""
class Field(object):
    """
    隻要一個類中出現瞭以下任意一個方法,這個類就是一個描述器類
    """
    def __get__(self, instance, owner):
        print('---get---方法被觸發瞭')
    def __set__(self, instance, value):
        """
        :param instance: 修改的對象
        :param value: 修改的值
        :return:
        """
        print('---set---方法被觸發瞭')
        self.value = value
    def __delete__(self, instance):
        pass
class Model(object):
    attr = Field()  # attr是一個描述器對象,修改時無法觸發__setattr__等方法,
                    # 隻能觸發描述器類內部的__set__方法
if __name__ == '__main__':
    m = Model()
    m.attr = 666  # 嘗試修改attr屬性
    print(m.attr)
    # 輸出:
    ---set - --方法被觸發瞭
    ---get - --方法被觸發瞭
    None
 
"""
隻要一個類中出現瞭:
    __get__(self,instance,owner)
    __set__(self,instance,value)
    __delete__(self,instance)
    __set_name__(self,owner,name)
    中的任意一個方法,這個類就不是一個普通的類瞭,
    應被稱為:
        描述器類(應用在ORM模型中)
"""
class Field(object):
    """
    隻要一個類中出現瞭以下任意一個方法,這個類就是一個描述器類
    """
    def __get__(self, instance, owner):
        print('---get---方法被觸發瞭')
        print(instance)     # <__main__.Model object at 0x7f80b81a09d0>
        print(owner)        # <class '__main__.Model'>
        return self.value
    def __set__(self, instance, value):
        """
        :param instance: 修改的對象
        :param value: 修改的值
        :return:
        """
        print('---set---方法被觸發瞭')
        self.value = value
    def __delete__(self, instance):
        pass
class Model(object):
    attr = Field()  # attr是一個描述器對象,修改時無法觸發__setattr__等方法,
                    # 隻能觸發描述器類內部的__set__方法
if __name__ == '__main__':
    m = Model()
    m.attr = 666  # 嘗試修改attr屬性
    print(m.attr)  # 666
    # 輸出:
    ---set - --方法被觸發瞭
    ---get - --方法被觸發瞭
    < __main__.Model object at 0x7f80b81a09d0 >
    < class '__main__.Model'>
    666
 

3.__delete__方法:刪除屬性

"""
隻要一個類中出現瞭:
    __get__(self,instance,owner)
    __set__(self,instance,value)
    __delete__(self,instance)
    __set_name__(self,owner,name)
    中的任意一個方法,這個類就不是一個普通的類瞭,
    應被稱為:
        描述器類(應用在ORM模型中)
"""
class Field(object):
    """
    隻要一個類中出現瞭以下任意一個方法,這個類就是一個描述器類
    """
    def __get__(self, instance, owner):
        return self.value
    def __set__(self, instance, value):
        """
        :param instance: 修改的對象
        :param value: 修改的值
        :return:
        """
        print('---set---方法被觸發瞭')
        self.value = value
    def __delete__(self, instance):
        print('---delete---方法被觸發瞭')
class Model(object):
    attr = Field()  # attr是一個描述器對象,修改時無法觸發__setattr__等方法,
                    # 隻能觸發描述器類內部的__set__方法
if __name__ == '__main__':
    m = Model()
    m.attr = 666  # 嘗試修改attr屬性
    del m.attr    # ---delete---方法被觸發瞭
    print(m.attr)  # None
    # 輸出:
    ---set - --方法被觸發瞭
    ---delete - --方法被觸發瞭
    666
 
"""
隻要一個類中出現瞭:
    __get__(self,instance,owner)
    __set__(self,instance,value)
    __delete__(self,instance)
    __set_name__(self,owner,name)
    中的任意一個方法,這個類就不是一個普通的類瞭,
    應被稱為:
        描述器類(應用在ORM模型中)
"""
class Field(object):
    """
    隻要一個類中出現瞭以下任意一個方法,這個類就是一個描述器類
    """
    def __get__(self, instance, owner):
        print('---get---方法被觸發瞭')
        print(instance)     # <__main__.Model object at 0x7f80b81a09d0>
        print(owner)        # <class '__main__.Model'>
        return self.value
    def __set__(self, instance, value):
        """
        :param instance: 修改的對象
        :param value: 修改的值
        :return:
        """
        print('---set---方法被觸發瞭')
        self.value = value
    def __delete__(self, instance):
        print('---delete---方法被觸發瞭')
        print(instance)     # <__main__.Model object at 0x7ff61806a160>
        self.value = None
class Model(object):
    attr = Field()  # attr是一個描述器對象,修改時無法觸發__setattr__等方法,
                    # 隻能觸發描述器類內部的__set__方法
if __name__ == '__main__':
    m = Model()
    m.attr = 666  # 嘗試修改attr屬性
    del m.attr    # ---delete---方法被觸發瞭
    print(m.attr)  # None
    # 輸出:
    ---set - --方法被觸發瞭
    ---delete - --方法被觸發瞭
    < __main__.Model object at 0x7ff61806a160 >
    ---get - --方法被觸發瞭
    < __main__.Model object at 0x7ff61806a160 >
    <class '__main__.Model'>
    None
 

 4.描述器實現ORM模型: 

"""
通過描述器實現ORM模型
"""
class CharField(object):
    def __init__(self,max_length=20):
        self.max_length = max_length
    def __get__(self, instance, owner):
        return self.value
    def __set__(self, instance, value):
        # 首先判斷是否為空
        if value is not None:
            # 再判斷是否是字符串
            if isinstance(value, str):
                # 再判斷長度是否符合要求
                if len(value) <= self.max_length:
                    self.value = value
                else:
                    raise TypeError('length need less than {}'.format(self.max_length))
            else:
                raise TypeError('need a str')
        else:
            raise TypeError("can not be None")
    def __delete__(self, instance):
        self.value = None
class IntegerField(object):
    def __get__(self, instance, owner):
        return self.value
    def __set__(self, instance, value):
        # 首先判斷是否為空
        if value is not None:
            # 再判斷是否是整數int
            if isinstance(value, int):
                self.value = value
            else:
                raise TypeError('need a int')
        else:
            raise TypeError("can not be None")
    def __delete__(self, instance):
        self.value = None
class UserModel(object):
    # 定義用戶信息的模型類
    name = CharField(max_length=20)  # 定義:name隻能賦值為字符串
    pwd = CharField(max_length=40)
    age = IntegerField()    # 定義:age隻能賦值為整數
if __name__ == '__main__':
    user = UserModel()
    user.name = "春田"
    print(user.name)    # 輸出: 春田
    user.age = 130
    print(user.age)  # 輸出: 130
    user.pwd = 'wsdgdgdrgerdsfs方式范德薩發阿瑟費薩法 sfa fda fsdf sdf  fg'
    print(user.pwd)     # 輸出: TypeError: length need less than 40
 

總結

本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!  

推薦閱讀: