shelve类似于一个key-value数据库,可以很方便的用来保存Python的内存对象,其内部使用pickle来序列化数据,简单来说,使用者可以将一个列表、字典、或者用户自定义的类实例保存到shelve中,下次需要用的时候直接取出来,就是一个Python内存对象,不需要像传统数据库一样,先取出数据,然后用这些数据重新构造一遍所需要的对象。下面是简单示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
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() |
这个有一个潜在的小问题,如下:
1
2
3
4
5
6
|
>>> 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']的时候,它又从源中读取了一个拷贝,所以,你新修改的内容并不会出现在拷贝中,解决的办法就是,第一个是利用一个缓存的变量,如下所示
1
2
3
4
5
|
>>> temp = s[ 'x' ] >>> temp.append( 'd' ) >>> s[ 'x' ] = temp >>> s[ 'x' ] [ 'a' , 'b' , 'c' , 'd' ] |
在python2.4以后有了另外的方法,就是把open方法的writeback参数的值赋为True,这样的话,你open后所有的内容都将在cache中,当你close的时候,将全部一次性写到硬盘里面。如果数据量不是很大的时候,建议这么做。
下面是一个基于shelve的简单数据库的代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
#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() |