from pathlib import path_python小课堂15 - 史上最详细的包和模块import讲解篇

be4264cb7f11161039acd2834b857cab.png

python小课堂15 - 史上最详细的包和模块import讲解篇

前言

在大量的代码设计中,我们不可能将所有代码都写在一个.py文件,所以有了包、模块,而为了代码可以重复利用(复用性),就有了类、函数的概念。类和函数在下次介绍。

python中的包

python中的包,对应到计算机中,可以理解为文件夹,但是文件加下必须有一个名为__init__.py的文件,若没有此文件,python则会认为其只是一个普通的文件夹。

打开pycharm,创建一个包,如下:

f373a28f6c490df329c7600e6909a602.png

94e93c24314bcabbcf3384dcc8f14860.png

python中的模块

python中的模块就非常好理解了,实际上,之前所有的.py文件,我们都可以称之为一个模块。单独的一个py文件就是一个模块。

d2eb77b787b923973c39e023ec38b15c.png

3ade6a0ba6e22c05c80e501c8f02c58a.png

test1和test2不同区别就是test2是和package这个包是同级目录,而test1是属于package包的。

再来看下总的概念:

24b09bff7b05d670610f1a0130a1e896.png

包和模块的引入

1.<font color = gree>模块处于同级目录</font>(<font color = red>并且不在包下 </font>)

当我们想在一个模块中使用另一个模块中的变量时,如何操作呢?test2、test3处于同一级目录。

ef1a0b21fcbcb754c5b2427ea5ab9850.png

我想在test3中引入test2的变量,test2.py中有个变量a = 2。

47cbd72be08670a4c49b636d38f8239f.png

03cc504e5cfc6863eeebbd52e747362b.png

ecb1f2d447cf06acb6fa5412aa80e08f.png

如上所示,只需要在当前模块,用import语句,即可导入模块,具体使用的时候需要用模块的名字.变量。

import 后面必须是模块的名称! ------> import modul name

还有一种写法如下图pycharm中:

da7329e5da98cb1ae014e50965b53efc.png

如上所示,只需要在当前模块, from 模块名字 import 变量

2.<font color = gree>模块处于同级目录</font>(<font color =red>在同一包下</font>)

来看下,test1,test4都属于package包下的模块。

bf915b00ef6c8051e63c30d743d0e194.png

test1.py中有着字符串a = 'I am success!'

0d934b1fab548ad32cb4abb004be9834.png

在test4.py中引用test1.py中的a,如何引用呢?

可以看到如下:

390a78acc0baf0e0d8f056067319c8d2.png
关键语法:import 包名.模块名 as 别名

<font color =red>但是!!!!!!!如果我们脱离pycharm,找到本机相应的python目录,通过cmd来运行下,看下效果如何:

c2ce1d56aa86e67eeddf572661d3d9d1.png

4ac74bf5386c964c5bdc427240bd7451.png

可以清晰的看到上图,通过命令行模式执行就会报错!错误显示模块没有被找到:没有模块叫'package'。这是为什么呢?在pycharm中通过右键run as运行test4,可以看到控制台成功输出,而本地调用命令行的形式就报错了!

打开pycharm的setting,搜索 python console,右侧其中有一项,add content roots to pythonpath,默认pycharm是勾选上此项的。此项的意思是将内容的根路径加到python的环境变量路径下。

008374c9fe52e5ec2d3eb78d8027dc19.png

可以看到上图下面代码块里写着一堆代码,正是这段代码,我们才可以在pycharm中正确运行。

我们可以在test1.py里来看下sys.path,顺便打印看下结果。

a3b4ea0819b4ef6c8a00f8832f16789a.png

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会将模块的绝对路径输出到控制台上。

165ea731920be719e35079f3d1708b4a.png

在用命令行执行下看看:

4986743a324019912de8a0b62e0dab8d.png

python额外小知识:可以看到上图有一个__pycache__的文件夹,这个文件夹在pycharm的目录中,我们是看不到的,那么此文件夹的意义何在呢?点进去看下:

