什么是序列化?

我们把变量从内存中变成可存储或传输的过程称之为序列化,在Python中叫pickling,在其他语言中也被称之为serialization,marshalling,flattening等等,都是一个意思。

将python中的列表,字典,元组,集合转换成字符串的过程就叫做序列化,反之叫做反序列化。

我们把变量从内存中变成可存储或传输的过程称之为序列化,序列化之后,就可以把序列化后的内容写入磁盘,或者通过网络传输到别的机器上。

把变量内容从序列化的对象重新读到内存里称之为反序列化,即unpickling。

序列化的目的

1、以某种存储形式使自定义对象持久化
2、将对象从一个地方传递到另一个地方。
3、使程序更具维护性。

Json模块

所有的语言都通用,默认序列化的数据是有限的:列表、字典、元组(元组被转化为字符串的列表的形式),布尔值,数字,None
  1. +-------------------+---------------+
  2. | Python | JSON |
  3. +===================+===============+
  4. | dict | object |
  5. +-------------------+---------------+
  6. | list, tuple | array |
  7. +-------------------+---------------+
  8. | str | string |
  9. +-------------------+---------------+
  10. | int, float | number |
  11. +-------------------+---------------+
  12. | True | true |
  13. +-------------------+---------------+
  14. | False | false |
  15. +-------------------+---------------+
  16. | None | null |
  17. +-------------------+---------------+
如果我们要在不同的编程语言之间传递对象,就必须把对象序列化为标准格式,比如XML,但更好的方法是序列化为JSON,因为JSON表示出来就是一个字符串,可以被所有语言读取,也可以方便地存储到磁盘或者通过网络传输。JSON不仅是标准格式,并且比XML更快,而且可以直接在Web页面中读取,非常方便。
 
查看json库所有的方法
  1. import json
  2.  
  3. print(json.__all__)

结果:

  1. ['dump', 'dumps', 'load', 'loads', 'JSONDecoder', 'JSONDecodeError', 'JSONEncoder']
例子:
  1. import json
  2. dic3=('e','g')
  3. str2=json.dumps(dic3)
  4. print(type(str2),str2)

结果:

  1. <class 'str'> ["e", "g"]
  1. loadsdumps主要转化类型
  1. import json
  2. dic = {'k1':'v1','k2':'v2','k3':'v3'}
  3. str_dic = json.dumps(dic) #序列化:将一个字典转换成一个字符串
  4. print(type(str_dic),str_dic) #<class 'str'> {"k3": "v3", "k1": "v1", "k2": "v2"}
  5. #注意,json转换完的字符串类型的字典中的字符串是由""表示的
  6.  
  7. dic2 = json.loads(str_dic) #反序列化:将一个字符串格式的字典转换成一个字典
  8. #注意,要用json的loads功能处理的字符串类型的字典中的字符串必须由""表示
  9. print(type(dic2),dic2) #<class 'dict'> {'k1': 'v1', 'k2': 'v2', 'k3': 'v3'}
  10.  
  11. list_dic = [1,['a','b','c'],3,{'k1':'v1','k2':'v2'}]
  12. str_dic = json.dumps(list_dic) #也可以处理嵌套的数据类型
  13. print(type(str_dic),str_dic) #<class 'str'> [1, ["a", "b", "c"], 3, {"k1": "v1", "k2": "v2"}]
  14. list_dic2 = json.loads(str_dic)
  15. print(type(list_dic2),list_dic2) #<class 'list'> [1, ['a', 'b', 'c'], 3, {'k1': 'v1', 'k2': 'v2'}]

关于json.dumps补充知识点一

json.dumps(s,ensure_ascii=True),默认的编码方式ascii,如果你输入的是中文,它会默认输出的是unicode编码结果,如果你想要得到正确的值,ensure_ascii=Flase

  1. import json
  2. s='中国'
  3. print(json.dumps(s))
  4. print(json.dumps(s,ensure_ascii=False))

结果:

  1. "\u4e2d\u56fd"#unicode的编码结果
  2. "中国"

