diff options
author | Inada Naoki <songofacandy@gmail.com> | 2019-03-12 08:25:44 (GMT) |
---|---|---|
committer | GitHub <noreply@github.com> | 2019-03-12 08:25:44 (GMT) |
commit | f2a186712bfe726d54723eba37d80c7f0303a50b (patch) | |
tree | 66667f84b514c1550af31d16c58bc2724aa514b1 | |
parent | fc06a192fdc44225ef1cc879f615a81931ad0a85 (diff) | |
download | cpython-f2a186712bfe726d54723eba37d80c7f0303a50b.zip cpython-f2a186712bfe726d54723eba37d80c7f0303a50b.tar.gz cpython-f2a186712bfe726d54723eba37d80c7f0303a50b.tar.bz2 |
bpo-30040: new empty dict uses key-sharing dict (GH-1080)
Sizeof new empty dict becomes 72 bytes from 240 bytes (amd64).
It is same size to empty dict created by dict.clear().
-rw-r--r-- | Lib/test/test_descr.py | 6 | ||||
-rw-r--r-- | Lib/test/test_sys.py | 4 | ||||
-rw-r--r-- | Misc/NEWS.d/next/Core and Builtins/2019-03-11-22-30-56.bpo-30040.W9z8X7.rst | 2 | ||||
-rw-r--r-- | Objects/dictobject.c | 9 |
4 files changed, 13 insertions, 8 deletions
diff --git a/Lib/test/test_descr.py b/Lib/test/test_descr.py index fc885c5..e39fea6 100644 --- a/Lib/test/test_descr.py +++ b/Lib/test/test_descr.py @@ -5346,16 +5346,16 @@ class SharedKeyTests(unittest.TestCase): a, b = A(), B() self.assertEqual(sys.getsizeof(vars(a)), sys.getsizeof(vars(b))) - self.assertLess(sys.getsizeof(vars(a)), sys.getsizeof({})) + self.assertLess(sys.getsizeof(vars(a)), sys.getsizeof({"a":1})) # Initial hash table can contain at most 5 elements. # Set 6 attributes to cause internal resizing. a.x, a.y, a.z, a.w, a.v, a.u = range(6) self.assertNotEqual(sys.getsizeof(vars(a)), sys.getsizeof(vars(b))) a2 = A() self.assertEqual(sys.getsizeof(vars(a)), sys.getsizeof(vars(a2))) - self.assertLess(sys.getsizeof(vars(a)), sys.getsizeof({})) + self.assertLess(sys.getsizeof(vars(a)), sys.getsizeof({"a":1})) b.u, b.v, b.w, b.t, b.s, b.r = range(6) - self.assertLess(sys.getsizeof(vars(b)), sys.getsizeof({})) + self.assertLess(sys.getsizeof(vars(b)), sys.getsizeof({"a":1})) class DebugHelperMeta(type): diff --git a/Lib/test/test_sys.py b/Lib/test/test_sys.py index 92aefd8..4bd54af 100644 --- a/Lib/test/test_sys.py +++ b/Lib/test/test_sys.py @@ -982,8 +982,10 @@ class SizeofTest(unittest.TestCase): check(int.__add__, size('3P2P')) # method-wrapper (descriptor object) check({}.__iter__, size('2P')) + # empty dict + check({}, size('nQ2P')) # dict - check({}, size('nQ2P') + calcsize('2nP2n') + 8 + (8*2//3)*calcsize('n2P')) + check({"a": 1}, size('nQ2P') + calcsize('2nP2n') + 8 + (8*2//3)*calcsize('n2P')) longdict = {1:1, 2:2, 3:3, 4:4, 5:5, 6:6, 7:7, 8:8} check(longdict, size('nQ2P') + calcsize('2nP2n') + 16 + (16*2//3)*calcsize('n2P')) # dictionary-keyview diff --git a/Misc/NEWS.d/next/Core and Builtins/2019-03-11-22-30-56.bpo-30040.W9z8X7.rst b/Misc/NEWS.d/next/Core and Builtins/2019-03-11-22-30-56.bpo-30040.W9z8X7.rst new file mode 100644 index 0000000..eacba67 --- /dev/null +++ b/Misc/NEWS.d/next/Core and Builtins/2019-03-11-22-30-56.bpo-30040.W9z8X7.rst @@ -0,0 +1,2 @@ +New empty dict uses fewer memory for now. It used more memory than empty +dict created by ``dict.clear()``. Patch by Inada Naoki. diff --git a/Objects/dictobject.c b/Objects/dictobject.c index 83cadda..108c612 100644 --- a/Objects/dictobject.c +++ b/Objects/dictobject.c @@ -691,10 +691,8 @@ clone_combined_dict(PyDictObject *orig) PyObject * PyDict_New(void) { - PyDictKeysObject *keys = new_keys_object(PyDict_MINSIZE); - if (keys == NULL) - return NULL; - return new_dict(keys, NULL); + dictkeys_incref(Py_EMPTY_KEYS); + return new_dict(Py_EMPTY_KEYS, empty_values); } /* Search index of hash table from offset of entry table */ @@ -1276,6 +1274,9 @@ _PyDict_NewPresized(Py_ssize_t minused) Py_ssize_t newsize; PyDictKeysObject *new_keys; + if (minused == 0) { + return PyDict_New(); + } /* There are no strict guarantee that returned dict can contain minused * items without resize. So we create medium size dict instead of very * large dict or MemoryError. |