파이썬 - 'urllib.request' 모듈의 명시적/암시적 로딩 차이
(파이썬 2 환경은 잊기로 하고) 기본 파이썬 환경에서는 urllib.request 모듈은 로딩돼 있지 않는데요, 즉 sys.modules에 등록돼 있지 않습니다.
$ python3
Python 3.8.10 (default, Mar 18 2025, 20:04:55)
[GCC 9.4.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import sys
>>> print(sys.modules['urllib.request'])
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
KeyError: 'urllib.request'
>>>
$ cat main.py
import sys
print(sys.modules['urllib.request'])
$ python3 main.py
Traceback (most recent call last):
File "main.py", line 3, in <module>
print(sys.modules['urllib.request'])
KeyError: 'urllib.request'
하지만, urllib.request 모듈을 로드하도록 유발하는 코드가 파이썬 내부에서 심심치 않게 실행되므로 이에 따른 부작용(side-effect)를 감안해야 합니다. 가령, 단순히 오류만 발생했어도,
$ python3
Python 3.8.10 (default, Mar 18 2025, 20:04:55)
[GCC 9.4.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> t # 이 코드의 실행으로 오류 처리하는 과정에서 urllib.request 모듈이 로드됨
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 't' is not defined
>>> import sys
>>> print(sys.modules['urllib.request'])
<module 'urllib.request' from '/usr/lib/python3.8/urllib/request.py'>
저런 식으로 urllib.request 모듈이 로딩되기도 합니다.
'urllib.request' 모듈에는 또 다른 특별함이 하나 더 있는데요, __import__('...') 함수와,
import sys
lm = __import__('urllib.request') # 여기서는 'urllib.request' 이름으로 로딩했지만,
print(lm)
print(sys.modules['urllib']) # sys.modules에는 'urllib' 이름으로 등록되고, 그 경로는 ./urllib/__init__.py
print(sys.modules['urllib.request']) # 오히려 동일한 이름으로는 ./urllib/request.py라는 다른 모듈이 등록됨
''' 실행 결과
$ python3 main.py
<module 'urllib' from '/usr/lib/python3.8/urllib/__init__.py'>
<module 'urllib' from '/usr/lib/python3.8/urllib/__init__.py'>
<module 'urllib.request' from '/usr/lib/python3.8/urllib/request.py'>
'''
import '...'에 의한 로딩 차이가 있다는 점입니다.
import sys
print(sys.modules.get('urllib.request')) # 'urllib.request' 이름으로 등록돼 있지 않은 것을 확인,
import urllib.request # import 문에 의해 로딩된 것은 'urllib.request' 이름으로 등록됨
print(sys.modules['urllib.request']) # 따라서 'urllib.request' 이름으로 모듈이 존재하고,
# 그 경로는 ./urllib/request.py
8
''' 실행 결과
$ python3 main.py
None
<module 'urllib.request' from '/usr/lib/python3.8/urllib/request.py'>
'''
__import__처럼 동적으로 로딩을 원하는데 import 구문과 유사한 동작을 원한다면 그나마 importlib.import_module가 더 나은 후보입니다.
import sys
import importlib
lm = importlib.import_module('urllib.request')
print(lm) # 출력 결과: <module 'urllib.request' from '/usr/lib/python3.8/urllib/request.py'>
print(sys.modules['urllib.request']) # 출력 결과: <module 'urllib.request' from '/usr/lib/python3.8/urllib/request.py'>
파이썬의 역사가 오래된 만큼, 여러 가지 이유로 인해 추가됐던 유사한 기능들의 세세한 차이를 이제 와서 이해하기란 어려운 것 같습니다. ^^
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]