一、概述

前面章节我们讲述了json和pickle模块的序列化和反序列化处理,他们有一个不足是在python 3中不能多次dump和load,shelve模块则可以规避这个问题。
shelve模块是一个简单的k,v将内存数据通过文件持久化的模块,可以持久化任何pickle可支持的python数据格式,是pickle 更上一层的封装。

二、shelve模块的用法

Shelve模块提供了基本的存储操作,Shelve中的open函数在调用的时候返回一个shelf对象,通过该对象可以存储内容,即像操作字典一样进行存储操作。当在该对象中查找元素时,对象会根据已经存储的版本进行重新构建,当给某个键赋值的时候,元素会被存储。

  • 持久化存储
      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9 with shelve.open('shelve_demo') as data:
    10 data['name'] = name
    11 data['info'] = info
    12 data['func'] = member_info

    代码执行后会生成三个文件:

    其中shelve_demo.bak和shelve_demo.dir文件内容相同,显示如下:

      1 'name', (0, 43)
    2 'info', (512, 45)
    3 'func', (1024, 27)

    shelve_demo.dat猜测为类数据库文件把。

  • 解析文件内容
      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 with shelve.open('shelve_demo') as data:
    7 print(data['name']) #这里的key与之前保存的文件内容里面的key完全对应,否则会报错
    8 print(data['info'])
    9 print(data['func']('Alex', 22))
    10
    11 #运行结果输出:
    12 ['Jack', 'Maxwell', 'Tom']
    13 {'name': 'Maxwell', 'age': 18}
    14 Member info: Alex 22
    15 None
  • value值的修改
    一般情况下,我们通过shelve来open一个对象后,只能进行一次赋值处理,赋值后不能再次更新处理。
      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9 with shelve.open('shelve_demo') as data:
    10 data['name'] = name
    11 data['info'] = info
    12 data['name'].append('Alex')
    13 print(data['name'])
    14
    15 程序输出:
    16 ['Jack', 'Maxwell', 'Tom'] #第一次赋值后apend的元素并没有生效

    再次open打开结果也是这样:

      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9
    10 with shelve.open('shelve_demo') as data:
    11 print(data['name'])

    造成上述情况的原因是:我们只是修改了shelve对象的副本,而它并木有被最终保存。此时我们除了下文要讲述的update方法外,还有以下两种方法:
    方法一: shelve open一个对象后,先用临时变量指向对象副本,在临时变量上修改后让对象副本再次指向临时变量,从而覆盖保存对象副本。这种方法的本质是对open后的对象重新赋新值,并非在原有基础上进行update,也就是open后的对象内存指向地址发生了变化。

      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9 with shelve.open('shelve_demo') as data:
    10 data['name'] = name
    11 data['info'] = info
    12 temp = data['name'] #这里的关键点在于对临时变量的使用
    13 temp.append('Alex')
    14 data['name'] = temp
    15 print(data['name'])
    16
    17 程序输出:
    18 ['Jack', 'Maxwell', 'Tom', 'Alex']

    方法二:借助open的writeback=True参数来实现,默认情况下该参数的值为False。

      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9 with shelve.open('shelve_demo', writeback=True) as data:
    10 data['name'] = name
    11 data['info'] = info
    12 data['name'].append('Alex')
    13 print(data['name'])
    14
    15 程序输出:
    16 ['Jack', 'Maxwell', 'Tom', 'Alex']
  • update方法
    value值的更新还有一个update方法,使用起来也比较方便:
      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9 with shelve.open('shelve_demo', writeback=True) as data:
    10 data['name'] = name
    11 data['info'] = info
    12 data.update({'name':['Jack', 'Maxwell', 'Tom', 'Alex']}) #这里也是重新赋值
    13 print(data['name'])
    14
    15 程序输出:
    16 ['Jack', 'Maxwell', 'Tom', 'Alex']

    重新load一下看看结果:

      1 import shelve
    2
    3 def member_info(name, age):
    4 print("Member info:", name, age)
    5
    6 name = ['Jack', 'Maxwell', 'Tom']
    7 info = {'name':'Maxwell', 'age':18}
    8
    9
    10 with shelve.open('shelve_demo') as data:
    11 print(data['name'])
    12
    13 程序输出:
    14 ['Jack', 'Maxwell', 'Tom', 'Alex']
  • get方法
    通过shelve.open反序列化load对象到内存后,可以通过get方法来获取key对应的value:
      1 __author__ = 'Maxwell'
    2
    3 import shelve
    4
    5 def member_info(name, age):
    6 print("Member info:", name, age)
    7
    8 name = ['Jack', 'Maxwell', 'Tom']
    9 info = {'name':'Maxwell', 'age':18}
    10
    11
    12 with shelve.open('shelve_demo') as data:
    13 print(data.get('name'))
    14
    15 程序输出:
    16 ['Jack', 'Maxwell', 'Tom', 'Alex']

三、总结

shelve概念总结:

  • shelve模块可以看做是pickle模块的升级版,因为shelve使用的就是pickle的序列化协议,但是shelve比pickle提供的操作方式更加简单、方便。
  • shelve模块相对于其它两个模块在将Python数据持久化到本地磁盘时有一个很明显的优点就是,它允许我们可以像操作dict一样操作被序列化的数据,而不必一次性的保存或读取所有数据。
  • shelve模块持久化支持更多的python数据类型

