举例简单讲解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脚本生成随机IP的简单方法
Jul 30 Python
Python登录并获取CSDN博客所有文章列表代码实例
Dec 28 Python
Python使用selenium实现网页用户名 密码 验证码自动登录功能
May 16 Python
Python使用pymysql从MySQL数据库中读出数据的方法
Jul 25 Python
python中dict字典的查询键值对 遍历 排序 创建 访问 更新 删除基础操作方法
Sep 13 Python
Python 实现「食行生鲜」签到领积分功能
Sep 26 Python
Django和Flask框架优缺点对比
Oct 24 Python
tf.concat中axis的含义与使用详解
Feb 07 Python
python实现堆排序的实例讲解
Feb 21 Python
Python GUI库PyQt5图形和特效样式QSS介绍
Feb 25 Python
python3中celery异步框架简单使用+守护进程方式启动
Jan 20 Python
一文读懂python Scrapy爬虫框架
Feb 24 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转义输出HTML到JavaScript
2015/03/27 PHP
PHP实现微信模拟登陆并给用户发送消息的方法【文字,图片,图文】
2017/06/29 PHP
PHP实现的自定义图像居中裁剪函数示例【测试可用】
2017/08/11 PHP
laravel 解决路由除了根目录其他都404的问题
2019/10/18 PHP
iframe自适应宽度、高度 ie6 7 8,firefox 3.86下测试通过
2010/07/29 Javascript
jquery调用asp.net 页面后台的实现代码
2011/04/27 Javascript
javascript window.confirm确认 取消对话框实现代码小结
2012/10/21 Javascript
javascript去除字符串左右两端的空格
2015/02/05 Javascript
DOM操作一些常用的属性汇总
2015/03/13 Javascript
javascript入门教程基础篇
2015/11/16 Javascript
iscroll碰到Select无法选择下拉刷新的解决办法
2016/05/21 Javascript
jQuery 选择同时包含两个class的元素的实现方法
2016/06/01 Javascript
浅谈Cookie的生命周期问题
2016/08/02 Javascript
微信小程序 image组件binderror使用例子与js中的onerror区别
2017/02/15 Javascript
JS传参及动态修改页面布局
2017/04/13 Javascript
JavaScript中发出HTTP请求最常用的方法
2018/07/12 Javascript
JS中的防抖与节流及作用详解
2019/04/01 Javascript
基于JS实现简单滑块拼图游戏
2019/10/12 Javascript
vue-router 控制路由权限的实现
2020/09/24 Javascript
Python编程实现数学运算求一元二次方程的实根算法示例
2017/04/02 Python
Python列表推导式与生成器表达式用法示例
2018/02/08 Python
python+numpy+matplotalib实现梯度下降法
2018/08/31 Python
Python3.5集合及其常见运算实例详解
2019/05/01 Python
树莓派采用socket方式文件传输(python)
2019/06/22 Python
使用Python实现文字转语音并生成wav文件的例子
2019/08/08 Python
HTML5 File API改善网页上传功能
2009/08/19 HTML / CSS
服务员岗位责任制
2014/02/11 职场文书
信息管理应届生求职信
2014/03/07 职场文书
优秀教导主任事迹材料
2014/05/09 职场文书
售后服务承诺书怎么写
2014/05/21 职场文书
新学期开学标语
2014/06/30 职场文书
小学校园广播稿集锦
2014/10/04 职场文书
500字小学生检讨书
2015/02/19 职场文书
2015年度房地产工作总结
2015/04/09 职场文书
Python基础之函数嵌套知识总结
2021/05/23 Python
python如何为list实现find方法
2022/05/30 Python