1. 程式人生 > 程式設計 >在django中自定義欄位Field詳解

在django中自定義欄位Field詳解

Django的Field類中方法有:

to_python() # 把資料庫資料轉成python資料

from_db_value() # 把資料庫資料轉成python資料

get_pre_value() # 把python資料壓縮準備存入資料庫

get_db_pre_value() # 把壓縮好的資料轉成資料庫查詢集

get_prep_lookup() # 指定過濾的條件

value_to_string() # 資料序列化

如果建立的Field比字串,日期,整數等更復雜的資料結構,可能需要重寫to_python 和from_db_value()方法(Django提供的一個SubfileBase元類,他在賦值時總是呼叫to_python()

一個簡單的ListField 欄位型別

# _*_ coding:utf-8 _*_
from django.db import models
import ast


class ListField(models.TextField):
  """自定義list欄位
  models.SubfieldBase  提供to_python  和 from_db_value
  把資料庫資料轉化成python資料
  現在主要是from_db_value 方法 把資料庫資料轉化成python資料
  to_python 主要是接受form表單
  """
  __metacalss__ = models.SubfieldBase
  description = 'Stores a python list'

  

def __init__(self,*args,**kwargs):

    super(ListField,self).__init__(*args,**kwargs)

  # def db_type(self,connection):
  #   if connection.setting_dict['ENGINE'] == 'django.db.backends.mysql':
  #     return 'listtype'
  

 def from_db_value(self,value,expression,connection,context):

  """資料庫資料轉成python資料"""

    if value is None:
      value=[]
      return value
    if isinstance(value,list):
      return value
    return ast.literal_eval(value)


  def to_python(self,value):
    """從資料庫中讀取的資料轉成python
    eval(value)讀取value原來的型別
    ast模組就是幫助Python應用來處理抽象的語法解析的。
    而該模組下的literal_eval()函式:
    則會判斷需要計算的內容計算後是不是合法的python型別,
    如果是則進行運算,否則就不進行運算。
    """
    if not value:
      value = []
    if isinstance(value,list):
      return value
    return ast.literal_eval(value)

  def get_prep_value(self,value):
    """
    把python資料壓縮後儲存到資料庫
    或者說把python物件轉化成查詢值
    返回值是個字串
    :param value:
    :return:
    """
    if value is None:
      return value
    return str(value)

  # def get_db_prep_value(self,prepared=False):
  #   """把查詢集資料轉化成資料庫值  一般不需要重寫 只需要覆蓋"""
  #   value = super(ListField,self).get_db_prep_value()
  #   if value is not None:
  #     return connection.Database.Binary(value)
  #   return value
  #

  def get_prep_lookup(self,lookup_type,value):
    """限制查詢方式"""
    if lookup_type == 'exact':
      return value
    elif lookup_type == 'in':
      return [self.get_prep_value(v) for v in value]
    else:
      return TypeError('lookup type %r not supported'%lookup_type)
  

def value_to_string(self,obj):

    """轉換欄位資料以進行序列化
    Field._get_val_from_obj(obj) 是獲取值序列化的最佳方式
    """
    value = self._get_val_from_obj(obj)
    return self.get_db_prep_value(value)

以上這篇在django中自定義欄位Field詳解就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支援我們。