使用建议:

  1. 需要与外部系统交互时用json模块;
  2. 需要将少量、简单Python数据持久化到本地磁盘文件时可以考虑用pickle模块;
  3. 需要将大量Python数据持久化到本地磁盘文件或需要一些简单的类似数据库的增删改查功能时,可以考虑用shelve模块。

以上内容摘自大神博客http://www.cnblogs.com/yyds/p/6563608.html

day5-shelve模块的更多相关文章

  1. 小白的Python之路 day5 shelve模块讲解

    shelve模块讲解 一.概述 之前我们说不管是json也好,还是pickle也好,在python3中只能dump一次和load一次,有什么方法可以向dump多少次就dump多少次,并且load不会出 ...

  2. day5模块学习--shelve模块

    shelve模块 shelve类似于一个key-value数据库,可以很方便的用来保存Python的内存对象,其内部使用pickle来序列化数据,简单来说,使用者可以将一个列表.字典.或者用户自定义的 ...

  3. Python学习-day5 常用模块

    day5主要是各种常用模块的学习 time &datetime模块 random os sys shutil json & picle shelve xml处理 yaml处理 conf ...

  4. Python 之路 Day5 - 常用模块学习

    本节大纲: 模块介绍 time &datetime模块 random os sys shutil json & picle shelve xml处理 yaml处理 configpars ...

  5. python序列化: json & pickle & shelve 模块

    一.json & pickle & shelve 模块 json,用于字符串 和 python数据类型间进行转换pickle,用于python特有的类型 和 python的数据类型间进 ...

  6. python pickle 和 shelve模块

    pickle和shelve模块都可以把python对象存储到文件中,下面来看看它们的用法吧 1.pickle 写: 以写方式打开一个文件描述符,调用pickle.dump把对象写进去 dn = {'b ...

  7. shelve模块

    #coding:utf-8 __author__ = 'similarface' #email:similarface@outlook.com ''' shelve模块: 映射容器 存储对象,被存储的 ...

  8. python 学习day5(模块)

    一.模块介绍 模块,用一砣代码实现了某个功能的代码集合. 类似于函数式编程和面向过程编程,函数式编程则完成一个功能,其他代码用来调用即可,提供了代码的重用性和代码间的耦合.而对于一个复杂的功能来,可能 ...

  9. s14 第5天 时间模块 随机模块 String模块 shutil模块(文件操作) 文件压缩(zipfile和tarfile)shelve模块 XML模块 ConfigParser配置文件操作模块 hashlib散列模块 Subprocess模块(调用shell) logging模块 正则表达式模块 r字符串和转译

    时间模块 time datatime time.clock(2.7) time.process_time(3.3) 测量处理器运算时间,不包括sleep时间 time.altzone 返回与UTC时间 ...

  10. python之shelve模块详解

    一.定义 Shelve是对象持久化保存方法,将对象保存到文件里面,缺省(即默认)的数据存储文件是二进制的. 二.用途 可以作为一个简单的数据存储方案. 三.用法 使用时,只需要使用open函数获取一个 ...

随机推荐

  1. 协程+IO切换实现并发

    from gevent import monkey # 以后代码中遇到IO都会自动执行greenlet的switch进行切换 monkey.patch_all() import requests im ...

  2. KMP算法最浅显理解——一看就明确

    说明 KMP算法看懂了认为特别简单,思路非常easy,看不懂之前.查各种资料,看的稀里糊涂.即使网上最简单的解释,依旧看的稀里糊涂. 我花了半天时间,争取用最短的篇幅大致搞明确这玩意究竟是啥. 这里不 ...

  3. (4.17)sql server中的uuid获取与使用

    sql server中的uuid  建表: 1.自增长 studentno int primary key identity(1,1)——bigint也是可以的 2.创建uuidcustomerid  ...

  4. mac截屏

    shift+command+3 : 截全屏 shift+command+4 : 出现十字架的坐标图标,画框截图

  5. r.js打包注意事项 r.js打包 这个是配合require.js打包的

    这个./代表的是当前文件的父目录....打包的资源一定要在这个父目录中下面才行,,,,一定一定,要放在这个目录一下才能被正确找到. 不然只是copy了一份一模一样的文件夹和文件过去,并不会处理压缩啥的 ...

  6. 199. Binary Tree Right Side View -----层序遍历

    Given a binary tree, imagine yourself standing on the right side of it, return the values of the nod ...

  7. JAVA接口中不可以有静态方法吗

    1. 接口中每一个方法也是隐式抽象的,接口中的方法会被隐式的指定为 public abstract(只能是 public abstract,其他修饰符都会报错),所以不能含有静态代码块以及静态方法(用 ...

  8. SET 语句积累

    SET IDENTITY_INSERT 表名字 off 注解: 把值插入到自动编号(或者说是标识列,IDENTITY)中去,需要设定 SET IDENTITY_INSERT 语法:SET IDENTI ...

  9. c语言URL通过Http下载mp3 格式

    通过http协议下载MP3的关键就是 整块打包,一块一块向文件里面存储.读取的时候用二进制 /***szWebAddr: 页面地址(包含host+addr) szMp3FileName:将要存储文件的 ...

  10. python3:利用smtplib库和smtp.qq.com邮件服务器发送邮件

    python3:利用smtplib库和smtp.qq.com邮件服务器发送邮件 使用qq的邮件服务器需要注意的两个地方主要是: 1.协议问题 使用465端口 SSL 协议 2.口令问题 出现SMTPA ...