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

作者:邓谦DQ 时间:2022-10-12 04:23:55 

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,shelve
0
投稿

猜你喜欢

  • django有外键关系的两张表如何相互查找

    2023-10-19 06:08:45
  • Python线程之定位与销毁的实现

    2023-09-25 11:47:19
  • python:批量统计xml中各类目标的数量案例

    2021-11-17 05:22:44
  • Python连接SQLServer2000的方法详解

    2023-07-23 09:02:12
  • python 中的np.zeros()和np.ones()函数详解

    2023-01-14 10:23:58
  • 通过字符串导入 Python 模块的方法详解

    2023-10-15 03:00:56
  • Python %r和%s区别代码实例解析

    2023-10-07 17:01:08
  • Python实现k-means算法

    2023-07-01 22:06:53
  • 100%全屏布局设计

    2009-05-15 12:24:00
  • OpenCV半小时掌握基本操作之色彩空间

    2021-01-05 05:27:31
  • python的staticmethod与classmethod实现实例代码

    2022-10-02 23:56:45
  • python中实现迭代器(iterator)的方法示例

    2021-07-12 19:32:41
  • python中操作文件的模块的方法总结

    2022-02-01 04:56:14
  • Python+Pygame制作简易版2048小游戏

    2022-12-01 18:05:03
  • python 中的@运算符使用

    2023-10-24 22:23:46
  • python基础教程之自定义函数介绍

    2021-09-12 13:10:24
  • 17个Python小技巧分享

    2022-04-17 23:26:53
  • 基于PyQT5制作一个桌面摸鱼工具

    2021-02-06 17:57:21
  • 利用Pyhton中的requests包进行网页访问测试的方法

    2021-09-11 05:12:48
  • js控制多图左右滚动切换效果代码分享

    2023-08-28 00:14:32
  • asp之家 网络编程 m.aspxhome.com