举例简单讲解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实现比较两个文件夹中代码变化的方法
Jul 10 Python
python中pygame针对游戏窗口的显示方法实例分析(附源码)
Nov 11 Python
python爬虫headers设置后无效的解决方法
Oct 21 Python
python实现分页效果
Oct 25 Python
详解Python开发中如何使用Hook技巧
Nov 01 Python
python实现自动登录
Sep 17 Python
Python实现查找数组中任意第k大的数字算法示例
Jan 23 Python
Python3安装psycopy2以及遇到问题解决方法
Jul 03 Python
Python基于jieba, wordcloud库生成中文词云
May 13 Python
Python Flask异步发送邮件实现方法解析
Aug 01 Python
python和node.js生成当前时间戳的示例
Sep 29 Python
Python  Asyncio模块实现的生产消费者模型的方法
Mar 01 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的MySQL连接类
2013/06/07 PHP
php启用zlib压缩文件的配置方法
2013/06/12 PHP
使用PHP下载CSS文件中的图片的代码
2013/09/24 PHP
PHP登录(ajax提交数据和后台校验)实例分享
2016/12/29 PHP
php获取linux命令结果的实例
2017/03/13 PHP
PHP单例模式数据库连接类与页面静态化实现方法
2019/03/20 PHP
laravel与thinkphp之间的区别与优缺点
2021/03/02 PHP
一个简单的JavaScript 日期计算算法
2009/09/11 Javascript
浏览器窗口加载和大小改变事件示例
2014/02/27 Javascript
JS使用replace()方法和正则表达式进行字符串的搜索与替换实例
2014/04/10 Javascript
nodejs文件操作模块FS(File System)常用函数简明总结
2014/06/05 NodeJs
深入理解Node.js的HTTP模块
2016/10/12 Javascript
AngularJS中的Promise详细介绍及实例代码
2016/12/13 Javascript
实例讲解DataTables固定表格宽度(设置横向滚动条)
2017/07/11 Javascript
jQuery动态添加.active 实现导航效果代码思路详解
2017/08/29 jQuery
JavaScript设计模式之观察者模式实例详解
2019/01/16 Javascript
[30:00]完美世界DOTA2联赛PWL S2 Rebirth vs LBZS 第二场 11.28
2020/12/01 DOTA
多版本Python共存的配置方法
2017/05/22 Python
python虚拟环境virtualenv的安装与使用
2017/09/21 Python
Python即时网络爬虫项目启动说明详解
2018/02/23 Python
python 文件查找及内容匹配方法
2018/10/25 Python
Mac下Anaconda的安装和使用教程
2018/11/29 Python
python3实现名片管理系统
2020/11/29 Python
python判断计算机是否有网络连接的实例
2018/12/15 Python
浅谈python requests 的put, post 请求参数的问题
2019/01/02 Python
在python中实现强制关闭线程的示例
2019/01/22 Python
Pycharm中出现ImportError:DLL load failed:找不到指定模块的解决方法
2019/09/17 Python
怎么快速自学python
2020/06/22 Python
Farnell德国:电子元器件供应商
2018/07/10 全球购物
BAILEY 44官网:美国制造的女性服装
2019/07/01 全球购物
广州御银科技股份有限公司试卷(C++)
2016/11/04 面试题
北京振戎融通Java面试题
2015/09/03 面试题
计算机专业自我鉴定
2013/10/15 职场文书
党员自我评价分享
2013/12/13 职场文书
小学生读书笔记
2015/07/01 职场文书
用golang如何替换某个文件中的字符串
2021/04/25 Golang