国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁技術(shù)文章
文章詳情頁

Django 再談一談json序列化

瀏覽:62日期:2024-05-28 18:33:30

我們知道JSON字符串是目前流行的數(shù)據(jù)交換格式,在pyhton中我們通過json模塊,將常用的數(shù)據(jù)類型轉(zhuǎn)化為json字符串。但是,json支持轉(zhuǎn)化的數(shù)據(jù)類型是有限的。

比如,我們通過ORM從數(shù)據(jù)庫查詢出的結(jié)果,試圖通過json序列化:

from .models import UserInfodef index(request): user_list = UserInfo.objects.all() import json return HttpResponse(json.dumps(user_list)) # TypeError: Object of type ’QuerySet’ is not JSON serializable

報錯,QuerySet不是JSON能序列化的對象。那么有什么辦法可以解決呢?

注意,如果是通過values查詢,如UserInfo.objects.values('name'),查詢出來的結(jié)果雖然也是QuerySet對象,但是其結(jié)構(gòu)是這樣的:<QuerySet [{’name’: ’egon’}, {’name’: ’sb’}]>, 類似于列表套字典的結(jié)構(gòu)。對于這種情況,我們可以通過list()方法將QuerySet 對象轉(zhuǎn)化為列表,這樣就可以直接用json.dumps()進行序列化了。

方法一:serializers

def index(request): user_list = UserInfo.objects.all() from django.core import serializers user_list_json = serializers.serialize('json', user_list) return HttpResponse(user_list_json)

將返回的結(jié)果放到bejson校驗結(jié)果如下:

[ { 'model': 'app01.userinfo', 'pk': 1, 'fields': { 'name': 'egon', 'pwd': '123' }},{ 'model': 'app01.userinfo', 'pk': 2, 'fields': { 'name': 'sb', 'pwd': '123' }}]

注:pk代表主鍵(可以是默認的id主鍵字段,也可以是用戶自定義的主鍵字段)

觀察序列化結(jié)果,發(fā)現(xiàn)這種方式將服務(wù)端數(shù)據(jù)庫的表名都暴露了;另外serializers不支持連表序列化,只能拿到另一張表的id。下面我們我們用一種新的方式。

方法二:自定義JSON處理器

查看json.dumps源碼,發(fā)現(xiàn)序列化時,用到了一個參數(shù)cls = JSONEncoder,我們可以繼承它,自定義一個類,重寫它的default方法,來處理我們需要的數(shù)據(jù)類型。比如自定義對時間對象進行轉(zhuǎn)化:

import jsonfrom datetime import datefrom datetime import datetimeclass JsonCustomEncoder(json.JSONEncoder): def default(self, field): if isinstance(field, datetime): return field.strftime(’%Y-%m-%d %H:%M:%S’) elif isinstance(field, date): return field.strftime(’%Y-%m-%d’) else: return json.JSONEncoder.default(self, field)

下面我們試著序列化一個datetime對象:

def index(request): now = datetime.now() import json return HttpResponse(json.dumps(now, cls=JsonCustomEncoder))

再次訪問http://127.0.0.1:8000/index.html:

Django 再談一談json序列化

補充知識:Django ORM對象Json序列化問題

碰到了一個問題:在使用json.dumps()序列化Django ORM的Queryset對象,傳遞給前端的時候,程序報錯:

Object of type ’QuerySet’ is not JSON serializable

在python 中,常用的json 的序列化是從simplejson 基礎(chǔ)上改變而來。這個json 包主要提供了dump,load 來實現(xiàn)dict 與 字符串之間的序列化與反序列化,這很方便的可以完成,但現(xiàn)在的問題是,這個json包不能序列化 django 的models 里面的對象的實例。

經(jīng)過一番度娘搜索,發(fā)現(xiàn)有如下解決方案:

使用django.core自帶的serializers模塊:

#django ORM的 Queryset對象默認無法被直接json.dumps()序列化,django.core提供的serializers模塊提供將其序列化成str類型#的功能,serializers處理后,再次json.dumps傳給前端,前端需要經(jīng)過兩次json.Parse()處理,才能得到原對象類型,但是格式發(fā)#生了變化,需要按新的方式取索引.例如:obj[’pk’]取主鍵,obj[’fields’]['caption']取obj的caption字段由QuerySet:[<Business: Business object>]變?yōu)榱耍篬{'model': 'cmdb.business', 'pk': 1, 'fields': {'caption': 'develop'}}]

這樣前端就可以正常獲取數(shù)據(jù)了,只不過此字段需要兩次json.Parse()處理。

至于使用models.Host.objects.get(id=xx)的方式獲取到單個對象,而非Queryset對象,serializers默認也無法處理的問題,可以自定義json方法來實現(xiàn)dumps序列化

json默認只支持python原生的list、tuple、dict數(shù)據(jù)類型對象的序列化,若需要擴展其他類型對象的序列化功能,可以這樣修改:

import json as default_jsonfrom json.encoder import JSONEncoderclass BaseResponse(object): def __init__(self): self.status = True self.message = None self.data = None self.error = Noneo=BaseResponse()class JsonCustomEncoder(JSONEncoder): def default(self, o): if isinstance(o, BaseResponse): return o.__dict__ return JSONEncoder.default(self, o)o1=json.dumps(o,cls=JsonCustomEncoder)>>> print(o1){'message': null, 'error': null, 'data': null, 'status': true}>>> print(type(o1))<class ’str’>#在序列化時指定cls參數(shù),cls=自定義的序列化類,在自定義序列化類的default方法中判斷,如果是指定的類的實例的話,則將該類轉(zhuǎn)換成dict格式返回,若指定類的實例,則使用json模塊默認的序列化方法。最終得到的return值為str類型。

以上這篇Django 再談一談json序列化就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持好吧啦網(wǎng)。

標簽: JavaScript
相關(guān)文章:
主站蜘蛛池模板: 林口县| 长宁区| 武定县| 莲花县| 吴江市| 兴业县| 庆城县| 莲花县| 建阳市| 铅山县| 友谊县| 河曲县| 邢台市| 略阳县| 霍邱县| 柞水县| 团风县| 集安市| 黎川县| 清徐县| 通城县| 桂东县| 盐亭县| 朔州市| 贺兰县| 镇安县| 阿合奇县| 齐齐哈尔市| 香格里拉县| 榆林市| 清新县| 徐闻县| 嘉禾县| 西城区| 柳江县| 红桥区| 珲春市| 花莲县| 石台县| 垦利县| 丰顺县|