summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlex Waygood <Alex.Waygood@Gmail.com>2023-05-07 17:45:09 (GMT)
committerGitHub <noreply@github.com>2023-05-07 17:45:09 (GMT)
commit1b19bd1a88e6c410fc9cd08db48e0d35cfa8bb5a (patch)
tree3b858db3317ce265f4830ec9556988874acfdb5c
parent60f588478f0a3d88e86b97acecbcb569142f4636 (diff)
downloadcpython-1b19bd1a88e6c410fc9cd08db48e0d35cfa8bb5a.zip
cpython-1b19bd1a88e6c410fc9cd08db48e0d35cfa8bb5a.tar.gz
cpython-1b19bd1a88e6c410fc9cd08db48e0d35cfa8bb5a.tar.bz2
gh-103193: cache calls to `inspect._shadowed_dict` in `inspect.getattr_static` (#104267)
Co-authored-by: Carl Meyer <carl@oddbird.net>
-rw-r--r--Doc/whatsnew/3.12.rst7
-rw-r--r--Lib/inspect.py8
-rw-r--r--Lib/test/test_inspect.py22
3 files changed, 32 insertions, 5 deletions
diff --git a/Doc/whatsnew/3.12.rst b/Doc/whatsnew/3.12.rst
index ec04178..65b3e9f 100644
--- a/Doc/whatsnew/3.12.rst
+++ b/Doc/whatsnew/3.12.rst
@@ -342,8 +342,9 @@ inspect
(Contributed by Thomas Krennwallner in :issue:`35759`.)
* The performance of :func:`inspect.getattr_static` has been considerably
- improved. Most calls to the function should be around 2x faster than they
- were in Python 3.11. (Contributed by Alex Waygood in :gh:`103193`.)
+ improved. Most calls to the function should be at least 2x faster than they
+ were in Python 3.11, and some may be 6x faster or more. (Contributed by Alex
+ Waygood in :gh:`103193`.)
pathlib
-------
@@ -597,7 +598,7 @@ typing
:func:`runtime-checkable protocols <typing.runtime_checkable>` has changed
significantly. Most ``isinstance()`` checks against protocols with only a few
members should be at least 2x faster than in 3.11, and some may be 20x
- faster or more. However, ``isinstance()`` checks against protocols with seven
+ faster or more. However, ``isinstance()`` checks against protocols with fourteen
or more members may be slower than in Python 3.11. (Contributed by Alex
Waygood in :gh:`74690` and :gh:`103193`.)
diff --git a/Lib/inspect.py b/Lib/inspect.py
index 95da7fb..a64e85e 100644
--- a/Lib/inspect.py
+++ b/Lib/inspect.py
@@ -1794,8 +1794,9 @@ def _check_class(klass, attr):
return entry.__dict__[attr]
return _sentinel
-def _shadowed_dict(klass):
- for entry in _static_getmro(klass):
+@functools.lru_cache()
+def _shadowed_dict_from_mro_tuple(mro):
+ for entry in mro:
dunder_dict = _get_dunder_dict_of_class(entry)
if '__dict__' in dunder_dict:
class_dict = dunder_dict['__dict__']
@@ -1805,6 +1806,9 @@ def _shadowed_dict(klass):
return class_dict
return _sentinel
+def _shadowed_dict(klass):
+ return _shadowed_dict_from_mro_tuple(_static_getmro(klass))
+
def getattr_static(obj, attr, default=_sentinel):
"""Retrieve attributes without triggering dynamic lookup via the
descriptor protocol, __getattr__ or __getattribute__.
diff --git a/Lib/test/test_inspect.py b/Lib/test/test_inspect.py
index 42e3d70..dd0325a 100644
--- a/Lib/test/test_inspect.py
+++ b/Lib/test/test_inspect.py
@@ -2111,6 +2111,28 @@ class TestGetattrStatic(unittest.TestCase):
self.assertEqual(inspect.getattr_static(foo, 'a'), 3)
self.assertFalse(test.called)
+ def test_mutated_mro(self):
+ test = self
+ test.called = False
+
+ class Foo(dict):
+ a = 3
+ @property
+ def __dict__(self):
+ test.called = True
+ return {}
+
+ class Bar(dict):
+ a = 4
+
+ class Baz(Bar): pass
+
+ baz = Baz()
+ self.assertEqual(inspect.getattr_static(baz, 'a'), 4)
+ Baz.__bases__ = (Foo,)
+ self.assertEqual(inspect.getattr_static(baz, 'a'), 3)
+ self.assertFalse(test.called)
+
def test_custom_object_dict(self):
test = self
test.called = False