一、概述

前面章节我们讲述了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. 流畅的python 使用一等函数实现设计模式

    案例分析:重构“策略”模式 经典的“策略”模式 电商领域有个功能明显可以使用“策略”模式,即根据客户的属性或订单中的商品计算折扣.假如一个网店制定了下述折扣规则. 有 1000 或以上积分的顾客,每个 ...

  2. x265探索与研究(六):main()函数

    x265探索与研究(六):main()函数 x265源代码的入口函数是main(),本文分析main()的主要功能. 首先给出main()函数的功能及其代码结构:其次给出main()函数源代码以及分析 ...

  3. 编译hadoop2.6.0 cdh 5.4.5 集成snappy压缩

    原文地址:http://www.cnblogs.com/qiaoyihang/p/6995146.html 自带的为32位库,故需要把64为重编译进去 1.下载源码:http://archive-pr ...

  4. Oracle DG备库强制switch_over过程

    故障描述: 主库异常下线,需要将备库强制启动为主库,切断日志时提示需要介质恢复,执行介质恢复后,再激活日志即可进行切换 1.  执行alter database recover managed sta ...

  5. delphi webbrowser 获取iframe

    procedure TForm1.Button4Click(Sender: TObject);var Index: Integer; Document: IHTMLDocument2; FrameId ...

  6. LocalReport Print with C# C#打印RDLC

             {             ;                                                      )             {        ...

  7. 一个用于实现并行执行的 Java actor 库

    即使 Java 6 和 Java 7 中引入并发性更新,Java 语言仍然无法让并行编程变得特别容易.Java 线程.synchronized 代码块.wait/notify 和java.util.c ...

  8. 关于获得MFC窗口其它类指针的方法(csdn)

    转自:http://tieba.baidu.com/p/252804018 访问应用程序的其它类 获得CWinApp: -在CMainFrame,CChildFrame,CDocument,CView ...

  9. svn / git SourceTree

    开发使用SourceTree 忽略文件这块老弄错,这次专门博客一下,使用CocoaPods 开发项目, 忽略步骤如下:  忽略文件内容如下 *.xcworkspace xcuserdata *.loc ...

  10. dirname和shell常用命令

    $ cd `dirname $0` 和PWD%}  显示当前目录名称${#var}             替换为变量字符个数特殊变量$ 当前SHELL的PID? 前一个命令的退出状态! 后台执行的上 ...