举例简单讲解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 网络编程起步(Socket发送消息)
Sep 06 Python
【Python】Python的urllib模块、urllib2模块批量进行网页下载文件
Nov 19 Python
Python微信公众号开发平台
Jan 25 Python
pandas 获取季度,月度,年度首尾日期的方法
Apr 11 Python
Python实现压缩文件夹与解压缩zip文件的方法
Sep 01 Python
Python设计模式之适配器模式原理与用法详解
Jan 15 Python
简单了解python的break、continue、pass
Jul 08 Python
python GUI图形化编程wxpython的使用
Jul 19 Python
Python‘==‘ 及 ‘is‘相关原理解析
Sep 05 Python
使用Python实现音频双通道分离
Dec 25 Python
使用Python下载抖音各大V视频的思路详解
Feb 06 Python
详解Python函数print用法
Jun 18 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
谈一谈收音机的高放电路
2021/03/02 无线电
php动态生成函数示例
2014/03/21 PHP
PHP date()函数警告: It is not safe to rely on the system解决方法
2014/08/20 PHP
WordPress中Gravatar头像缓存到本地及相关优化的技巧
2015/12/19 PHP
php抽象方法和抽象类实例分析
2016/12/07 PHP
PHP中快速生成随机密码的几种方式
2017/04/17 PHP
phpstorm激活码2020附使用详细教程
2020/09/25 PHP
js内置对象 学习笔记
2011/08/01 Javascript
解析使用JS 清空File控件的路径值
2013/07/08 Javascript
jQuery代码实现对话框右上角菜单带关闭×
2016/05/03 Javascript
AngularJS入门教程之过滤器用法示例
2016/11/02 Javascript
Vue 2.X的状态管理vuex记录详解
2017/03/23 Javascript
js按条件生成随机json:randomjson实现方法
2017/04/07 Javascript
js实现简单页面全屏
2019/09/17 Javascript
微信小程序中网络请求缓存的解决方法
2019/12/29 Javascript
[01:12]快闪回顾DOTA2亚洲邀请赛(DAC) 静候2018新征程开启
2018/03/11 DOTA
Python 开发Activex组件方法
2009/11/08 Python
python 实现堆排序算法代码
2012/06/05 Python
通过Python爬虫代理IP快速增加博客阅读量
2016/12/14 Python
使用Python3制作TCP端口扫描器
2017/04/17 Python
python实现监控某个服务 服务崩溃即发送邮件报告
2018/06/21 Python
Django 实现图片上传和显示过程详解
2019/07/18 Python
python KNN算法实现鸢尾花数据集分类
2019/10/24 Python
Python爬虫JSON及JSONPath运行原理详解
2020/06/04 Python
python中selenium库的基本使用详解
2020/07/31 Python
自1926年以来就为冰岛保持温暖:66°North
2020/11/27 全球购物
公务员党的群众路线教育实践活动学习心得体会
2014/10/30 职场文书
婚礼庆典答谢词
2015/01/20 职场文书
跳高加油稿
2015/07/21 职场文书
高中同学会致辞
2015/08/01 职场文书
农贸批发市场管理制度
2015/08/07 职场文书
2016大学迎新晚会开场白
2015/11/24 职场文书
《检阅》教学反思
2016/02/22 职场文书
《夹竹桃》教学反思
2016/02/23 职场文书
创业计划书之冷饮店
2019/09/27 职场文书
Python+Selenium自动化环境搭建与操作基础详解
2022/03/13 Python