Python序列化和反序列化

Python序列化和反序列化

通過將對象序列化可以將其存儲在變量或者文件中瘪弓,可以保存當(dāng)時對象的狀態(tài)邑狸,實現(xiàn)其生命周期的延長。并且需要時可以再次將這個對象讀取出來个从。Python中有幾個常用模塊可實現(xiàn)這一功能侣监。

pickle模塊

存儲在變量中

dumps(obj)返回存入的字節(jié)

dic = {'age': 23, 'job': 'student'}
byte_data = pickle.dumps(dic)
# out -> b'\x80\x03}q\x00(X\x03\x00\x00\...'
print(byte_data)

讀取數(shù)據(jù)

數(shù)據(jù)以字節(jié)保存在了byte_data變量中鸭轮,需要再次使用的時候使用loads函數(shù)就行了。

obj = pickle.loads(byte_data)
print(obj)

存儲在文件中

也可以存在文件中橄霉,使得對象持久化窃爷。使用的是dumpload函數(shù),注意和上面的區(qū)別姓蜂,少了s按厘。由于pickle寫入的是二進制數(shù)據(jù),所以打開方式需要以wbrb的模式钱慢。

# 序列化
with open('abc.pkl', 'wb') as f:
    dic = {'age': 23, 'job': 'student'}
    pickle.dump(dic, f)
# 反序列化
with open('abc.pkl', 'rb') as f:
    aa = pickle.load(f)
    print(aa)
    print(type(aa))  # <class 'dict'>

序列化用戶自定義對象

假如我寫了個類叫做Person

class Person:
    def __init__(self, name, age, job):
        self.name = name
        self.age = age
        self.job = job

    def work(self):
        print(self.name, 'is working...')

pickle當(dāng)然也能寫入逮京,不僅可以寫入類本身,也能寫入它的一個實例束莫。

# 將實例存儲在變量中造虏,當(dāng)然也能存在文件中
a_person = Person('abc', 22, 'waiter')
person_abc = pickle.dumps(a_person)
p = pickle.loads(person_abc)
p.work()
# 將類本身存儲在變量中,loads的時候返回類本身麦箍,而非它的一個實例
class_Person = pickle.dumps(Person)
Person = pickle.loads(class_Person)
p = Person('Bob', 23, 'Student')
p.work()

# 下面這個例子演示的就是將類存儲在文件中
# 序列化
with open('person.pkl', 'wb') as f:
    pickle.dump(Person, f)
# 反序列化
with open('person.pkl', 'rb') as f:
    Person = pickle.load(f)
    aa = Person('gg', 23, '6')
    aa.work()

json模塊

pickle可以很方便地序列化所有對象。不過json作為更為標(biāo)準(zhǔn)的格式陶珠,具有更好的可讀性(pickle是二進制數(shù)據(jù))和跨平臺性挟裂。是個不錯的選擇。

json使用的四個函數(shù)名和pickle一致揍诽。

序列化為字符串

dic = {'age': 23, 'job': 'student'}
dic_str = json.dumps(dic)
print(type(dic_str), dic_str)
# out: <class 'str'> {"age": 23, "job": "student"}

dic_obj = json.loads(dic_str)
print(type(dic_obj), dic_obj)
# out: <class 'dict'> {'age': 23, 'job': 'student'}

可以看到诀蓉,dumps函數(shù)將對象轉(zhuǎn)換成了字符串栗竖。loads函數(shù)又將其恢復(fù)成字典。

存儲為json文件

也可以存儲在json文件中

dic = {'age': 23, 'job': 'student'}
with open('abc.json', 'w', encoding='utf-8') as f:
    json.dump(dic, f)

with open('abc.json', encoding='utf-8') as f:
    obj = json.load(f)
    print(obj)

存儲自定義對象

還是上面的Person對象渠啤。如果直接序列化會報錯

aa = Person('Bob', 23, 'Student')
with open('abc.json', 'w', encoding='utf-8') as f:
    json.dump(aa, f) # 報錯

Object of type 'Person' is not JSON serializable此時dump函數(shù)里傳一個參default就可以了狐肢,這個參數(shù)接受一個函數(shù),這個函數(shù)可以將對象轉(zhuǎn)換為字典沥曹。

寫一個就是了

def person2dict(person):
    return {'name': person.name,
            'age': person.age,
            'job': person.job}

這樣返回的就是一個字典了份名,對象實例有個方法可以簡化這一過程。直接調(diào)用實例的__dict__妓美。例如

print(aa.__dict) # {'name': 'Bob', 'age': 23, 'job': 'Student'}

很方便僵腺。

同時在讀取的時候load出來的是一個字典,再轉(zhuǎn)回對象就可壶栋,同樣需要一個object_hook參數(shù)辰如,該參數(shù)接收一個函數(shù),用于將字典轉(zhuǎn)為對象贵试。

def dict2person(dic):
    return Person(dic['name'], dic['age'], dic['job'])

于是完整的程序應(yīng)該寫成下面這樣

with open('abc.json', 'w', encoding='utf-8') as f:
    json.dump(aa, f, default=person2dict)

with open('abc.json', encoding='utf-8') as f:
    obj = json.load(f, object_hook=dict2person)
    print(obj.name, obj.age, obj.job)
    obj.work()

由于可以使用__dict__代替person2dict函數(shù)琉兜,再使用lambda函數(shù)簡化。

with open('abc.json', 'w', encoding='utf-8') as f:
   json.dump(aa, f, default=lambda obj: obj.__dict__)

