summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRaymond Hettinger <python@rcn.com>2004-12-16 16:23:40 (GMT)
committerRaymond Hettinger <python@rcn.com>2004-12-16 16:23:40 (GMT)
commitb0900e6a217480def244b0b55d718ce705e8e73a (patch)
treeded97348a4d0d9b1fd519acd9b0d1cfad4b9cc86
parente6bdb37e5bc7d250e17c3d4fef6961e178e02b64 (diff)
downloadcpython-b0900e6a217480def244b0b55d718ce705e8e73a.zip
cpython-b0900e6a217480def244b0b55d718ce705e8e73a.tar.gz
cpython-b0900e6a217480def244b0b55d718ce705e8e73a.tar.bz2
SF #1085304: Make array.array pickle-able
-rwxr-xr-xLib/test/test_array.py19
-rw-r--r--Misc/NEWS2
-rw-r--r--Modules/arraymodule.c25
3 files changed, 46 insertions, 0 deletions
diff --git a/Lib/test/test_array.py b/Lib/test/test_array.py
index b15298f..c24b41b 100755
--- a/Lib/test/test_array.py
+++ b/Lib/test/test_array.py
@@ -7,6 +7,10 @@ import unittest
from test import test_support
from weakref import proxy
import array, cStringIO, math
+from cPickle import loads, dumps
+
+class ArraySubclass(array.array):
+ pass
tests = [] # list to accumulate all tests
typecodes = "cubBhHiIlLfd"
@@ -81,6 +85,21 @@ class BaseTest(unittest.TestCase):
self.assertNotEqual(id(a), id(b))
self.assertEqual(a, b)
+ def test_pickle(self):
+ for protocol in (0, 1, 2):
+ a = array.array(self.typecode, self.example)
+ b = loads(dumps(a, protocol))
+ self.assertNotEqual(id(a), id(b))
+ self.assertEqual(a, b)
+
+ a = ArraySubclass(self.typecode, self.example)
+ a.x = 10
+ b = loads(dumps(a, protocol))
+ self.assertNotEqual(id(a), id(b))
+ self.assertEqual(a, b)
+ self.assertEqual(a.x, b.x)
+ self.assertEqual(type(a), type(b))
+
def test_insert(self):
a = array.array(self.typecode, self.example)
a.insert(0, self.example[0])
diff --git a/Misc/NEWS b/Misc/NEWS
index 6bf7ab4..570adda 100644
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -17,6 +17,8 @@ Core and builtins
Extension Modules
-----------------
+- array.array objects are now picklable.
+
- the cPickle module no longer accepts the deprecated None option in the
args tuple returned by __reduce__().
diff --git a/Modules/arraymodule.c b/Modules/arraymodule.c
index 7ed3b73..6430333 100644
--- a/Modules/arraymodule.c
+++ b/Modules/arraymodule.c
@@ -1133,6 +1133,29 @@ Byteswap all items of the array. If the items in the array are not 1, 2,\n\
4, or 8 bytes in size, RuntimeError is raised.");
static PyObject *
+array_reduce(arrayobject *array)
+{
+ PyObject *dict, *result;
+
+ dict = PyObject_GetAttrString((PyObject *)array, "__dict__");
+ if (dict == NULL) {
+ PyErr_Clear();
+ dict = Py_None;
+ Py_INCREF(dict);
+ }
+ result = Py_BuildValue("O(cs#)O",
+ array->ob_type,
+ array->ob_descr->typecode,
+ array->ob_item,
+ array->ob_size * array->ob_descr->itemsize,
+ dict);
+ Py_DECREF(dict);
+ return result;
+}
+
+PyDoc_STRVAR(array_doc, "Return state information for pickling.");
+
+static PyObject *
array_reverse(arrayobject *self, PyObject *unused)
{
register int itemsize = self->ob_descr->itemsize;
@@ -1490,6 +1513,8 @@ PyMethodDef array_methods[] = {
pop_doc},
{"read", (PyCFunction)array_fromfile, METH_VARARGS,
fromfile_doc},
+ {"__reduce__", (PyCFunction)array_reduce, METH_NOARGS,
+ array_doc},
{"remove", (PyCFunction)array_remove, METH_O,
remove_doc},
{"reverse", (PyCFunction)array_reverse, METH_NOARGS,