举例简单讲解Python中的数据存储模块shelve的用法


Posted in Python onMarch 03, 2016

shelve类似于一个key-value数据库,可以很方便的用来保存Python的内存对象,其内部使用pickle来序列化数据,简单来说,使用者可以将一个列表、字典、或者用户自定义的类实例保存到shelve中,下次需要用的时候直接取出来,就是一个Python内存对象,不需要像传统数据库一样,先取出数据,然后用这些数据重新构造一遍所需要的对象。下面是简单示例:

import shelve


def test_shelve():
  # open 返回一个Shelf类的实例
  #
  # 参数flag的取值范围:
  # 'r':只读打开
  # 'w':读写访问
  # 'c':读写访问,如果不存在则创建
  # 'n':读写访问,总是创建新的、空的数据库文件
  #
  # protocol:与pickle库一致
  # writeback:为True时,当数据发生变化会回写,不过会导致内存开销比较大
  d = shelve.open('shelve.db', flag='c', protocol=2, writeback=False)
  assert isinstance(d, shelve.Shelf)

  # 在数据库中插入一条记录
  d['abc'] = {'name': ['a', 'b']}
  d.sync()

  print d['abc']

  # writeback是False,因此对value进行修改是不起作用的
  d['abc']['x'] = 'x'
  print d['abc'] # 还是打印 {'name': ['a', 'b']}

  # 当然,直接替换key的value还是起作用的
  d['abc'] = 'xxx'
  print d['abc']

  # 还原abc的内容,为下面的测试代码做准备
  d['abc'] = {'name': ['a', 'b']}
  d.close()

  # writeback 为 True 时,对字段内容的修改会writeback到数据库中。
  d = shelve.open('shelve.db', writeback=True)

  # 上面我们已经保存了abc的内容为{'name': ['a', 'b']},打印一下看看对不对
  print d['abc']

  # 修改abc的value的部分内容
  d['abc']['xx'] = 'xxx'
  print d['abc']
  d.close()

  # 重新打开数据库,看看abc的内容是否正确writeback
  d = shelve.open('shelve.db')
  print d['abc']
  d.close()

这个有一个潜在的小问题,如下:

>>> import shelve 
>>> s = shelve.open('test.dat') 
>>> s['x'] = ['a', 'b', 'c'] 
>>> s['x'].append('d') 
>>> s['x'] 
['a', 'b', 'c']

存储的d到哪里去了呢?其实很简单,d没有写回,你把['a', 'b', 'c']存到了x,当你再次读取s['x']的时候,s['x']只是一个拷贝,而你没有将拷贝写回,所以当你再次读取s['x']的时候,它又从源中读取了一个拷贝,所以,你新修改的内容并不会出现在拷贝中,解决的办法就是,第一个是利用一个缓存的变量,如下所示

>>> temp = s['x'] 
>>> temp.append('d') 
>>> s['x'] = temp 
>>> s['x'] 
['a', 'b', 'c', 'd']

在python2.4以后有了另外的方法,就是把open方法的writeback参数的值赋为True,这样的话,你open后所有的内容都将在cache中,当你close的时候,将全部一次性写到硬盘里面。如果数据量不是很大的时候,建议这么做。

下面是一个基于shelve的简单数据库的代码

#database.py 
import sys, shelve 
 
def store_person(db): 
  """ 
  Query user for data and store it in the shelf object 
  """ 
  pid = raw_input('Enter unique ID number: ') 
  person = {} 
  person['name'] = raw_input('Enter name: ') 
  person['age'] = raw_input('Enter age: ') 
  person['phone'] = raw_input('Enter phone number: ') 
  db[pid] = person 
 
def lookup_person(db): 
  """ 
  Query user for ID and desired field, and fetch the corresponding data from 
  the shelf object 
  """ 
  pid = raw_input('Enter ID number: ') 
  field = raw_input('What would you like to know? (name, age, phone) ') 
  field = field.strip().lower() 
  print field.capitalize() + ':', \ 
    db[pid][field] 
 
def print_help(): 
  print 'The available commons are: ' 
  print 'store :Stores information about a person' 
  print 'lookup :Looks up a person from ID number' 
  print 'quit  :Save changes and exit' 
  print '?   :Print this message' 
 
def enter_command(): 
  cmd = raw_input('Enter command (? for help): ') 
  cmd = cmd.strip().lower() 
  return cmd 
 
