经常有小伙伴提问,为什么有时候执行某个脚本就会报错:
报 no module named xxx
这种问题一定会有人会热心的出来说,这个模块导入失败,没有找到这个模块。
但是问题并不是这么简单...
原来小伙伴的问题难点在于,同样的脚本,在pycharm中运行就没问题。在终端运行就报这个错误。
本节课就是要探寻一下这种报错到底是怎么回事,又要怎么解决?
首先看简单的例子:
这个简单的demo中,t1模块下有 t2模块和p1.py文件,t2模块内有p2.py文件。
p1.py的内容是输出一句话:
p2.py 的内容则是直接导入p1 :
注意,p1 其实 是位于 p2的上级模块 下的,正常来说,在pycharm中这么从上级导入并不是很被推荐。
现在让我们在pycharm中运行一下p2.py
结果:
可以看到,p1成功的被导入进来了。
然后我们再去终端执行p2,结果如下:
看到了吧,问题出现了。
这里我大致给大家解释下原因。在pycharm的项目TTT中,我们执行p2.py的时候,其实Pycharm是可以搜寻到TTT下的模块的,也就是 t1 模块。所以能正常执行。
但是当我们在终端对同一个脚本p2.py 执行时,终端却没有pycharm那么大的能力,去搜索到这个所谓的t1模块。所以就会报错 ,找不到t1。
如果导入的并不是上级模块的文件,而是同级/下级的文件,那么就可以找到了。这也是为什么不推荐从上级文件夹/模块导入文件的原因之一。
不过既然已经这样了,我们总要有个解决办法,让这个脚本即使在终端也能够找到t1模块。
那么就需要在脚本中加入 代码:
也就是临时寻址路径。
然后接下来我们要做的就是在那个sys.path.append() 的括号内,写入我们要增加的路径即可。
但是这里很多小伙伴都懵了,不知道写什么,如果写错了,那么结果还是一样报错找不到模块。
那现在我们再来看看之前的报错:
注意,这里是报错t1模块找不到。
那么这个路径的值,既不是写到t1本身,更不是写到t1下面的p1.py 。!!!
此时 你去终端执行p2.py仍然会报一样的错误。
为什么这两种都不对?因为你要给系统说个地址,而不是要写到这个地址之内明白么?
好比说你要找一个房间,你有房号。
那么你应该提供给系统的是这个房间所在的具体楼层即可,系统会自动在这个楼层上用房号去找到这个房间。
结果你提供的是这个房间内的,然后系统就会傻傻的去这个房间内用房号去找,那能找到么?那就属于是,不识庐山真面目,只缘身在此山中 了。
所以正确答案是写到 TTT下即可:
这时候你再去终端执行,结果就正常了:
好了,为什么要突然说这个呢?因为在我的培训中,有测试平台关联游离脚本的设计,此时用的shell命令去调用可插拔的脚本,结果在终端执行的时候,有的小伙伴路径写错就会提示找不到脚本的父级模块等问题。
所以这里集中讲一下原理,以免大家之后遇到同样问题 就只能瞎试。