这个错误是因为在Python 3.9及以下版本中,类型注解不支持使用"|"来联合类型。在Python 3.10及以上版本中才支持这种写法。
解决方案有两种:
升级Python到3.10或以上版本。(可能会涉及该一系列的依赖,可能要重新配置环境)
修改DINOv2的源代码,将类型注解中的"|"改为使用"Optional"或"Union"。
成功解决,但有一堆,提供一个专门的修复脚本来批量修复这些问题
# fix_dinov2_type_hints.py import os import re def fix_type_hints_in_file(file_path): """修复单个文件中的类型注解语法""" with open(file_path, 'r', encoding='utf-8') as f: content = f.read() # 备份原始内容 original_content = content # 修复所有 "type | None" 语法 content = re.sub(r'\bfloat\s*\|\s*None\b', 'Optional[float]', content) content = re.sub(r'\bint\s*\|\s*None\b', 'Optional[int]', content) content = re.sub(r'\bbool\s*\|\s*None\b', 'Optional[bool]', content) content = re.sub(r'\bstr\s*\|\s*None\b', 'Optional[str]', content) content = re.sub(r'\btorch\.Tensor\s*\|\s*None\b', 'Optional[torch.Tensor]', content) # 修复其他可能的类型联合语法(针对Python 3.10+) content = re.sub(r'\bfloat\s*\|\s*int\b', 'Union[float, int]', content) content = re.sub(r'\bint\s*\|\s*float\b', 'Union[int, float]', content) # 检查是否需要添加导入 if 'Optional' in content and 'from typing import Optional' not in content: # 在文件顶部添加导入 lines = content.split('\n') for i, line in enumerate(lines): if line.startswith('import ') or line.startswith('from ') or line.strip() == '': continue else: # 在第一个非导入/空行前插入 lines.insert(i, 'from typing import Optional, Union') break content = '\n'.join(lines) # 如果内容有变化,保存文件 if content != original_content: with open(file_path, 'w', encoding='utf-8') as f: f.write(content) print(f"已修复: {file_path}") return True return False def find_and_fix_dinov2_files(): """查找并修复DINOv2的所有Python文件""" import torch.hub hub_dir = torch.hub.get_dir() dinov2_path = os.path.join(hub_dir, 'facebookresearch_dinov2_main') if not os.path.exists(dinov2_path): print(f"未找到DINOv2路径: {dinov2_path}") return print(f"正在修复DINOv2文件在: {dinov2_path}") # 修复所有.py文件 for root, dirs, files in os.walk(dinov2_path): for file in files: if file.endswith('.py'): file_path = os.path.join(root, file) fix_type_hints_in_file(file_path) print("修复完成!") if __name__ == "__main__": find_and_fix_dinov2_files()