sys.path是一个list,存放了当前所有import搜索的路径。
绝大多数情况下,首个元素是"",即空串,也就是说,当你import re的时候,首先搜索的是"re",这时候就会直接在当前目录搜索到re。如下图:
解决方法也有不止一个:
1,在当前目录创建一个子文件夹,例如mine,在mine里放一个__init__.py,不用有代码,空文件即可。使用的时候:import mine.re
2,在sys.path上做手脚,把第一个空串挪到最后面去:
import sys
sys.path = sys.path[1:]
sys.path.append("")
这样当前目录会跑到最后面去,从而达到最后搜索的目的。
不过这很可能有副作用,即你创建的模块,不能和前面搜索路径里的任何文件重名,否则会无法import你的模块。
这也是为什么默认把当前路径放在最前搜索的原因,因为是用户自己写的,显然优先级最高。
尽管就你的问题作出分析和提出一些可能的解决办法,但是严重不建议在实际工程起和系统模块重名的文件名,这是自找麻烦。
当然,研究探讨它的实现原理,始终是值得肯定的。
最后强调一下,当你在当前文件夹创建了一个sys.py时,方法2不好用!
如果你有两个同名的模块,那么你只能导人它们中的一个——默认情况下,Python总是会选择在模块搜索路径sys.path中最左边的那一项。如果你偏爱的模块和顶层脚本在同一目录下,那就不成问题由于顶层脚本的主目录总是模块搜索路径中的第一项,因此它的内容总是会首先被自动定位。然而对于跨目录的导入,模块搜索路径的线性本质意味着同名的文件会产生冲突。要修复这一冲突,要么避免同名文件。如果你需要同时访问两个同名的文件,那么就要把两个源文件分别放入子目录中,这样包导入目录名称将使得模块引用唯一。只要外围的包目录名称是唯一的,你就能访问同名模块中的任意一个,或是全部的两个。
注意,如果你不小心为自己的模块使用了一个名称,而它碰巧和你需要使用的标准库模块的名称相同,那么也会出现这一问题。这是因为程序主目录(或是模块路径中靠前的另一个目录)下的本地模块会隐藏和替换标准库模块。
要修复这种覆盖,要么避免使用和你需要的另一模块相同的名称,要么把模块放到一个包目录下然后使用Python 3.X的包相对导入模型(包相对导入在2.X版本中是一个可选的功能)。在包相对导入模型下,普通导入会跳过包目录,因此你可以获取标准库版本,但在必要时特殊的点号开头导入语句仍然可以选取同名模块的本地版本。
import os# 参数设置
# 自行定义源文件地址和目标地址
_TARGET_DIR = "./copied_files/"
_SOURCE_DIR = "./source_files/"
# 你自己定义的提取特定信息的方法:
def extract(filename):
# 提取信息后返回
with open(filename, "r") as f:
info = f.readlines()
return info
# 使用os.listdir()方法获取源文件夹中所有文件
# 有时系统内会有些隐藏文件以"."开头,需要剔除
files = [file for file in os.listdir(_SOURCE_DIR) if not file.startswith(".")]
for filename in files:
# 1.读取文件并提取信息:
print("正在处理{}...".format(filename))
info = extract(_SOURCE_DIR + filename)
# 2.在目标文件夹创建同名文件,并将信息写入
# 写入部分或需要根据需要调整
with open(_TARGET_DIR + filename, "w") as f:
for line in info:
f.write(line)
print("处理完成!")