664e30800e2acbdd09d14a6c60748f5c.png
Python程序运行时不需要编译成二进制代码,而直接从源码运行程序,简单来说是,Python解释器将源码转换为字节码,然后再由解释器来执行这些字节码。而解释器的具体工作: 1、完成模块的加载和链接。 2、将源代码编译为PyCodeObject对象(即字节码),写入内存中,供CPU读取。 3、从内存中读取并执行,结束后将PyCodeObject写回硬盘当中,也就是复制到.pyc或.pyo文件中,以保存当前目录下所有脚本的字节码文件。 4、若再次执行该脚本,它先检查【本地是否有上述字节码文件】和【该字节码文件的修改时间是否在其源文件之后】,是就直接执行,否则重复上述步骤。 第一次执行代码的时候,Python解释器已经把编译的字节码放在__pycache__文件夹中,这样以后再次运行的话,如果被调用的模块未发生改变,那就直接跳过编译这一步,直接去__pycache__文件夹中去运行相关的 *.pyc 文件,大大缩短了项目运行前的准备时间。 CSDN来源: https:// blog.csdn.net/index2000 1/article/details/73501375

继续回归正题:

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:

1bef3c480c57ce6f96bb1ce70fdf8dc4.png

可以看到上图结果:

__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)

通过命令行执行来看下:

af83edaf35505d2e10043d0fe141f139.png

有了以上的所有操作步骤,我们可以完美的将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)

8560fa7424520b6e3931cddbc8cae1e1.png

成功!

写到这里涉及的知识点就已经这么多了。。。继续写。。。

3.<font color = gree>包处于同级目录</font>(<font color =red>包和包同级,包1下的模块引入包2下的模块变量</font >)

756290a5f2974609a154252d1c660479.png

可以看到,通过from test3 import c,pycharm中是正常输出的,控制台是报错的!原因实际和“2.模块处于同级目录(在同一包下)”的解释是一样的,只需要在引入自定义包之前,将我们项目的根路径加到python的系统变量中即可。

4.<font color = gree>模块处于不同级目录</font>(<font color =red>包和模块同级,模块引入包下模块的变量</font>)

f2783d450f782709d00d604371d73f09.png

b169cea552750c360d1b510b70ee73a8.png

9bd9e4123f4551d8b7512d42f29e5e8e.png

若属于3的情况,可以看到,不需要对python系统进行sys.append,可以正常使用import 或者 from 语句进行导入。

5.<font color = gree>模块处于不同级目录</font>(<font color =red>包和模块同级,包下模块引入与包模块同级的变量</font>)

test3.py 中有: c = 123455666 在packeage下的test1.py调用:

8c1be0799ae9cfbe62b668e0c8e1812a.png

可以看到,通过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)

255efa0ea2cf8428a58488c4db1c92be.png

模块之前的相对引入

什么叫相对引入呢?相对路径大家可能听说过,相对引入和相对路径是一个道理的,比如 .代表的是当前目录,..代表的是上级目录,此处的写法就是相对路径,相对于某个文件来说,即相对!

实际上,在2.模块处于同级目录(在同一包下)中,还有一种相对引入的写法,但是对于test4.py引入test1.py来说,不能直接运行test4.py,否则会报错。来看下:

e64996a08d0bd14ada7a14e1c37fad5a.png

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,就会报错!

fcfc63f59b2b5ac747436c6ea74ecabf.png
ModuleNotFoundError: No module named '__main__.test1'; '__main__' is not 
a package

如果此时,我通过test2.py间接行调用test4.py中的b

from package.test4 import b
print(b)

无论是pycharm还是命令行,都是有成功运行的:

6e5dd08d91405ecbc54e0250b025bb93.png

也就是说python对于相对引入来说,主动引入的函数不能作为主体去运行!

pycharm中可能会遇到的import报错

有人可能会遇到,当一个新项目导入到pycharm中,python代码的import有可能会报错,可以将项目设置为根路径,这样import错误即可消失,操作如下:

e4b776abdc5553d68582caed21a2cb85.png

包和模块自身的额外小知识点

  1. 关于包下的 init.py

init,中文意思是初始化的意思,而__init__.py实际上就是作为包名来配合的,当我们调用一个包时,第一步python就会去调用__init__.py模块,所以,经常我们可以将包下的__init__.py中放入一些需要初始化的操作。

举个例子:

15cbfcfb1cf6c961dc884d2248c78e66.png
__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 !'

可以看到下图运行结果,先输出了初始化模块中的字符串:

d863a3179b7949011fadac6d0cc37c16.png
  1. 关于模块中的限定变量写法

依然是test2.py引入test1.py的变量:

2156100ea38c1594c876910bfc2bda96.png
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单独引入三个变量,运行结果:

2560b445ac3715a9be9edb813dceb0f8.png

若将import 后面改成* ,则会限制变量。

cfc38c32c4ca4723a24670419889ce40.png

而此处所说,就是因为在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)

有想学python的同学,欢迎关注公号:migezatan.(咪哥杂谈)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值