python小课堂15 - 史上最详细的包和模块import讲解篇
前言
在大量的代码设计中,我们不可能将所有代码都写在一个.py文件,所以有了包、模块,而为了代码可以重复利用(复用性),就有了类、函数的概念。类和函数在下次介绍。
python中的包
python中的包,对应到计算机中,可以理解为文件夹,但是文件加下必须有一个名为__init__.py的文件,若没有此文件,python则会认为其只是一个普通的文件夹。
打开pycharm,创建一个包,如下:
python中的模块
python中的模块就非常好理解了,实际上,之前所有的.py文件,我们都可以称之为一个模块。单独的一个py文件就是一个模块。
test1和test2不同区别就是test2是和package这个包是同级目录,而test1是属于package包的。
再来看下总的概念:
包和模块的引入
1.<font color = gree>模块处于同级目录</font>(<font color = red>并且不在包下 </font>)
当我们想在一个模块中使用另一个模块中的变量时,如何操作呢?test2、test3处于同一级目录。
我想在test3中引入test2的变量,test2.py中有个变量a = 2。
如上所示,只需要在当前模块,用import语句,即可导入模块,具体使用的时候需要用模块的名字.变量。
import 后面必须是模块的名称! ------> import modul name
还有一种写法如下图pycharm中:
如上所示,只需要在当前模块, from 模块名字 import 变量
2.<font color = gree>模块处于同级目录</font>(<font color =red>在同一包下</font>)
来看下,test1,test4都属于package包下的模块。
test1.py中有着字符串a = 'I am success!'
在test4.py中引用test1.py中的a,如何引用呢?
可以看到如下:
关键语法:import 包名.模块名 as 别名
<font color =red>但是!!!!!!!如果我们脱离pycharm,找到本机相应的python目录,通过cmd来运行下,看下效果如何:
可以清晰的看到上图,通过命令行模式执行就会报错!错误显示模块没有被找到:没有模块叫'package'。这是为什么呢?在pycharm中通过右键run as运行test4,可以看到控制台成功输出,而本地调用命令行的形式就报错了!
打开pycharm的setting,搜索 python console,右侧其中有一项,add content roots to pythonpath,默认pycharm是勾选上此项的。此项的意思是将内容的根路径加到python的环境变量路径下。
可以看到上图下面代码块里写着一堆代码,正是这段代码,我们才可以在pycharm中正确运行。
我们可以在test1.py里来看下sys.path,顺便打印看下结果。
pycharm控制台输出:
['F:pycharmpython14package', 'F:pycharmpython14',
'D:python3.6python36.zip',
'D:python3.6DLLs', 'D:python3.6lib', 'D:python3.6',
'C:UserssyAppDataRoamingPythonPython36site-packages',
'D:python3.6libsite-packages',
'D:python3.6libsite-packageswin32',
'D:python3.6libsite-packageswin32lib',
'D:python3.6libsite-packagesPythonwin']
实际通过命令行输出,应该没有'F:pycharmpython14' 这一项,因为这一项是pycharm中setting自动加上的!
实际控制台输出:
['F:pycharmpython14package',
'D:python3.6python36.zip',
'D:python3.6DLLs', 'D:python3.6lib', 'D:python3.6',
'C:UserssyAppDataRoamingPythonPython36site-packages',
'D:python3.6libsite-packages',
'D:python3.6libsite-packageswin32',
'D:python3.6libsite-packageswin32lib',
'D:python3.6libsite-packagesPythonwin']
sys.path是一个list。默然情况下python导入文件或者模块的话,他会先在sys.path里找模块的路径。如果没有的话,程序就会报错。可以看到,sys路径下有package的包名,而没有test4.py中引用test1.py模块。
而pycharm能够成功运行,正是因为它已经帮我们把项目的根路径添加到了python的环境变量中。所以我们仿照其类似写法也可以完成!
解决方案:
这里不得不说几个重要的python自带模块了,如下:
①
__file__ : python模块自身的名称
pycharm打印下__file__:
可以看到pycharm会将模块的绝对路径输出到控制台上。
在用命令行执行下看看:
python额外小知识:可以看到上图有一个__pycache__的文件夹,这个文件夹在pycharm的目录中,我们是看不到的,那么此文件夹的意义何在呢?点进去看下:
Python程序运行时不需要编译成二进制代码,而直接从源码运行程序,简单来说是,Python解释器将源码转换为字节码,然后再由解释器来执行这些字节码。而解释器的具体工作: 1、完成模块的加载和链接。 2、将源代码编译为PyCodeObject对象(即字节码),写入内存中,供CPU读取。 3、从内存中读取并执行,结束后将PyCodeObject写回硬盘当中,也就是复制到.pyc或.pyo文件中,以保存当前目录下所有脚本的字节码文件。 4、若再次执行该脚本,它先检查【本地是否有上述字节码文件】和【该字节码文件的修改时间是否在其源文件之后】,是就直接执行,否则重复上述步骤。 第一次执行代码的时候,Python解释器已经把编译的字节码放在__pycache__文件夹中,这样以后再次运行的话,如果被调用的模块未发生改变,那就直接跳过编译这一步,直接去__pycache__文件夹中去运行相关的 *.pyc 文件,大大缩短了项目运行前的准备时间。
继续回归正题:
②
import sys,os : sys ,os模块是python系统自带模块
os模块: operate system 操作系统的意思,一般可以通过调用此模块来对系统进行相关操作
sys 模块: system 系统的意思,通过此模块来实现对python自定义包和模块的导入
有了以上两个知识点,我们可以对test4.py进行如下操作:
import sys,os
print(__file__)
print(os.path.abspath(__file__))
print(os.path.dirname(os.path.abspath(__file__)))
print(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
因为pycharm会对__file__进行路径补充,所以我们用命令行来执行test4.py:
可以看到上图结果:
__file__ 模块名字
test4.py
os.path.abspath(__file__) 模块名字的绝对路径
F:pycharmpython14packagetest4.py
os.path.dirname(os.path.abspath(__file__)) 模块的包名绝对路径
F:pycharmpython14package
os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
F:pycharmpython14 项目本身的绝对路径
通过最后一步,我们可以将项目本身的路径直接拼入python的sys下
base_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(base_path)
验证究竟有没有加到我们的python环境变量中,最终代码为:
import sys, os
print(sys.path)
print(__file__)
print(os.path.abspath(__file__))
print(os.path.dirname(os.path.abspath(__file__)))
print(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
base_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(base_path)
print(sys.path)
通过命令行执行来看下:
有了以上的所有操作步骤,我们可以完美的将test1.py的a变量引入test4.py中了!来看下命令执行:
import sys, os
base_path = os.path.dirname(os.path.dirname(
os.path.abspath(__file__)))
sys.path.append(base_path)
---------------- sys拼接 一定要在自定义包引入之前定义 ----------------------------------
import package.test1 as test1 注意import的顺序。
print(test1.a)
成功!
写到这里涉及的知识点就已经这么多了。。。继续写。。。
3.<font color = gree>包处于同级目录</font>(<font color =red>包和包同级,包1下的模块引入包2下的模块变量</font >)
可以看到,通过from test3 import c,pycharm中是正常输出的,控制台是报错的!原因实际和“2.模块处于同级目录(在同一包下)”的解释是一样的,只需要在引入自定义包之前,将我们项目的根路径加到python的系统变量中即可。
4.<font color = gree>模块处于不同级目录</font>(<font color =red>包和模块同级,模块引入包下模块的变量</font>)
若属于3的情况,可以看到,不需要对python系统进行sys.append,可以正常使用import 或者 from 语句进行导入。
5.<font color = gree>模块处于不同级目录</font>(<font color =red>包和模块同级,包下模块引入与包模块同级的变量</font>)
test3.py 中有: c = 123455666
在packeage下的test1.py调用:
可以看到,通过from test3 import c,pycharm中是正常输出的,控制台是报错的!原因实际和“2.模块处于同级目录(在同一包下)”的解释是一样的,只需要在引入自定义包之前,将我们项目的根路径加到python的系统变量中即可。
import sys, os
base_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(base_path)
from test3 import c
print(c)
模块之前的相对引入
什么叫相对引入呢?相对路径大家可能听说过,相对引入和相对路径是一个道理的,比如 .代表的是当前目录,..代表的是上级目录,此处的写法就是相对路径,相对于某个文件来说,即相对!
实际上,在2.模块处于同级目录(在同一包下)中,还有一种相对引入的写法,但是对于test4.py引入test1.py来说,不能直接运行test4.py,否则会报错。来看下:
test1.py中有个a字符串:
test1.py
a = 'I am success !'
test4.py中,我用from .test1 import a来引入test1.py的变量a,注意,包下同级目录,我使用的是.test1 !!!!
test4.py
from .test1 import a
b = a + 'I am test4.py import .test1'
如果此时我直接将test4.py运行,并且打印b,就会报错!
ModuleNotFoundError: No module named '__main__.test1'; '__main__' is not
a package
如果此时,我通过test2.py间接行调用test4.py中的b
from package.test4 import b
print(b)
无论是pycharm还是命令行,都是有成功运行的:
也就是说python对于相对引入来说,主动引入的函数不能作为主体去运行!
pycharm中可能会遇到的import报错
有人可能会遇到,当一个新项目导入到pycharm中,python代码的import有可能会报错,可以将项目设置为根路径,这样import错误即可消失,操作如下:
包和模块自身的额外小知识点
- 关于包下的 init.py
init,中文意思是初始化的意思,而__init__.py实际上就是作为包名来配合的,当我们调用一个包时,第一步python就会去调用__init__.py模块,所以,经常我们可以将包下的__init__.py中放入一些需要初始化的操作。
举个例子:
__init__.py:
init_a = 'I am __init__.py'
print(init_a)
在package包下定义了初始化的字符串。 而test2.py调用package下的test1.py中的a变量时:
test2.py:
from package.test1 import a
print(a)
test1.py:
a = 'I am success !'
可以看到下图运行结果,先输出了初始化模块中的字符串:
- 关于模块中的限定变量写法
依然是test2.py引入test1.py的变量:
test2.py:
from package.test1 import a,b,c
print(a)
print(b)
print(c)
test1.py:
__all__ = ['a','b']
a = 'I am success !'
b = 'I am fail !'
c = 'I am fuc*** you!!! !'
在test2中引入test1通过import单独引入三个变量,运行结果:
若将import 后面改成* ,则会限制变量。
而此处所说,就是因为在test1.py中有着__all__ = [] ,这样的写法可以限定住import * 的限制,test4.py import *时,则会被限制住骂人的语句!
import 模块的万金油方法
上面说了这么多种情况,如果你实在是记不住,那么请记住一点,万金油的import方式,就是在你所有模块的入口模块处,以下面代码为例,将你项目本身的绝对路径拼入到python 的系统path下,<font color =red>这样自定义的包一定不会出错!!!
import sys, os
base_path = os.path.dirname(os.path.dirname(
os.path.abspath(__file__)))
sys.path.append(base_path)