def main(): 
  database = shelve.open('database.dat') 
  try:  
    while True: 
      cmd = enter_command() 
      if cmd == 'store': 
        store_person(database) 
      elif cmd == 'lookup': 
        lookup_person(database) 
      elif cmd == '?': 
        print_help() 
      elif cmd == 'quit': 
        return  
  finally: 
    database.close() 
if __name__ == '__main__': main()
Python 相关文章推荐
Python批量修改文件后缀的方法
Jan 26 Python
django模型中的字段和model名显示为中文小技巧分享
Nov 18 Python
解析Python中while true的使用
Oct 13 Python
机器学习10大经典算法详解
Dec 07 Python
Python通过OpenCV的findContours获取轮廓并切割实例
Jan 05 Python
Python实现抓取HTML网页并以PDF文件形式保存的方法
May 08 Python
Python图像处理之识别图像中的文字(实例讲解)
May 10 Python
对python 匹配字符串开头和结尾的方法详解
Oct 27 Python
Django如何防止定时任务并发浅析
May 14 Python
python Django编写接口并用Jmeter测试的方法
Jul 31 Python
详解Matplotlib绘图之属性设置
Aug 23 Python
Python执行时间的几种计算方法
Jul 31 Python
Python中的数据对象持久化存储模块pickle的使用示例
Mar 03 #Python
Python和Perl绘制中国北京跑步地图的方法
Mar 03 #Python
python套接字流重定向实例汇总
Mar 03 #Python
Python设计模式中单例模式的实现及在Tornado中的应用
Mar 02 #Python
Python使用设计模式中的责任链模式与迭代器模式的示例
Mar 02 #Python
详解Python设计模式编程中观察者模式与策略模式的运用
Mar 02 #Python
Python设计模式编程中解释器模式的简单程序示例分享
Mar 02 #Python
You might like
解析php入库和出库
2013/06/25 PHP
php读取excel文件示例分享(更新修改excel)
2014/02/27 PHP
php实现通用的从数据库表读取数据到数组的函数实例
2015/03/21 PHP
在PHP中使用FastCGI解析漏洞及修复方案
2015/11/10 PHP
简单PHP会话(session)说明介绍
2016/08/21 PHP
tp5框架的增删改查操作示例
2019/10/31 PHP
制作高质量的JQuery Plugin 插件的方法
2010/04/20 Javascript
js播放wav文件(源码)
2013/04/22 Javascript
javascript中encodeURI和decodeURI方法使用介绍
2013/05/06 Javascript
JQuery.Ajax之错误调试帮助信息介绍
2013/07/04 Javascript
jquery将一个表单序列化为一个对象的方法
2013/12/02 Javascript
JavaScript也谈内存优化
2014/06/06 Javascript
jquery中的工具使用方法$.isFunction, $.isArray(), $.isWindow()
2015/08/09 Javascript
js 连续赋值的简单实现
2016/06/13 Javascript
BootStrap中Table隐藏后显示问题的实现代码
2017/08/31 Javascript
axios拦截设置和错误处理方法
2018/03/05 Javascript
vue实现的微信机器人聊天功能案例【附源码下载】
2019/02/18 Javascript
在Python中处理字符串之isdecimal()方法的使用
2015/05/20 Python
python获取list下标及其值的简单方法
2016/09/12 Python
python Django的web开发实例(入门)
2019/07/31 Python
python3 实现爬取TOP500的音乐信息并存储到mongoDB数据库中
2019/08/24 Python
python为QT程序添加图标的方法详解
2020/03/09 Python
Python自动化操作实现图例绘制
2020/07/09 Python
python3环境搭建过程(利用Anaconda+pycharm)完整版
2020/08/19 Python
html5教程调用绘图api画简单的圆形代码分享
2013/12/04 HTML / CSS
建筑毕业生自我鉴定
2013/10/18 职场文书
应急处置方案
2014/06/16 职场文书
班主任经验交流材料
2014/12/16 职场文书
党校学习个人总结
2015/02/15 职场文书
财务人员个人工作总结
2015/02/27 职场文书
公司内部升职自荐信
2015/03/27 职场文书
2015暑假假期总结
2015/07/13 职场文书
2016年植树节红领巾广播稿
2015/12/17 职场文书
2019年最新感恩节祝福语(28句)
2019/11/27 职场文书
详解JVM系列之内存模型
2021/06/10 Javascript
Redis字典实现、Hash键冲突及渐进式rehash详解
2021/09/04 Redis