-
-
Notifications
You must be signed in to change notification settings - Fork 31.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Speed up importing typing
just for TYPE_CHECKING
#132049
base: main
Are you sure you want to change the base?
Conversation
This makes importing anything else from typing slower, and it likely breaks |
It also means the A |
By the cost of a string comparison, a lookup for Before:
After:
While the numbers are certainly measurably different:
Which would likely be fixed by defining |
Can you test with This is a (rough) script I use to benchmark import time measurements: import subprocess, sys
import statistics
BASE_CMD = (sys.executable, '-Ximporttime', '-S', '-c',)
def run_importtime(mod: str) -> str:
return subprocess.run(BASE_CMD + (f'import {mod}',), check=True, capture_output=True, encoding='utf-8').stderr
for mod in sys.argv[1:]:
for _ in range(5): # warmup
lines = run_importtime(mod)
print(lines.partition('\n')[0])
own_times = []
cum_times = []
for _ in range(50):
lines = run_importtime(mod)
final_line = lines.rstrip().rpartition('\n')[-1]
# print(final_line)
# import time: {own} | {cum} | {mod}
own, cum = map(int, final_line.split()[2:5:2])
own_times.append(own)
cum_times.append(cum)
own_times.sort()
cum_times.sort()
own_times[:] = own_times[10:-10]
cum_times[:] = cum_times[10:-10]
for label, times in [('own', own_times), ('cumulative', cum_times)]:
print()
print(f'import {mod}: {label} time')
print(f'mean: {statistics.mean(times):.3f} µs')
print(f'median: {statistics.median(times):.3f} µs')
print(f'stdev: {statistics.stdev(times):.3f}')
print('min:', min(times))
print('max:', max(times)) |
(For my own edification, sorry for the noise) "the cache" here is a combo of If so, IIUC I think the numbers are precisely showing us that this change as it stands has a small-but-measurable penalty precisely because we don't use the (latter) cache. I'd venture to guess |
Changing the code to: import sys
__mod__ = sys.modules[__name__]
def __getattr__(name):
if name == "TYPE_CHECKING":
return False
import _typing
attr = getattr(_typing, name)
__mod__.__dict__[name] = attr
return attr brings it down to |
(Somewhat demonstrative PR, but happy to give it the rest of "the treatment" (e.g. tests, documentation, etc...) if it seems OK)
Made in the context of https://discuss.python.org/t/pep-781-make-type-checking-a-built-in-constant/85728/122
I also think
_typing.py
likely is a bad name (undoubtedly going to collide with some user's usage of_typing.py
) so happy to bikeshed on alternate names.On a fresh Ubuntu VM, I installed
uv
, thenuv install 3.14
) :Interpreter baseline:
Importing
typing
forTYPE_CHECKING
After this change: