自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

如何在 Python Web 框架 Django 中使用序列化器

開發(fā) 后端
序列化用于將數(shù)據(jù)轉(zhuǎn)換為方便存儲或傳輸?shù)母袷?,然后將其重新?gòu)建以供使用。DRF 是最具有知名的序列化器。

[[356442]]

序列化用于將數(shù)據(jù)轉(zhuǎn)換為方便存儲或傳輸?shù)母袷?,然后將其重新?gòu)建以供使用。DRF 是最具有知名的序列化器。

序列化是將數(shù)據(jù)轉(zhuǎn)換為可以存儲或傳輸?shù)母袷?,然后對其進行重新構(gòu)建的過程。在開發(fā)應(yīng)用程序或?qū)?shù)據(jù)存儲在數(shù)據(jù)庫、內(nèi)存或?qū)⑵滢D(zhuǎn)換為文件時,一直會用到它。

我最近幫助 Labcodes 的兩名初級開發(fā)人員理解序列化器,我想也可以與諸位讀者分享一下我的方法。

假設(shè)你正在編寫一個電子商務(wù)網(wǎng)站,你有一個訂單,該訂單記錄了某人在某個日期以某種價格購買了一個產(chǎn)品:

  1. class Order:
  2.     def __init__(self, product, customer, price, date):
  3.         self.product = product
  4.         self.customer = customer
  5.         self.price = price
  6.         self.date = date

現(xiàn)在,假設(shè)你想從一個鍵值數(shù)據(jù)庫中存儲和檢索訂單數(shù)據(jù)。幸運的是,它的接口可以接受和返回字典,因此你需要將對象轉(zhuǎn)換成字典:

  1. def serialize_order(order):
  2.     return {
  3.         'product': order.product,
  4.         'customer': order.customer,
  5.         'price': order.price,
  6.         'date': order.date
  7.     }

如果你想從數(shù)據(jù)庫中獲取一些數(shù)據(jù),你可以獲取字典數(shù)據(jù)并將其轉(zhuǎn)換為訂單對象(Order):

  1. def deserialize_order(order_data):
  2.     return Order(
  3.         product=order_data['product'],
  4.         customer=order_data['customer'],
  5.         price=order_data['price'],
  6.         date=order_data['date'],
  7.     )

這對于簡單的數(shù)據(jù)非常直接了當,但是當你需要處理一些由復(fù)雜屬性構(gòu)成的復(fù)雜對象時,這種方法就無法很好地擴展。你還需要處理不同類型字段的驗證,這需要手工完成大量工作。

此時框架的序列化可以很方便的派上用場。它們使你可以創(chuàng)建帶有少量模板的序列化器,這將適用于復(fù)雜的情況。

Django 提供了一個序列化模塊,允許你將模型“轉(zhuǎn)換”為其它格式:

  1. from django.core import serializers
  2.  
  3. serializers.serialize('json', Order.objects.all())

它涵蓋了 Web 應(yīng)用程序最常用的種類,例如 JSON、YAML 和 XML。但是你也可以使用第三方序列化器或創(chuàng)建自己的序列化器。你只需要在 settings.py 文件中注冊它:

  1. # settings.py
  2. SERIALIZATION_MODULES = {
  3.     'my_format': appname.serializers.MyFormatSerializer,
  4. }

要創(chuàng)建自己的 MyFormatSerializer,你需要實現(xiàn) .serialize() 方法并接受一個查詢集和其它選項作為參數(shù):

  1. class MyFormatSerializer:
  2.     def serialize(self, queryset, **options):
  3.         # serious serialization happening

現(xiàn)在,你可以將查詢集序列化為新格式:

  1. from django.core import serializers
  2.  
  3. serializers.serialize('my_format', Order.objects.all())

你可以使用選項參數(shù)來定義序列化程序的行為。例如,如果要定義在處理 ForeignKeys 時要使用嵌套序列化,或者只希望數(shù)據(jù)返回其主鍵,你可以傳遞一個 flat=True 參數(shù)作為選項,并在方法中處理:

  1. class MyFormatSerializer:
  2.     def serializer(self, queryset, **options):
  3.         if options.get('flat', False):
  4.             # don't recursively serialize relationships
  5.         # recursively serialize relationships

使用 Django 序列化的一種方法是使用 loaddata 和 dumpdata 管理命令。

DRF 序列化器

在 Django 社區(qū)中,Django REST 框架(DRF)提供了最著名的序列化器。盡管你可以使用 Django 的序列化器來構(gòu)建將在 API 中響應(yīng)的 JSON,但 REST 框架中的序列化器提供了更出色的功能,可以幫助你處理并輕松驗證復(fù)雜的數(shù)據(jù)。

在訂單的例子中,你可以像這樣創(chuàng)建一個序列化器:

  1. from restframework import serializers
  2.  
  3. class OrderSerializer(serializers.Serializer):
  4.     product = serializers.CharField(max_length=255)
  5.     customer = serializers.CharField(max_lenght=255)
  6.     price = serializers.DecimalField(max_digits=5, decimal_places=2)
  7.     date = serializers.DateField()

