shelve
— Python object persistence¶
Source code: Lib/shelve.py
Un «estante» o shelve, es un objeto persistente similar a un diccionario. La diferencia con las bases de datos «dbm» es que los valores (¡no las claves!) en un estante pueden ser esencialmente objetos Python arbitrarios — cualquier cosa que el módulo pickle
pueda manejar. Esto incluye la mayoría de las instancias de clase, tipos de datos recursivos y objetos que contienen muchos subobjetos compartidos. Las claves son cadenas ordinarias.
- shelve.open(filename, flag='c', protocol=None, writeback=False, *, serializer=None, deserializer=None)¶
Abre un diccionario persistente. El nombre de archivo especificado es el nombre de archivo base para la base de datos subyacente. Como efecto secundario, se puede agregar una extensión al nombre de archivo y se puede crear más de un archivo. De forma predeterminada, el archivo de base de datos subyacente se abre para leer y escribir. El parámetro opcional flag tiene la misma interpretación que el parámetro flag de
dbm.open()
.By default, pickles created with
pickle.DEFAULT_PROTOCOL
are used to serialize values. The version of the pickle protocol can be specified with the protocol parameter.Debido a la semántica de Python, un estante no puede saber cuándo se modifica una entrada de diccionario persistente mutable. De forma predeterminada, los objetos modificados se escriben sólo cuando se asignan al estante (consulte Ejemplo). Si el parámetro opcional writeback se establece en
True
, todas las entradas a las que se accede también se almacenan en caché en la memoria y se vuelven a escribir ensync()
yclose()
; esto puede hacer que sea más práctico mutar entradas mutables en el diccionario persistente, pero, si se accede a muchas entradas, puede consumir grandes cantidades de memoria para la memoria caché, y puede hacer que la operación de cierre sea muy lenta ya que todas las entradas a las que se accede se vuelven a escribir (no hay manera de determinar qué entradas a las que se accede son mutables, ni cuáles se mutaron realmente).By default,
shelve
usespickle.dumps()
andpickle.loads()
for serializing and deserializing. This can be changed by supplying serializer and deserializer, respectively.The serializer argument must be a callable which takes an object
obj
and the protocol as inputs and returns the representationobj
as a bytes-like object; the protocol value may be ignored by the serializer.The deserializer argument must be callable which takes a serialized object given as a
bytes
object and returns the corresponding object.A
ShelveError
is raised if serializer is given but deserializer is not, or vice-versa.Distinto en la versión 3.10:
pickle.DEFAULT_PROTOCOL
is now used as the default pickle protocol.Distinto en la versión 3.11: Acepta path-like object por nombre de archivo.
Distinto en la versión 3.15.0a0 (unreleased): Accepts custom serializer and deserializer functions in place of
pickle.dumps()
andpickle.loads()
.Nota
No confíe en que el estante se cerrará automáticamente; siempre llame a
close()
explícitamente cuando ya no lo necesite, o useshelve.open()
como administrador de contexto:with shelve.open('spam') as db: db['eggs'] = 'eggs'
Advertencia
Debido a que el módulo shelve
está respaldado por pickle
, es inseguro cargar un estante desde una fuente que no es de confianza. Al igual que con el pickle, cargar un estante puede ejecutar código arbitrario.
Los objetos de estante admiten la mayoría de los métodos y operaciones admitidos por los diccionarios (excepto copiar, constructores y los operadores |
y |=
). Esto facilita la transición de scripts basados en diccionarios a aquellos que requieren almacenamiento persistente.
Se admiten dos métodos adicionales:
- Shelf.sync()¶
Write back all entries in the cache if the shelf was opened with writeback set to
True
. Also empty the cache and synchronize the persistent dictionary on disk, if feasible. This is called automatically whenreorganize()
is called or the shelf is closed withclose()
.
- Shelf.reorganize()¶
Calls
sync()
and attempts to shrink space used on disk by removing empty space resulting from deletions.Added in version 3.15.0a0 (unreleased).
- Shelf.close()¶
Sincronice y cierre el objeto persistente dict. Las operaciones en un estante cerrado fallarán con un
ValueError
.
Ver también
Persistent dictionary recipe with widely supported storage formats and having the speed of native dictionaries.
Restricciones¶
La elección de qué paquete de base de datos se utilizará (como
dbm.ndbm
odbm.gnu
) depende de la interfaz disponible. Por lo tanto, no es seguro abrir la base de datos directamente usandodbm
. La base de datos también está (desafortunadamente) sujeta a las limitaciones dedbm
, si se usa — esto significa que (la representación pickle de) los objetos almacenados en la base de datos debe ser bastante pequeña, y en casos raros las colisiones de claves pueden hacer que la base de datos rechace las actualizaciones.El módulo
shelve
no admite el acceso concurrent de lectura/escritura a los objetos almacenados. (Los accesos de lectura múltiples simultáneos son seguros). Cuando un programa tiene un estante abierto para escritura, ningún otro programa debe tenerlo abierto para lectura o escritura. El bloqueo de archivos Unix se puede usar para resolver esto, pero esto difiere entre las versiones de Unix y requiere conocimiento sobre la implementación de la base de datos utilizada.On macOS
dbm.ndbm
can silently corrupt the database file on updates, which can cause hard crashes when trying to read from the database.Shelf.reorganize()
may not be available for all database packages and may temporarily increase resource usage (especially disk space) when called. Additionally, it will never run automatically and instead needs to be called explicitly.
- class shelve.Shelf(dict, protocol=None, writeback=False, keyencoding='utf-8', *, serializer=None, deserializer=None)¶
Una subclase de
collections.abc.MutableMapping
que almacena valores pickle en el objeto dict.By default, pickles created with
pickle.DEFAULT_PROTOCOL
are used to serialize values. The version of the pickle protocol can be specified with the protocol parameter. See thepickle
documentation for a discussion of the pickle protocols.Si el parámetro writeback es
True
, el objeto mantendrá un caché de todas las entradas a las que se accedió y las volverá a escribir en el dict en las horas de sincronización y cierre. Esto permite operaciones naturales en entradas mutables, pero puede consumir mucha más memoria y hacer que la sincronización y el cierre tomen mucho tiempo.El parámetro keyencoding es la codificación utilizada para codificar las claves antes de que se utilicen con el dict subyacente.
The serializer and deserializer parameters have the same interpretation as in
open()
.El objeto
Shelf
también se puede utilizar como administrador de contexto, en cuyo caso se cerrará automáticamente cuando finalice el bloquewith
.Distinto en la versión 3.2: Se agregó el parámetro keyencoding; anteriormente, las claves siempre estaban codificadas en UTF-8.
Distinto en la versión 3.4: Agregado soporte para administrador de contexto.
Distinto en la versión 3.10:
pickle.DEFAULT_PROTOCOL
is now used as the default pickle protocol.Distinto en la versión 3.15.0a0 (unreleased): Added the serializer and deserializer parameters.
- class shelve.BsdDbShelf(dict, protocol=None, writeback=False, keyencoding='utf-8', *, serializer=None, deserializer=None)¶
A subclass of
Shelf
which exposesfirst()
,next()
,previous()
,last()
andset_location()
methods. These are available in the third-partybsddb
module from pybsddb but not in other database modules. The dict object passed to the constructor must support those methods. This is generally accomplished by calling one ofbsddb.hashopen()
,bsddb.btopen()
orbsddb.rnopen()
. The optional protocol, writeback, keyencoding, serializer and deserializer parameters have the same interpretation as inopen()
.Distinto en la versión 3.15.0a0 (unreleased): Added the serializer and deserializer parameters.
- class shelve.DbfilenameShelf(filename, flag='c', protocol=None, writeback=False, *, serializer=None, deserializer=None)¶
A subclass of
Shelf
which accepts a filename instead of a dict-like object. The underlying file will be opened usingdbm.open()
. By default, the file will be created and opened for both read and write. The optional flag parameter has the same interpretation as for theopen()
function. The optional protocol, writeback, serializer and deserializer parameters have the same interpretation as inopen()
.Distinto en la versión 3.15.0a0 (unreleased): Added the serializer and deserializer parameters.
Ejemplo¶
Para resumir la interfaz (key
es una cadena de caracteres, data
es un objeto arbitrario):
import shelve
d = shelve.open(filename) # open -- file may get suffix added by low-level
# library
d[key] = data # store data at key (overwrites old data if
# using an existing key)
data = d[key] # retrieve a COPY of data at key (raise KeyError
# if no such key)
del d[key] # delete data stored at key (raises KeyError
# if no such key)
flag = key in d # true if the key exists
klist = list(d.keys()) # a list of all existing keys (slow!)
# as d was opened WITHOUT writeback=True, beware:
d['xx'] = [0, 1, 2] # this works as expected, but...
d['xx'].append(3) # *this doesn't!* -- d['xx'] is STILL [0, 1, 2]!
# having opened d without writeback=True, you need to code carefully:
temp = d['xx'] # extracts the copy
temp.append(5) # mutates the copy
d['xx'] = temp # stores the copy right back, to persist it
# or, d=shelve.open(filename,writeback=True) would let you just code
# d['xx'].append(5) and have it work as expected, BUT it would also
# consume more memory and make the d.close() operation slower.
d.close() # close it
Exceptions¶
- exception shelve.ShelveError¶
Exception raised when one of the arguments deserializer and serializer is missing in the
open()
,Shelf
,BsdDbShelf
andDbfilenameShelf
.The deserializer and serializer arguments must be given together.
Added in version 3.15.0a0 (unreleased).