json.dumps(s,default=f),dump中有一个default的参数这个参数默认为空,如果你想使用它,default后边跟一个函数名,这个函数是用来序列化对象的.

我们知道对象是不能被序列化的,我们可以编写方法来序列化对象

比如

  1. import json
  2.  
  3. class Boy:
  4.  
  5. def __init__(self, name, age):
  6. self.name = name
  7. self.age = age
  8.  
  9. obj = Boy('Will', 20)
  10. # # 我们知道json是不能序列化对象的,我们可以编写方法来序列化对象
  11. def json_boy(obj):
  12. if isinstance(obj, Boy):
  13. return {'name': obj.name, 'age': obj.age}
  14. return obj
  15.  
  16. json_data = json.dumps(obj, default=json_boy)
  17.  
  18. print(json_data)

结果为:

  1. {"name": "Will", "age": 20}
  1. loaddump主要与写入文件有关系
  2.  
  3. import json
  4. f = open('json_file','w')
  5. dic = {'k1':'v1','k2':'v2','k3':'v3'}
  6. json.dump(dic,f) #dump方法接收一个文件句柄,直接将字典转换成json字符串写入文件
  7. f.close()
  8.  
  9. f = open('json_file')
  10. dic2 = json.load(f) #load方法接收一个文件句柄,直接将文件中的json字符串转换成数据结构返回
  11. f.close()
  12. print(type(dic2),dic2)

序列非支持的数据类型

在JSONEncoder不知道怎么去把这个数据转换成json字符串的时候,它就会去调用default()函数,所以都是重写这个函数来处理它本身不支持的数据类型,default()函数默认是直接抛异常的。

  1. def default(self, o):
  2. """Implement this method in a subclass such that it returns
  3. a serializable object for ``o``, or calls the base implementation
  4. (to raise a ``TypeError``).
  5.  
  6. For example, to support arbitrary iterators, you could
  7. implement default like this::
  8.  
  9. def default(self, o):
  10. try:
  11. iterable = iter(o)
  12. except TypeError:
  13. pass
  14. else:
  15. return list(iterable)
  16. # Let the base class default method raise the TypeError
  17. return JSONEncoder.default(self, o)
  18.  
  19. """
  20. raise TypeError("Object of type '%s' is not JSON serializable" %
  21. o.__class__.__name__)

官方举了一个序列化迭代器的例子,

我们知道json是不能序列迭代器的

  1. i=[1,2,3,4]
  2.  
  3. b=iter(i)#生成迭代器
  4.  
  5. class JsonTex(json.JSONEncoder):
  6. def default(self, o):
  7. try:
  8. pass
  9. except TypeError as e:
  10. pass
  11. else:
  12. return list(o)
  13. tt=json.dumps(b,cls=JsonTex)
  14. print(tt,type(tt))

结果:

  1. [1, 2, 3, 4] <class 'str'>

序列datetime类型

datetime json并不支持这种类型,如果我们要序列化,就要重写JSONEncode中的default方法

  1. import datetime
  2. time=datetime.datetime.today()
  3. class JsonTex(json.JSONEncoder):
  4. def default(self, o):
  5. if isinstance(o,datetime.datetime):
  6. return o.strftime('%Y-%m-%d %H:%M:%S')
  7. else:
  8. raise TypeError("Object of type '%s' is not JSON serializable" %
  9. o.__class__.__name__)
  10.  
  11. t=json.dumps(time,cls=JsonTex)
  12.  
  13. print(t,type(t))

结果:

  1. "2018-07-19 13:27:19" <class 'str'>

pickle模块

pickle模块 可以序列化python中任何类型,python专有的不能和其他语言混用,序列化的结果是bytes类型

列子:

  1. import pickle
  2. dic={'k1':'k2'}
  3. str_dic=pickle.dumps(dic)
  4. print(str_dic)

结果:

  1. b'\x80\x03}q\x00X\x02\x00\x00\x00k1q\x01X\x02\x00\x00\x00k2q\x02s.'

