1. 模块
模块的查找顺序
python中的一个模块可以简单的理解成一个py文件,在导入自定义模块时模块的查找顺序可以通过sys.path 输出查看:
import sys
print(sys.path)
我电脑上的路径:
[‘D:\Pycharm\workspace\PracticePro\pack1’, ‘D:\Pycharm\workspace\PracticePro’, ‘C:\Users\Lenovo\AppData\Local\Programs\Python\Python36\python36.zip’, ‘C:\Users\Lenovo\AppData\Local\Programs\Python\Python36\DLLs’, ‘C:\Users\Lenovo\AppData\Local\Programs\Python\Python36\lib’, ‘C:\Users\Lenovo\AppData\Local\Programs\Python\Python36’,
‘C:\Users\Lenovo\AppData\Local\Programs\Python\Python36\lib\site-packages’, ‘D:\Pycharm\install\PyCharm2018.1.4\helpers\pycharm_matplotlib_backend’]
第一个路径为当前脚本的所在目录,第二个为当前项目的所在目录,第七个为在cmd控制台通过pip命令安装的包目录,重点需要关注这三个。
越靠前的路径越先查找,找到了停止查找并导入,否则一直找到最后一个目录,还没找到要导入的模块名则报ModuleNotFoundError异常。(需要注意这种查找的优先级可能会导致命名冲突,自定义模块覆盖了系统模块,所以就要求在自定义模块时不要和系统模块命名一致)
自定义模块的导入
在导入自定义模块时需要注意,如果执行脚本与被导入的模块处于同一目录下,可以直接 import 模块名,否则需要 import 包名.模块名(因为本目录找不到,会到项目目录下查找,不带包名项目路径下是找不到这个模块的)
# 导入整个模块
import 模块名
from 包名 import 模块名
from 包名 import *
# 调用
模块名.方法名()
# 导入模块中的方法/变量/类
from [包名.] 模块名 import 方法名
from [包名.] 模块名 import *
# 调用
方法名()
理解 if name == ‘main’:
需要注意的是在导入模块的时候会首先执行一遍模块中的代码,而我们导入只是为了直接拿被导入模块中的某些方法来复用,为了避免导入时所有代码执行,可以通过在被导入模块中添加if __name__ == '__main__': ,_name_ 是python中的一个魔术方法,也是一个内置方法,在当前执行的脚本中值为 _main_,在被导入的脚本中值为该脚本名。
我的一些个人理解:模块导入可以看成是把模块中的所有代码搬到当前这个脚本当中了,二者逻辑上是一个文件,物理上是两个文件。而我们在导入时需要的是模块中定义的那些类,属性,方法等的代码,不需要操作这些类、属性、方法的代码,所以在写py脚本时可以通过if __name__ == '__main__': ,把非操作代码放到main外面,把操作代码放到main里面。
__all__控制导入内容
在脚本里可以通过__all__ = [xxx, xxx, xxx] 指定哪些类/方法/变量可以被导入,这种设置只对from 模块名 import * 这种导入方式生效,对import 模块名 和指定导入from 模块名 import 方法名 这两种导入方式无效。
2. 包
包的理解
个人理解就是一组功能相关的模块放到一起,通过__init__.py 文件组织起来。这里的组织起来的意思是:让这些零散的模块,对外逻辑上为一个整体,当import 包名 时导入包下的所有子模块。
包的__init__.py文件
在导入包文件时(无论是整个包还是包里的某个子模块),包内的__init__.py 都会最先执行,同样在该文件里可以像模块中的那样通过__all__ = [xxx, xxx, xxx] 指定哪些子模块允许被导入,同样只针对from 包名 import * 这种导入方式有效。
导入包时使用from 包名 import * 方式导入要求__init__.py文件中必须有__all__指定哪些子模块可以被导入,否则包下的所有子模块都不允许被导入。这点和导入模块不同,导入模块时不__all__指定会默认导入所有的非私有方法/属性/类等。
根据这种最先执行的特点,可以根据需要在该文件里设置一些导入包时的预操作。
|