python—day15 包的认识、执行顺序、执行流程、循环导入、包的导入、绝对、相对导入
一、包的认识
包通过文件夹来管理一系列功能相近的模块
包:一系列模块的集合体
重点:包中一定有一个专门用来管理包中所有模块的文件
包名:存放一系列模块的文件夹名字
包名(包对象)存放的是管理模块的那个文件的地址,指向其全局名称空间
二、模块的加载顺序
# 模块的加载顺序:内存 => 内置 => sys.path(一系列自定义模块)
import sys
sys.path # 环境变量:存放文件路径的列表
# 重点:默认列表第一个元素就是当前被执行文件所在的目录
# 可以自定义往sys.path添加路径
sys.path.append(r'想导入的模块的绝对路径') # 添加到环境变量最后,最后被查找
sys.path.insert(0, r'想导入的模块的绝对路径') # 添加到指定索引,索引就决定了自定义模块的查找顺序
import time
print(time)# 输出结果<module 'time' (built-in)> # 第一次导入:内存》内置》自定义,最终是在自定义中找到,完成导入,并在内存中缓存模块的内存地址
import m1
print(m1.num) # 输出100
time.sleep(5) # 输出'D:\\fullstack_s41\\day16\\2、模块的加载顺序' # 再次导入,从内存中可以找到,即当前文件为删除状态,内存中的地址仍然被引用
import m1 as mm1
print(mm1.num) import sys
# 环境变量:就是存放文件路径的列表
# 重点:默认列表第一个元素就是当前被执行文件所在的目录
print(sys.path) # 可以自定义往sys.path添加路径
sys.path.append(r'D:\fullstack_s41\day16\2、模块的加载顺序')
sys.path.insert(0,r'D:\fullstack_s41\day16\2、模块的加载顺序')
print(sys.path) # 可以自定义往sys.path添加路径
sys.path.append(r'D:\fullstack_s41\day16\1、包的认识') # sys.path.append(r'想导入的模块的绝对路径')添加到环境变量后,最后被查找
sys.path.insert(0,r'想导入的模块的绝对路径') # 添加到指定索引,索引就决定了自定义模块的查找顺序
三、模块导入的执行流程
导入模块的指令:
-- 相对于 函数名() 调用函数体,函数调用会进入函数体,从上至下逐句解释执行函数体代码
-- 导入模块,会进入模块文件,从上至下逐句解释执行模块文件代码
-- 如果在模块中又遇到导入其他模块,会接着进入导入的模块,从上至下逐句解释执行文件中代码,依次类推
# 导入模块
import m3
print(m3.a)
print('end') # 导入的模块要一步步进行执行在返回之前的原模块在进行一步步执行(如果在添加一个模块则从第一个导入的模块运行完再运行新加入的模块最后返回第一个原模块第一步进行运行)
# 这些操作步骤和执行顺序也造就了模块的循环导入调用函数体
# 操作流程:相当于函数名(),函数调用会进入函数体,从上至下逐句解释执行函数体代码
# 导入模块,会进入模块文件,从上至下逐句解释执行模块文件代码
# 如果在模块中又遇见其他模块,接着进入导入的模块,从上至下逐句解释执行文件中代码,以此类推
四、循环导入
模块之间出现了环状导入,如:m1.py 中导入了m2,m2.py 中又导入了m1.py
循环导入的问题:
-- 导入模块是要使用模块中的变量
-- 正常逻辑都是在文件最上方先完成对模块的导入,再在下方定义自身模块变量,以及使用导入的模块中的变量
-- 由于导入模块的特殊机制,第一次导入模块会编译执行导入的模块,也就是会进入模块逐句执行模块内容,再次导入只是使用内存中的名字
-- 就会出现下面的情况,m2在使用m1中的变量x,但变量x却并未产生,这就出现了循环导入问题
m1.py文件
import m2
x = 10
print(m2.y)
m2.py文件
import m1
y = 10
print(m2.x)
解决循环导入的问题:延后导入
1、将循环导入对应包要使用的变量提前定义,再导入响应的包
2、将导包的路径放到函数体中,保证存放导包逻辑的函数调用在要使用的变量定义之后
重点:
问题:from导包极容易出现循环导入问题
解决:建议from导入方式改用import导入方式
# 循环导入导致的问题: # 两个模块直接互相导入,且相互使用其名称空间中的名字,但是有些名字没有产生就使用,就会出现循环导入问题 # 解决import m4循环导入问题:延后导入---但产生对方要使用的名字,再去完成带入对方 # 1、from导入马上就会使用名字,极容易出现错误,建议循环导入情况下使用import导入
# 2、先提前产生名字,再导入模块
# 3、将导入逻辑放在函数中,将导入的逻辑延后到函数的调用,只有调用在产生名字之后即可 import m4 调用m4模块
import mm4 调用mm4模块
五、包的导入
# import本质:通过查找环境变量(sys.path)中的绝对路径来完成导入
# 导包:
# 1.保证包所在文件夹在环境变量中
# 2.导入的文件夹名就是包名
import pk
pk文件夹
-- __init__.py
# 右键执行该文件,sys.path第一个元素是当前文件夹的路径 #是主文件右键中copy path键是拷贝绝对路径
import sys
print(sys.path) # 第一个元素D:\Python36\python3.exe D:/fullstack_s41/day16/5、包/5、包.py就是这个地址 # 重点:后期包基本上不和执行文件同一目录,只有保证包所在的文件夹路径在环境变量中即可
# 将5中的包pk放到4的文件夹下,就会报错
sys.path.append(r'D:\fullstack_s41\day16\4、循环导入') # 报错 del sys.path[0]
print(sys.path) # 将当前所在文件路径移除,统一会出现异常 # import 本质:通过查找环境变量中的绝对路径来完成导入
# 导包
# 1、保证包所在的文件夹在环境变量中
# 2、导入的文件夹就是包名
# eg:import pk # pk:就是导入的包名
print(pk.a)
print(pk.b) from pk import a,b
print(a)
print(b) # 外置取值
# 包名是一个文件夹,名称空间是:建包是产生的_init_.py文件产生的
六、导包完成的三件事
导包完成的三项事:
1.编译执行包中的__init__.py文件,会在包中__pycache__创建对应的pyc文件
2.产生__init__.py文件的全局名称空间,用来存放__init__出现的名字
3.产生包名指向__init__.py文件的全局名称空间 | 指定变量名指向包中指定名字
# 总结:包名为文件夹名,名称空间是__init__.py产生的
import pk6 print(pk6.pm1)
print(pk6.pm1.num) import pkk6
print(pkk6.num) import pkk6.pkkm1
pkk6.pkkm1.fn()
pkk6.pk.fn() #import pkk6.pkkm1 as pk
pk.fn()
pkk6.pkkm1.fn() 如果使用import来导入包并使用包中模块中的名字
直接在要使用的文件中用import一层层找到你想要的名字
import ppk6.ppkm1 as pk
print(ppk6.ppkm1.num) # 起完别名,原名不可以再使用
print(pk.num) import ppk6.ppkm1.num as num
import part6.ppk6.ppkm1
导包的点语法:必须保证所有.左侧都是文件夹
七、使用包中模块中的名字:采用import导入
注意点:
1.在包__init__.py中不建议使用import导入
2、在包__init__.py中不建议使用as起别名
总结:不建议__init__.py中采用import管理名字 ==> 空着不写
在使用文件中
直接在要使用的文件中用import一层层找到你想要的名字
import 包名.文件名 as 别名
# 起完别名,原名不可以再使用
原名:包名.文件名 => 包名.文件名.变量名
别名:别名 => 别名.变量名
import pk7 print(pk7.pkm7)
print(pk7.m) import pk7.pkm7
pk7.pkm7.func() 错误:所有.前都必须为文件夹
import pk7.pkm7.func as func 外界通过import导入方式最多只能导到模板
外界通过from导入方式和import差不多,最多可以直接导到具体的名字
from pk7.pkm7 import func
func() 错误:所有.前都必须为文件夹
from pk7.pkm7.func import func 在所有包的init文件没有管理下,直接通过外界导入访问
import pk7.sub
print(pk7.sub.sub_a)
import pk7.sub.subm as sub
print(sub.subm_a) from pk7.sub import sub_a
print(sub_a) from pk7.sub.subm import subm_a
print(subm_a)
八、包中使用import导入:绝对导入
# 在包的__init__文件中
import 模块名 # 问题:所属包不在环境变量,报错
import 包名.模块名 # 问题:包所属文件夹不在文件变量,报错
import 包名.模块名 as 别名 # 在外界:包名.模块名 | 包名.别名 都可以访问
import 包名.模块名.名字 # 问题:导包语句.语法左侧必须全部是包(文件夹)
# 外界
import 包名
包名.名字 # 访问的是__init__中的名字
包名.模块 # 访问的模块这个地址
包名.模块.名字 # 访问的模块中的名字
import 包名.模块
包名.模块 # 访问的模块这个地址
包名.模块.名字 # 访问的模块中的名字
from 包名 import 模块
模块 # 访问的模块这个地址
模块.名字 # 访问的模块中的名字
from 包名.模块 import 名字
名字 # 访问的模块中的名字
# .代表当前文件夹
from .pkm8 import num
from .pkm8 import fn # from .sub import x
# from .sub.subm import y
from . import sub from .sub import subm
九、包中使用from导入:相对导入
# 没有子包
''' 1)
pk包
-- __init__.py
-- 名字 a = 10
-- pkm.py
-- 名字 b = 20 在外界
import pk
pk.a 访问a
pk.b 访问b
init管理文件
a不需要操作
from .pkm import b
# 有子包
1)第一种
pk包
-- __init__.py
sub子包
-- __init__.py
名字 x = 10
-- subm.py
名字 y = 10 在外界
import pk
pk.x 访问x
pk.y 访问y
在pk的init管理文件
from .sub import x
from .sub.subm import y
2)第二种
pk包
-- __init__.py
sub子包
-- __init__.py
名字 x = 10
-- subm.py
名字 y = 10 在外界
import pk
pk.sub.x 访问x
pk.sub.y 访问y
在pk的init管理文件:要产生sub名字
from . import sub => pk.sub
在sub的init管理文件:要产生x,y名字
x不需要操作 => pk.sub.x
from .subm import y => pk.sub.y
# 注:有相对导入.语法的文件都不能自执行
# .代表当前 | ..代表上一级目录 | ...代表上上一级目录
from ..pkm2 import name as pkm2_name
from ..sub1.sub1m import name as sub1m_name def fn2():
print(pkm2_name)
print(sub1m_name)
python—day15 包的认识、执行顺序、执行流程、循环导入、包的导入、绝对、相对导入的更多相关文章
- Golang 包了解以及程序的执行
Golang 包了解以及程序的执行 引言 Go 语言是使用包来组织源代码的,包(package)是多个 Go 源码的集合,是一种高级的代码复用方案.Go 语言中为我们提供了很多内置包,如 fmt.o ...
- Django多个中间件的执行顺序
Django中的中间件是一个轻量级.底层的插件系统,可以介入Django的请求和响应处理过程,修改Django的输入或输出.中间件的设计为开发者提供了一种无侵入式的开发方式,增强了Django框架的健 ...
- 错误的理解引起的bug async await 执行顺序
今天有幸好碰到一个bug,让我知道了之前我对await async 的理解有点偏差. 错误的理解 之前我一直以为 await 后面的表达式,如果是直接返回一个具体的值就不会等待,而是继续执行asyn ...
- created与mounted执行顺序(关于微任务与宏任务)
1.ps:只要你只使用created或者mounted中的一个不就好了吗[dog].这样只要在第一个异步任务代码跳出前,嵌套第二个任务函数就好了 最后面的两个链接一个是微任务与宏任务的通俗例子,一个是 ...
- python中try except执行顺序
python中try except finally的执行顺序 先执行try中语句 如果try中抛出异常, 执行异常中语句. 如果try 或 except 中没有return语句,执行完try 或者 e ...
- python导入模块时的执行顺序
当python导入模块,执行import语句时,到底进行了什么操作?按照python的文档,她执行了如下的操作: 第一步,创建一个新的module对象(它可能包含多个module) 第二步,把这个mo ...
- Python装饰器执行顺序详解
探究多个装饰器执行顺序 装饰器是Python用于封装函数或代码的工具,网上可以搜到很多文章可以学习,我在这里要讨论的是多个装饰器执行顺序的一个迷思. 疑问 大部分涉及多个装饰器装饰的函数调用顺序时都会 ...
- 利用“Java同包同名类执行顺序”取消Java 网站应用程序Licence验证
如果是在tomcat里运行,lib目录下一大堆的JAR包,不同的JAR包里可能会有相同的包名类名,JRE按照JAR名字的字母顺序加载JAR文件,同名类如果已加载,则后面的同名类会忽略. 公司购买的一款 ...
- python顺序执行多个py文件
python顺序执行多个py文件 假如我要执行code目录下的python程序,假设该目录下有1.py,2.py,3.py,4.py四个文件,但是我想执行1.py,2.py,4.py,则可在该目录下创 ...
随机推荐
- Linux ☞ Good good study,day day up
一. 修改桌面程序图标 linux的桌面图标都是在/usr/share/applications 目录下的那些 *.desktop文件,修改桌面程序图标就是修改.desktop图标配置文件中Icon的 ...
- Ubuntu 备份系统为ISO镜像 & 解决ISO限制4GB大小 & Clone当前系统到其他电脑
看标题,标题涵盖了3个部分,Ubuntu 备份系统为ISO镜像, 解决ISO限制4GB大小 , Clone当前系统到其他电脑 我们就从三个部分说起. Ubuntu 备份系统为ISO镜像 在Win ...
- 从BAT这种公司平薪跳槽头条,是否值得?
有一个朋友之前就职于阿里,之前交流关于跳槽的问题,具体是这样的:阿里工作3年,拿到了头条的offer.但是非常纠结要不要接的问题.于是几个朋友聚在了一起讨论了这个问题 而且最近好多读者也在参加面试,接 ...
- c++继承学习
继承分类: 虚表继承 class D{ public : d(){ } ~d(){ } private: }; 单重继承 class D{ public : d(){ } ~d(){ } privat ...
- VSTO 函数InStrRev
返回某一字符串从另一字符串的右侧开始算起第一次出现的位置. 参数 StringCheck 类型:System.String 必选. 搜索的 String 表达式. StringMatch 类型:Sys ...
- 【web安全】-- springboot实现两次MD5加密
一.为什么要做两次MD5 客户端MD5:HTTP在网络上是使用明文传输,用户输入的明文密码直接在网络上传输太危险.所以,在客户端先进行一次MD5(明文+固定盐). 服务端:服务端接受到后,也不是直接写 ...
- CentOS7配置mailx使用外部smtp服务器发送邮件
转自huskiesir的博客: 发送邮件的两种方式: 1.连接现成的smtp服务器去发送(此方法比较简单,直接利用现有的smtp服务器比如qq.新浪.网易等邮箱,只需要直接配置mail.rc文件即可实 ...
- Metasploit中aggregator插件无法使用
Metasploit中aggregator插件无法使用 aggregator是Metasploit自带的一个插件,用来管理会话Session.该插件使用metasploit-aggreator库. ...
- webServices 使用GET请求接口方法
webServices 若要使用GET请求接口方法在Web.config 下添加这段 <webServices> <protocols> <add ...
- tf.contrib.slim add_arg_scope
上一篇文章中我们介绍了arg_scope函数,它在每一层嵌套中update当前字典中参数形成新的字典,并入栈.那么这些参数是怎么作用到代码块中的函数的呢?比如说如下情况: with slim.arg_ ...