輕松序列化其數(shù)據(jù):

  1. order = Order('pen', 'renato', 10.50, date.today())
  2. serializer = OrderSerializer(order)
  3.  
  4. serializer.data
  5. # {'product': 'pen', 'customer': 'renato', 'price': '10.50', 'date': '2020-08-16'}

為了能夠從數(shù)據(jù)返回實例,你需要實現(xiàn)兩個方法:create 和 update

  1. from rest_framework import serializers
  2.  
  3. class OrderSerializer(serializers.Serializer):
  4.     product = serializers.CharField(max_length=255)
  5.     customer = serializers.CharField(max_length=255)
  6.     price = serializers.DecimalField(max_digits=5, decimal_places=2)
  7.     date = serializers.DateField()
  8.  
  9.     def create(self, validated_data):
  10.         # 執(zhí)行訂單創(chuàng)建
  11.         return order
  12.  
  13.     def update(self, instance, validated_data):
  14.        # 執(zhí)行實例更新
  15.        return instance

之后,你可以通過調(diào)用 is_valid() 來驗證數(shù)據(jù),并通過調(diào)用 save() 來創(chuàng)建或更新實例:

  1. serializer = OrderSerializer(**data)
  2. ## 若要驗證數(shù)據(jù),在調(diào)用 save 之前必須執(zhí)行
  3. serializer.is_valid()
  4. serializer.save()

模型序列化器

序列化數(shù)據(jù)時,通常需要從數(shù)據(jù)庫(即你創(chuàng)建的模型)進行數(shù)據(jù)處理。ModelSerializer 與 ModelForm 一樣,提供了一個 API,用于從模型創(chuàng)建序列化器。假設(shè)你有一個訂單模型:

  1. from django.db import models
  2.  
  3. class Order(models.Model):
  4.     product = models.CharField(max_length=255)
  5.     customer = models.CharField(max_length=255)
  6.     price = models.DecimalField(max_digits=5, decimal_places=2)
  7.     date = models.DateField()    

你可以像這樣為它創(chuàng)建一個序列化器:

  1. from rest_framework import serializers
  2.  
  3. class OrderSerializer(serializers.ModelSerializer):
  4.     class Meta:
  5.         model = Order
  6.         fields = '__all__'

Django 會自動在序列化器中包含所有模型字段,并創(chuàng)建 create 和 udpate 方法。

在基于類的視圖(CBV)中使用序列化器

像 Django CBV 中的 Forms 一樣,序列化器可以很好地與 DRF 集成。你可以設(shè)置 serializer_class 屬性,方便序列化器用于視圖:

  1. from rest_framework import generics
  2.  
  3. class OrderListCreateAPIView(generics.ListCreateAPIView):
  4.     queryset = Order.objects.all()
  5.     serializer_class = OrderSerializer

你也可以定義 get_serializer_class() 方法:

  1. from rest_framework import generics
  2.  
  3. class OrderListCreateAPIView(generics.ListCreateAPIView):
  4.     queryset = Order.objects.all()
  5.    
  6.     def get_serializer_class(self):
  7.         if is_free_order():
  8.             return FreeOrderSerializer
  9.         return OrderSerializer

在 CBV 中還有其它與序列化器交互的方法。例如,get_serializer() 返回一個已經(jīng)實例化的序列化器,get_serializer_context() 返回創(chuàng)建實例時傳遞給序列化器的參數(shù)。對于創(chuàng)建或更新數(shù)據(jù)的視圖,有 create 和 update,它們使用 is_valid 方法驗證數(shù)據(jù),還有 perform_create 和 perform_update 調(diào)用序列化器的 save 方法。

了解更多

要了解更多資源,參考我朋友 André Ericson 的經(jīng)典 Django REST 框架網(wǎng)站。它是一個基于類的經(jīng)典視圖的 REST 框架版本,可讓你深入查看組成 DRF 的類。當然,官方文檔也是一個很棒的資源。 

責(zé)任編輯:龐桂玉 來源: Linux中國
相關(guān)推薦

2021-06-09 09:36:18

DjangoElasticSearLinux

2023-12-13 13:49:52

Python序列化模塊

2022-06-23 08:00:53

PythonDateTime模塊

2022-08-06 08:41:18

序列化反序列化Hessian

2024-11-04 15:30:43

Python裝飾器函數(shù)

2025-03-21 09:58:59

Python數(shù)據(jù)類型安全

2020-09-09 07:00:00

TensorFlow神經(jīng)網(wǎng)絡(luò)人工智能

2011-06-01 15:05:02

序列化反序列化

2022-05-10 09:12:16

TypeScript裝飾器

2009-08-24 17:14:08

C#序列化

2010-01-08 13:25:07

ibmdwXML

2009-03-10 13:38:01

Java序列化字節(jié)流

2025-03-05 10:49:32

2011-06-01 15:18:43

Serializabl

2022-03-09 07:10:50

Quarkus框架項目

2009-08-06 11:16:25

C#序列化和反序列化

2011-05-18 15:20:13

XML

2018-03-19 10:20:23

Java序列化反序列化

2025-01-20 08:40:00

Python對象

2025-02-10 09:57:23

點贊
收藏

51CTO技術(shù)棧公眾號