以上是存儲到文件毙玻,存儲到變量也是類似操作豌蟋。

不過就我現(xiàn)在所學(xué),不知道如何像pickle一樣方便的將我們自定義的類本身使用json序列化淆珊,或許要用到其他擴展函數(shù)夺饲。以后用到了再說。

shelve模塊

還有一個模塊施符,不太常用往声,通常使用一個open就好。shelve以鍵值對的形式存儲數(shù)據(jù)戳吝。

with shelve.open('aa') as f:
    f['person'] = {'age': 23, 'job': 'student'}
    f['person']['age'] = 44  # 這里試圖改變原來的年齡23
    f['numbers'] = [i for i in range(10)]

with shelve.open('aa') as f:
    person = f['person']
    print(person) # {'age': 23, 'job': 'student'}
    nums = f['numbers']
    print(nums) # [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

文件不要有后綴名浩销,在windows下會生成aa.bak, aa.dat, aa.dir三個文件(有點多)。其中bak和dir文件是可以查看的(貌似兩個文件內(nèi)容一樣)在下面這個例子中生成這樣的數(shù)據(jù)听哭。

'person', (0, 44)
'numbers', (512, 28)

允許寫回--writeback

有個細節(jié)慢洋,我們讀取鍵person時候,發(fā)現(xiàn)age還是23歲陆盘,f['person']['age'] = 44后并沒有變成44普筹。下面的寫法

with shelve.open('aa', writeback=True) as f:
    dic = {'age': 23, 'job': 'student'}
    f['person'] = dic
    dic['age'] = 44
    f['person'] = dic

相當(dāng)于賦值了兩次,這種方法是可以改變值的隘马。

默認情況下直接使用f['person']改變其中的值之后太防,不會更新已存儲的值,也就是沒有把更新寫回到文件酸员,即使是文件被close后蜒车。如果有此需要讳嘱,在open函數(shù)中添加一個參數(shù)writeback=True。再次運行下看看年齡就被改變了酿愧。

寫入自定義對象

依然使用上面的Person對象

with shelve.open('aa') as f:
    f['class'] = Person
    
# 寫入類本身
with shelve.open('aa') as f:
    Person = f['class']
    a = Person('Bob', 23, 'Student')
    a.work()

上面的例子說明shelve也可以序列化類本身沥潭。當(dāng)然序列化實例肯定可以。

with shelve.open('aa') as f:
    a = Person('God', 100, 'watch')
    f['class'] = a

with shelve.open('aa') as f:
    god = f['class']
    god.work()

注意嬉挡,由于我們使用with open打開钝鸽,故不用寫close語句,此模塊是有close函數(shù)的棘伴,如果不是with方法打開的一定要記得主動close寞埠。


by @sunhaiyu

2017.6.27

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市焊夸,隨后出現(xiàn)的幾起案子仁连,更是在濱河造成了極大的恐慌,老刑警劉巖阱穗,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件饭冬,死亡現(xiàn)場離奇詭異,居然都是意外死亡揪阶,警方通過查閱死者的電腦和手機昌抠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鲁僚,“玉大人炊苫,你說我怎么就攤上這事”常” “怎么了侨艾?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長拓挥。 經(jīng)常有香客問我唠梨,道長,這世上最難降的妖魔是什么侥啤? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任当叭,我火速辦了婚禮,結(jié)果婚禮上盖灸,老公的妹妹穿的比我還像新娘蚁鳖。我一直安慰自己,他們只是感情好赁炎,可當(dāng)我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布才睹。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪琅攘。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天松邪,我揣著相機與錄音坞琴,去河邊找鬼。 笑死逗抑,一個胖子當(dāng)著我的面吹牛剧辐,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播邮府,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼荧关,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了褂傀?” 一聲冷哼從身側(cè)響起忍啤,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎仙辟,沒想到半個月后同波,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡叠国,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年未檩,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片粟焊。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡冤狡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出项棠,到底是詐尸還是另有隱情悲雳,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布沾乘,位于F島的核電站怜奖,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏翅阵。R本人自食惡果不足惜歪玲,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望掷匠。 院中可真熱鬧滥崩,春花似錦、人聲如沸讹语。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至短条,卻和暖如春导匣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背茸时。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工贡定, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人可都。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓缓待,卻偏偏與公主長得像,于是被迫代替她去往敵國和親渠牲。 傳聞我的和親對象是個殘疾皇子旋炒,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)签杈,斷路器瘫镇,智...
    卡卡羅2017閱讀 134,599評論 18 139
  • 1 前言 在“通過簡單示例來理解什么是機器學(xué)習(xí)”這篇文章里提到了pickle庫的使用,本文來做進一步的闡述芹壕。 pi...
    leenard閱讀 1,968評論 0 2
  • JAVA序列化機制的深入研究 對象序列化的最主要的用處就是在傳遞,和保存對象(object)的時候,保證對象的完整...
    時待吾閱讀 10,837評論 0 24
  • pyton review 學(xué)習(xí)指南 https://www.zhihu.com/question/29138020...
    孫小二wuk閱讀 1,039評論 0 2
  • 每天晚上睡之前汇四,給孩子洗臉洗腳喝奶刷牙,這些都準(zhǔn)備完畢后踢涌,她會非常期待的坐在床上等著我通孽,我把她擁入懷中,開始一天最...
    飛天蘿莉想閱讀 1,044評論 1 0