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的更多內容!