本文小編為大家詳細(xì)介紹“怎么在Flutter上優(yōu)雅地序列化一個對象”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“怎么在Flutter上優(yōu)雅地序列化一個對象”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學(xué)習(xí)新知識吧。
成都創(chuàng)新互聯(lián)專注于扎賚諾爾網(wǎng)站建設(shè)服務(wù)及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗。 熱誠為您提供扎賚諾爾營銷型網(wǎng)站建設(shè),扎賚諾爾網(wǎng)站制作、扎賚諾爾網(wǎng)頁設(shè)計、扎賚諾爾網(wǎng)站官網(wǎng)定制、小程序開發(fā)服務(wù),打造扎賚諾爾網(wǎng)絡(luò)公司原創(chuàng)品牌,更為您提供扎賚諾爾網(wǎng)站排名全網(wǎng)營銷落地服務(wù)。序列化一個對象才是正經(jīng)事
對象的序列化和反序列化是我們?nèi)粘>幋a中一個非?;A(chǔ)的需求,尤其是對一個對象的json encode/decode操作。每一個平臺都會有相關(guān)的庫來幫助開發(fā)者方便得進(jìn)行這兩個操作,比如Java平臺上赫赫有名的GSON,阿里巴巴開源的fastJson等等。
而在Flutter上,借助官方提供的JsonCodec,只能對primitive/Map/List這三種類型進(jìn)行json的encode/decode操作,對于復(fù)雜類型,JsonCodec提供了receiver/toEncodable兩個函數(shù)讓使用者手動“打包”和“解包”。
顯然,JsonCodec提供的功能看起來相當(dāng)?shù)脑迹陂e魚app中存在著大量復(fù)雜對象序列化需求,如果使用這個類,就會出現(xiàn)集體“帶薪序列化”的盛況,而且還無法保證正確性。
官方推薦
機智如Google官方,當(dāng)然不會坐視不理。json_serializable的出現(xiàn)就是官方給出的推薦,它借助Dart Build System中的*buildrunner和json_annotation庫,來自動生成fromJson/toJson函數(shù)內(nèi)容。(關(guān)于使用build_runner*生成代碼的原理,之前興往同學(xué)的文章已經(jīng)有所提及)
關(guān)于如何使用json_serializable網(wǎng)上已經(jīng)有很多文章了,這里只簡單提一些步驟:
Step 1 創(chuàng)建一個實體類。
Step 2 生成代碼:
讓build runner生成序列化代碼。運行完成后文件夾下會出現(xiàn)一個xxx.g.dart文件,這個文件就是生成后的文件。
Step 3 代理實現(xiàn):
把fromJson和toJson操作代理給上面生成出來的類。
我們?yōu)槭裁床挥盟鼘崿F(xiàn)?
json_serializable完美實現(xiàn)了需求,但它也有不滿足需求的一面:
使用起來有些繁瑣,多引入了一個類
很重要的一點是,大量的使用"as"會給性能和最終產(chǎn)物大小產(chǎn)生不小的影響。實際上閑魚內(nèi)部的《flutter編碼規(guī)范》中,是不建議使用"as"的。(對包大小的影響可以參見三笠同學(xué)的文章,同時dart linter也對as的性能影響有所描述)
基于上面的分析,很明顯的,需要一種新的方式來解決我們面臨的問題,我們暫且叫它fish-serializable
1需要實現(xiàn)的功能
我們首先來梳理一下,一個序列化庫需要用到:
獲取可序列化對象的所有field以及它們的類型信息
能夠構(gòu)造出一個可序列化對象,并對它里面的fields賦值,且類型正確
支持自定義類型
最好能夠解決泛型的問題,這會讓使用更加方便
最好能夠輕松得在不同的序列化/反序列化方式中切換,例如json和protobuf。
2困難在哪
flutter禁用了dart:mirrors,反射API無法使用,也就無法通過反射的方式new一個instance、掃描class的fields。
泛型的問題由于dart不進(jìn)行類型擦出,可以獲取,但泛型嵌套后依然無法解開。
3Let's rock
無法使用dart:mirrors是個“硬”問題,沒有反射的支持,類的內(nèi)容就是一個黑盒。于是我們在邁出第一步的時候就卡殼了- -!
這個時候筆者腦子里閃過了很多畫面,白駒過隙,烏飛兔走,啊,不是...是c++,c++作為一種無法使用反射的語言,它是如何實現(xiàn)對象的 序列化/反序列化 操作的呢?
一頓搜索猛如虎之后,發(fā)現(xiàn)大神們使用創(chuàng)建類對象的回調(diào)函數(shù)配合宏的方式來實現(xiàn)c++中類似反射這樣的操作。
這個時候,筆者又想到了曾經(jīng)朝夕相處的Android(現(xiàn)在已經(jīng)變成了flutter),Android中的Parcelable序列化協(xié)議就是一個很好的參照,它通過writeXXX APIs將類的數(shù)據(jù)寫入一個中間存儲進(jìn)行序列化,再通過readXXX APIs進(jìn)行反序列化,這就解決了我們上面提到的第一個問題,既如何將一個類的“黑盒子”打開。
同時,Parcelable協(xié)議中還需要使用者提供一個叫做Creator的靜態(tài)內(nèi)部類,用來在反序列化的時候反射創(chuàng)建一個該類的對象或?qū)ο髷?shù)組,對于沒有反射可用的我們來說,用c++的那種回調(diào)函數(shù)的方式就可以完美解決反序列化中對象創(chuàng)建的問題。
ValueHolder
這是一個數(shù)據(jù)中轉(zhuǎn)存儲的基類,它內(nèi)部的writeXXX APIs提供展開類內(nèi)部的fields的能力,而readXXX則
用來將ValueHolder中的內(nèi)容讀取賦值給類的fields。
readList/readMap/readSerializable函數(shù)中的type argument,我們把它作為外部想要解釋數(shù)據(jù)的
方式,比如readSerializable<T>(key: 'object'),表示外部想要把key為object的值解釋為T類
型。
FishSerializable
FishSerializable是一個interface,creator是個一個get函數(shù),用來返回一個“創(chuàng)建類對象的回調(diào)”,writeTo函數(shù)則用來在反序列化的時候放置ValueHoder->fields的代碼。
JsonSerializer
它繼承于FishSerializer接口,實現(xiàn)了encode/decode函數(shù),并額外提供encodeToMap和decodeFromMap功能。JsonSerializer類似JsonCodec,直接面向使用者用來json encode/decode
以上,我們已經(jīng)基本做好了一個flutter上支持對象序列化/反序列化操作的庫的基本架構(gòu)設(shè)計,對象的序列化過程可以簡化為:
由于ValueHolder中間存儲的存在,我們可以很方便得切換 序列化/反序列器,比如現(xiàn)有的JsonSerializer用來實現(xiàn)json的encode/decode,如果有類似protobuf的需求,我們則可以使用ProtoBufSerializer來將ValueHolder中的內(nèi)容轉(zhuǎn)換成我們需要的格式。
困難是不存在的
1如何匹配類型
為了能支持泛型容器的解析,我們需要類似下面這樣的邏輯:
List<SerializableObject> list
= holder.readList<SerializableObject>(key: 'list');
List<E> readList<E>({String key}){
List<dynamic> list = _read(key);
}
E _flattenList<E>(List<dynamic> list){
list?.map<E>((dynamic item){
// 比較E是否屬于某個類型,然后進(jìn)行對應(yīng)類型的轉(zhuǎn)換
});
}
在Java中,可以使用Class#isAssignableFrom,而在flutter中,我們沒有發(fā)現(xiàn)類似功能的API提供。而且,如果做下面這個測試,你還會發(fā)現(xiàn)一些很有意思的細(xì)節(jié):
void main() {
print('int test');
test<int>(1);
print('\r\nint list test');
test<List<int>>(<int>[]);
print('\r\nobject test');
test<A<int>>(A<int>());
}
void test<T>(T t){
print(T);
print(t.runtimeType);
print(T == t.runtimeType);
print(identical(T, t.runtimeType));
}
class A<T>{
}
輸出的結(jié)果是:
可以看到,對于List這樣的容器類型,函數(shù)的type argument與instance的runtimeType無法比較,當(dāng)然如果使用t is T,是可以返回正確的值的,但需要構(gòu)造大量的對象。所以基本上,我們無法進(jìn)行類型匹配然后做類型轉(zhuǎn)換。
2如何解析泛型嵌套
接下去就是如何分解泛型容器嵌套的問題,考慮如下場景:
Map<String, List<int>> listMap;
listMap = holder.readMap<String, List<int>>(key: 'listMap');
readMap中得到的value type是一個 List<int>
,而我們沒有API去切割這個type argument。所以我們采用了一種比較“笨”也相對實用的方式。我們使用字符串切割了type argument,比如:
List<int> => <String>[List<int>, List, int]
然后在內(nèi)部展開List或Map的時候,使用字符串匹配的方式匹配類型,在目前的使用中,完美得支持了標(biāo)準(zhǔn)List和Map
容器互相嵌套。但目前無法支持標(biāo)準(zhǔn)List
和Map之外的其他容器類型。
1IDE插件輔助
寫過Android的Parcelable的同學(xué)應(yīng)該有種很深刻的體會,Parcelable協(xié)議中有大量的“機械”代碼需要寫,類似設(shè)計的fish-serializable也一樣。
為了不被老板和使用庫的同學(xué)打死,同時開發(fā)了fish-serializable-intelij-plugin來自動生成這些“機械”代碼。
2與json_serializable的對比
fish-serializable
在使用上配合IDE插件,減少了大量的"as"操作符的使用,同時在步驟上也更加簡短方便。
相比于 json_annotation
生成的代碼, fish-serializable
生成的代碼也更具可讀性,方便手動修改一些代碼實現(xiàn)。
fish-serializable
可以通過手動接管 序列化/反序列化 過程的方式完美兼容 json_annotation
等其他方案。
讀到這里,這篇“怎么在Flutter上優(yōu)雅地序列化一個對象”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領(lǐng)會,如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注創(chuàng)新互聯(lián)-成都網(wǎng)站建設(shè)公司行業(yè)資訊頻道。
當(dāng)前題目:怎么在Flutter上優(yōu)雅地序列化一個對象-創(chuàng)新互聯(lián)
當(dāng)前URL:http://m.rwnh.cn/article42/dsdgec.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供做網(wǎng)站、虛擬主機、靜態(tài)網(wǎng)站、域名注冊、自適應(yīng)網(wǎng)站、響應(yīng)式網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容