循环导入是模块间相互导入且访问未初始化对象所致,并非仅因相互import就报错;典型场景是模块A导入B时,B尝试from A import尚未定义的名称;可通过延迟导入、提取公共模块或字符串类型注解解决。
Python 循环导入(circular import)发生在两个或多个模块相互 import 对方,且在导入过程中试图访问尚未完全初始化的对象时。
它不是单纯“a.py 导入 b.py,b.py 又导入 a.py”就立刻报错,而是在导入链中某处尝试使用了另一个模块

import B,同时 B 的顶层代码立即执行 from A import func
func 定义在 import B 语句之后,B 尝试导入时该函数还不存在Python 的 import 机制会缓存已开始导入但未完成的模块(放入 sys.modules),后续 import 同一模块时直接返回这个半成品模块对象。所以:
import A(而非 from A import X),通常能成功,因为只用到了模块对象本身from A import x,而 x 在 A 中定义较晚,就会触发 ImportError: cannot import name 'x' from 'A'
报错信息里常含 “cannot import name … from …”,并指向某个 import 语句。解决思路是打破依赖时机:
def f(x: 'SomeClass') -> None:),避免运行时需要解析类本质是模块初始化顺序与符号可见性的冲突,理清“谁在什么时候需要谁的什么”就能避开。