用pickle序列化的数据,反序列化的时候也要用pickle。

  1. import pickle
  2. dic = {'k1':'v1','k2':'v2','k3':'v3'}
  3. str_dic = pickle.dumps(dic)
  4. print(str_dic) #一串二进制内容
  5.  
  6. dic2 = pickle.loads(str_dic)
  7. print(dic2) #字典
  8.  
  9. import time
  10. struct_time = time.localtime(1000000000)
  11. print(struct_time)
  12. f = open('pickle_file','wb')
  13. pickle.dump(struct_time,f) dump把数据类型序列化到文档中,以bytes类型
  14. f.close()
  15.  
  16. f = open('pickle_file','rb')
  17. struct_time2 = pickle.load(f)
  18. print(struct_time2.tm_year)
  19.  

shelve模块

shelve模块也是python提供给我们的序列化工具,比pickle用起来更简单一些。
shelve只提供给我们一个open方法,是用key来访问的,使用起来和字典类似。

  1. import shelve
  2. f = shelve.open('shelve_file')
  3. f['key'] = {'int':10, 'float':9.5, 'string':'Sample data'} #直接对文件句柄操作,就可以存入数据
  4. f.close()
  5.  
  6. import shelve
  7. f1 = shelve.open('shelve_file')
  8. existing = f1['key'] #取出数据的时候也只需要直接用key获取即可,但是如果key不存在会报错
  9. f1.close()
  10. print(existing)
  11.  
  12. shelve

这个模块有个限制,它不支持多个应用同一时间往同一个DB进行写操作。所以当我们知道我们的应用如果只进行读操作,我们可以让shelve通过只读方式打开DB

  1. import shelve
  2. f = shelve.open('shelve_file', flag='r')
  3. existing = f['key']
  4. f.close()
  5. print(existing)

由于shelve在默认情况下是不会记录待持久化对象的任何修改的,所以我们在shelve.open()时候需要修改默认参数,否则对象的修改不会保存。

  1. import shelve
  2. f1 = shelve.open('shelve_file')
  3. print(f1['key'])
  4. f1['key']['new_value'] = 'this was not here before'
  5. f1.close()
  6.  
  7. f2 = shelve.open('shelve_file', writeback=True)
  8. print(f2['key'])
  9. f2['key']['new_value'] = 'this was not here before'
  10. f2.close()
  11.  
  12. 设置writeback

writeback方式有优点也有缺点。优点是减少了我们出错的概率,并且让对象的持久化对用户更加的透明了;但这种方式并不是所有的情况下都需要,首先,使用writeback以后,shelf在open()的时候会增加额外的内存消耗,并且当DB在close()的时候会将缓存中的每一个对象都写入到DB,这也会带来额外的等待时间。因为shelve没有办法知道缓存中哪些对象修改了,哪些对象没有修改,因此所有的对象都会被写入。

