举例简单讲解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函数的周期性执行实现方法
Aug 13 Python
python3个性签名设计实现代码
Jun 19 Python
Python设计模式之享元模式原理与用法实例分析
Jan 11 Python
python实现多层感知器MLP(基于双月数据集)
Jan 18 Python
Python对象转换为json的方法步骤
Apr 25 Python
Python3.5 Json与pickle实现数据序列化与反序列化操作示例
Apr 29 Python
python实现函数极小值
Jul 10 Python
Python终端输出彩色字符方法详解
Feb 11 Python
python框架Django实战商城项目之工程搭建过程图文详解
Mar 09 Python
Python调用SMTP服务自动发送Email的实现步骤
Feb 07 Python
Python中使用Selenium环境安装的方法步骤
Feb 22 Python
Python如何加载模型并查看网络
Jul 15 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
给初学者的30条PHP最佳实践(荒野无灯)
2011/08/02 PHP
洪恩在线成语词典小偷程序php版
2012/04/20 PHP
Linux下手动编译安装PHP扩展的例子分享
2014/07/15 PHP
php查看当前Session的ID实例
2015/03/16 PHP
[原创]php集成安装包wampserver修改密码后phpmyadmin无法登陆的解决方法
2016/11/23 PHP
javascript 写类方式之八
2009/07/05 Javascript
jquery监控数据是否变化(修正版)
2011/04/12 Javascript
JQuery防止退格键网页后退的实现代码
2012/03/23 Javascript
点击页面其它地方隐藏该div的两种思路
2013/11/18 Javascript
jQuery中find()方法用法实例
2015/01/07 Javascript
在linux中使用包管理器安装node.js
2015/03/13 Javascript
如何使用AngularJs打造权限管理系统【简易型】
2016/05/09 Javascript
javascript 数组去重复(在线去重工具)
2016/12/17 Javascript
vue实现简单实时汇率计算功能
2017/01/15 Javascript
Node.js中.pfx后缀文件的处理方法
2017/03/10 Javascript
Bootstrap多级菜单的实现代码
2017/05/23 Javascript
Vue请求java服务端并返回数据代码实例
2019/11/28 Javascript
html2canvas属性和使用方法以及如何使用html2canvas将HTML内容写入Canvas生成图片
2020/01/12 Javascript
js实现数据导出为EXCEL(支持大量数据导出)
2020/03/31 Javascript
vue 遮罩层阻止默认滚动事件操作
2020/07/28 Javascript
使用Python生成XML的方法实例
2017/03/21 Python
centos6.4下python3.6.1安装教程
2017/07/21 Python
python实现数独游戏 java简单实现数独游戏
2018/03/30 Python
python爬取个性签名的方法
2018/06/17 Python
Python连接Mssql基础教程之Python库pymssql
2018/09/16 Python
Python一行代码解决矩阵旋转的问题
2019/11/30 Python
keras实现调用自己训练的模型,并去掉全连接层
2020/06/09 Python
踩坑:pytorch中eval模式下结果远差于train模式介绍
2020/06/23 Python
Python实现简单猜数字游戏
2021/02/03 Python
使用CSS3实现一个3D相册效果实例
2016/12/03 HTML / CSS
html5开发之viewport使用
2013/10/17 HTML / CSS
美国马匹用品和马钉购物网站:State Line Tack
2018/08/05 全球购物
战略合作意向书
2014/07/29 职场文书
2015年教师见习期工作总结
2015/05/20 职场文书
2015年“我们的节日·重阳节”活动总结
2015/07/29 职场文书
Java实现给Word文件添加文字水印
2022/02/15 Java/Android