目錄
一、模塊與包回顧
二、反序列化校驗源碼分析(了解)
三、斷言
四、drf之請求
【1】源碼分析
【2】配置視圖類能處理的編碼格式
五、drf之響應
【1】源碼
【2】響應編碼格式
一、模塊與包回顧
- 模塊與包
- 什么是模塊?
- 一個py文件,被別的py文件導入使用,它就是模塊
- 如果py文件,直接右鍵運行,它叫腳本文件
- 什么是包?
- 一個文件夾,下有 __init__.py ,和很多py文件,這個就是包
- 什么是模塊?
導入模塊或包使用的規則
'''0 導入模塊有相對導入和絕對導入,絕對的路徑是從環境變量開始的1 導入任何模塊,如果使用絕對導入,都是從環境變量開始導入起 import xx #### xx所在路徑必須在環境變量from yy import ####yy所在路徑必須在環境變量中2 腳本文件執行的路徑,會自動加入環境變量3 相對導入的話,是從當前py文件開始計算的4 以腳本運行的文件,不能使用相對導入,只能用絕對導入5 pycharm 會把項目根路徑加入到環境變量---> 離開pycharm就不行了6 我們看到第三方包,都是絕對導入form 它自己的包名 import xx下載的第三方包,都在site-package中,而site-package一定在環境變量中
'''
二、反序列化校驗源碼分析(了解)
1 ?執行 ser.is_valid() 就會執行 反序列化的校驗 --->字段自己 --> 局部鉤子 ---> 全局鉤子
2 ?入口是:ser.is_valid() ---> BaseSerializer 找到了
????1 自己寫的BookSerializer ---> serializer.Serializer ----> BaseSerializer?
????2 源碼如下
def is_valid(self, *, raise_exception=False):# self 是 ser對象---》自己寫的BookSerializer的對象--》一開始沒有# 一旦有了,就不執行了,優化is_valid被多次調用,只會走一次校驗if not hasattr(self, '_validated_data'):try:# 一旦執行過,以后self中就有_validated_data# 接下來看self.run_validation(self.initial_data)self._validated_data = self.run_validation(self.initial_data)except ValidationError as exc:self._validated_data = {}self._errors = exc.detailelse:self._errors = {}if self._errors and raise_exception:raise ValidationError(self.errors)return not bool(self._errors)
?????3 self.run_validation(self.initial_data) ---> serializer.Serializer類的,不要按住ctrl點擊,否則會進 Field 類,看錯了
??????4 serializer.Serializer類的run_validation
def run_validation(self, data=empty):# data前端傳入的--{"name":"張三","age":68}# value是---》前端傳入的,字段自己校驗通過的字典---{"name":"張三","age":68}value = self.to_internal_value(data) # 執行局部鉤子try:self.run_validators(value) # 先不看,忽略掉# self 是 BookSerializer的對象,如果我們寫了全局鉤子,走我們自己的,如果沒寫,走父類的,父類 的根本沒做校驗# value={"name":"張三","age":68}value = self.validate(value)# 執行全局鉤子except (ValidationError, DjangoValidationError) as exc:raise ValidationError(detail=as_serializer_error(exc))return value
??????5 全局鉤子讀完了:self 是 BookSerializer的對象,如果我們寫了全局鉤子,走我們自己的,如果沒寫,走父類的,父類的根本沒做校驗
? ? ? 6 局部鉤子:value = self.to_internal_value(data)--》Serializer類的
? ? ? ? ?for循環著去BookSerializer的對象中反射 ?validate_字段名的方法,如果有就執行,沒有就不執行
def to_internal_value(self, data):for field in fields: # 序列化類中所有字段類的對象 name=CharField()# self 是BookSerializer類的對象# 去BookSerializer類中,反射 validate_field字段類的對象.field_namevalidate_method = getattr(self, 'validate_' + field.field_name, None)try:# 如果能拿到,說明咱么寫了局部鉤子if validate_method is not None:# 執行局部鉤子--》傳入了當前字段的value值validated_value = validate_method(validated_value)except ValidationError as exc:# 如果拋異常,會被捕獲errors[field.field_name] = exc.detailexcept DjangoValidationError as exc:errors[field.field_name] = get_error_detail(exc)except SkipField:passelse:set_value(ret, field.source_attrs, validated_value)if errors:raise ValidationError(errors)return ret
##### 讀了局部和全局鉤子的執行位置 #####
"保存,修改也好,都要用validated_data,它是最準確的"
三、斷言
a = 10# assert 后寫條件,只要不符合條件,就會拋AssertionError異常,后面寫異常信息
assert a == 11, ("不等于11,報錯了")# 等同于--->上面只要一行代碼,源碼中喜歡用
if not a == 11:raise Exception('不等于11,報錯了')# 源碼中使用
assert value is not None, '.validate() should return the validated data'
四、drf之請求
# Request 類的對象
from rest_framework.request import Request
【1】源碼分析
1 新的request
2 request.data 前端傳入的請求體中得數據,無論那種編碼
3 用起來跟之前一樣
4 老的request在request._request
【2】配置視圖類能處理的編碼格式
# 默認,視圖類的方法,可以處理任意編碼格式-urlencoded-form-data-json# 比如有的接口,只能接收json格式,其他格式都不能處理# 配置方式一:視圖類上配置
from rest_framework.parsers import JSONParser, FormParser, MultiPartParser
# JSONParser:json
# FormParser:urlencoded
# MultiPartParser:form-data
class TestView(APIView):# parser_classes = [JSONParser]parser_classes = [JSONParser,FormParser]def post(self, request):print(request.data)return Response('ok')
# 配置方式二:settings.py 配置文件中配置## 所有drf的配置,都要寫在REST_FRAMEWORK字典中 ##REST_FRAMEWORK = {'DEFAULT_PARSER_CLASSES': [# 'rest_framework.parsers.JSONParser',# 'rest_framework.parsers.FormParser','rest_framework.parsers.MultiPartParser',],}# 全局使用某種,單某個視圖類,局部使用---> 優先用視圖類配置的---> 視圖類沒配置--> 項目配置文件 ---> 項目配置文件如果沒配置---> drf配置文件(默認三個都支持)-配置文件正常寫class TestView(APIView):parser_classes = [JSONParser,FormParser]# 一般我們做---> 只支持json
# 對于上傳的文件的接口---》單獨配置 只允許 form-data
五、drf之響應
from rest_framework.response import Response
【1】源碼
data=None # 咱們給的字典或列表或字符串---> 最終放到了http響應體中返回了status=None # http響應狀態碼,默認是200,你可以改,改成from rest_framework import status 狀態碼
"http響應狀態碼分別代表啥意思:200成功 201創建成功"template_name=None:用瀏覽器訪問好看的頁面--> 指定的--> 默認使用drf提供的--> 后期可以自己寫頁面,使用
headers=None, # 響應頭
content_type=None # 響應編碼格式"""我們需要記住的
data
status
headers
"""
【2】響應編碼格式
# 配置方式跟請求解析類似# 方式一:在視圖類上配置
class TestView(APIView):renderer_classes = [JSONRenderer,BrowsableAPIRenderer]# 方式二:配置文件中配置
REST_FRAMEWORK = {'DEFAULT_RENDERER_CLASSES': ['rest_framework.renderers.JSONRenderer','rest_framework.renderers.BrowsableAPIRenderer',],}# 優先用 視圖類的---> 項目配置文件---> drf內置的(兩個都支持)