序列化 json 模块的更多相关文章

  1. python中序列化json模块和pickle模块

    内置模块和第三方模块 json模块和pickle 模块(序列化模块) 什么是序列化? 序列化就是将内粗这种的数据类型转成另一种格式 序列化:字典类型——>序列化——>其他格式——>存 ...

  2. Python基础(12)_python模块之sys模块、logging模块、序列化json模块、pickle模块、shelve模块

    5.sys模块 sys.argv 命令行参数List,第一个元素是程序本身路径 sys.exit(n) 退出程序,正常退出时exit(0) sys.version 获取Python解释程序的版本信息 ...

  3. 序列化json模块

    1.用json模块来进行序列化和反序列化 注意:用json序列化的数据类型得到的文件后缀名必须是json.因为如果不是json后缀,别人也不知道这是用json序列化的文件. 序列化:json.dump ...

  4. Python--模块之sys模块、logging模块、序列化json模块、序列化pickle模块

    sys模块 sys.argv 命令行参数List,第一个元素是程序本身路径 sys.exit(n) 退出程序,正常退出时exit() sys.path 返回模块的搜索路径,初始化时使用PYTHONPA ...

  5. python之序列化json模块与pickle模块(待补充)

    一.json是所有语言都通用的一种序列化格式 只支持 : 列表,字典字符串,数字,且字典的key必须是字符串 ''' 1. dumps , loads 在内存中做数据转换: dumps : 数据类型 ...

  6. Python入门-模块4(序列化----json模块和pickle模块)

    序列化是指把内存里的数据类型转变成字符串,以使其能存储到硬盘或通过网络传输到远程,因为硬盘或网络传输时只能接受bytes.反之,把硬盘里面的数据读到内存里,叫反序列化.

  7. Python进阶(九)----json模块, pickle模块, os模块,sys模块,hashlib模块

    Python进阶----json模块, pickle模块, os模块,sys模块,hashlib模块 一丶序列化模块 什么是序列化: ​ 将一种数据结构,转换成一个特殊的序列(特殊字符串,用于网络传输 ...

  8. python学习之文件读写,序列化(json,pickle,shelve)

    python基础 文件读写 凡是读写文件,所有格式类型都是字符串形式传输 只读模式(默认) r  f=open('a.txt','r')#文件不存在会报错 print(f.read())#获取到文件所 ...

  9. Python第十四天 序列化 pickle模块 cPickle模块 JSON模块 API的两种格式

    Python第十四天 序列化  pickle模块  cPickle模块  JSON模块  API的两种格式 目录 Pycharm使用技巧(转载) Python第一天  安装  shell  文件 Py ...

随机推荐

  1. Apache+PHP+MySQL+phpMyAdmin+WordPress搭建

    一 .安装Apache 下载地址:http://www.apachelounge.com/download/,选择Apache 2.4.25 Win64,解压缩,修改配置文件中如下地方: 1.Serv ...

  2. python中的IO模块

    1.简介 读写文件是常见的IO操作,python内置了读写文本的函数. 读写文件的模式描述如下: 模式 描述 r 以只读方式打开文件.文件的指针将会放在文件的开头.这是默认模式. rb 以二进制格式打 ...

  3. .NET面试基本问题

    1..NET和C#的区别? .NET:一般指的是.NET FrameWork框架,是平台,技术. C#:是一编程语言,是基本.NET平台. 2.C#的委托是什么?事件是不是委托? 委托可以把一个方法作 ...

  4. 一、I/O操作(流的概念)

    一.流(Stream) 所谓流(Stream),就是一系列的数据. 当不同的介质之间有数据交互的时候,java就会使用流来实现. 数据源可以使文件,还可以是数据库,网络,甚至是其他的程序 不如读取文件 ...

  5. oracle having字句

    现在要求查询出职位的平均每个职位的名称,工资,但是要求显示的职位的平均工资高于2000.        即:按照职位先进行分组,同时统计出每个职位的平均工资        随后要求直显示哪些平均工资高 ...

  6. [LeetCode] 94. Binary Tree Inorder Traversal(二叉树的中序遍历) ☆☆☆

    二叉树遍历(前序.中序.后序.层次.深度优先.广度优先遍历) 描述 解析 递归方案 很简单,先左孩子,输出根,再右孩子. 非递归方案 因为访问左孩子后要访问右孩子,所以需要栈这样的数据结构. 1.指针 ...

  7. 使用VAE、CNN encoder+孤立森林检测ssl加密异常流的初探——真是一个忧伤的故事!!!

    ssl payload取1024字节,然后使用VAE检测异常的ssl流. 代码如下: from sklearn.model_selection import train_test_split from ...

  8. win10 将本地项目上传到github (第一次+再次上传)

    推荐网址: https://blog.csdn.net/zamamiro/article/details/70172900 https://blog.csdn.net/qq_15800305/arti ...

  9. Express 路径请求及对应的获取路径形式

    req.query // GET /search?q=tobi+ferret req.query.q // => "tobi ferret" // GET /shoes?or ...

  10. react router @4 和 vue路由 详解(全)

    react router @4 和 vue路由 本文大纲: 1.vue路由基础和使用 2.react-router @4用法 3.什么是包容性路由?什么是排他性路由? 4.react路由有两个重要的属 ...