包
python中的lib中有很多模块和包,比如collections就是一个文件夹,包含多个模块,这就是一个包.
将能解决一大类问题的模块放在同一个文件夹中,这个文件夹就被称为包。
在pycharm中创建一个package会生成一个文件夹,且在这个文件夹中含有一个__int__文件,那么这个这个文件夹就是一个包。注意这个__int__文件在python3中无所谓,可有可无,但是在python2.7中则必须含有。
1. 无论是import形式还是from...import形式,凡是在导入语句中(而不是在使用时)遇到带点的,都要第一时间提高警觉:这是关于包才有的导入语法
2. 包是目录级的(文件夹级),文件夹是用来组成py文件(包的本质就是一个包含__init__.py文件的目录)
3. import导入文件时,产生名称空间中的名字来源于文件,import 包,产生的名称空间的名字同样来源于文件,即包下的__init__.py,导入包本质就是在导入该文件
强调:
1. 在python3中,即使包下没有__init__.py文件,import 包仍然不会报错,而在python2中,包下一定要有该文件,否则import 包报错
2. 创建包的目的不是为了运行,而是被导入使用,记住,包只是模块的一种形式而已,包即模块
创建一个包,并以此为例进行演示
创建包:
import os os.makedirs('glance/api') os.makedirs('glance/cmd') os.makedirs('glance/db') l = [] l.append(open('glance/__init__.py','w')) l.append(open('glance/api/__init__.py','w')) l.append(open('glance/api/policy.py','w')) l.append(open('glance/api/versions.py','w')) l.append(open('glance/cmd/__init__.py','w')) l.append(open('glance/cmd/manage.py','w')) l.append(open('glance/db/models.py','w')) map(lambda f:f.close() ,l)
在相应的文件中输入内容
#文件内容 #policy.py def get(): print('from policy.py') #versions.py def create_resource(conf): print('from version.py: ',conf) #manage.py def main(): print('from manage.py') #models.py def register_models(engine): print('from models.py: ',engine)
注意事项
1.关于包相关的导入语句也分为import和from ... import ...两种,但是无论哪种,无论在什么位置,在导入时都必须遵循一个原则:凡是在导入时带点的,点的左边都必须是一个包,否则非法。可以带有一连串的点,如item.subitem.subsubitem,但都必须遵循这个原则。
2.对于导入后,在使用时就没有这种限制了,点的左边可以是包,模块,函数,类(它们都可以用点的方式调用自己的属性)。
3.对比import item 和from item import name的应用场景:
如果我们想直接使用name那必须使用后者。
# import glance.api.policy as policy # policy.get() from glance.api import policy policy.get()
等价于
from glance.api.policy import get
get()
注意只能写作from glance.api import policy,而不能写作from glance import .api policy,就是import后面不能带点。否则会有语法错误,如:from a import b.c是错误语法
from glance.db import models models.register_models('mysql') #等同于下列代码 from glance.db.models import register_models register_models('mysql')
那么为何不能直接一步导入policy?
因为根据上节课所学可知,import导入的模块必须在sys.path中,但是如下结果可知sys路径中并没有policy,所以需要一步步像树状结构一样直至到最后,找到policy
import sys print(sys.path) D:\anoconda\python.exe F:/python/python学习/人工智能/第一阶段day2/练习.py ['F:\\python\\python学习\\人工智能\\第一阶段day2', 'F:\\python\\python学习\\人工智能', 'D:\\anoconda\\python37.zip', 'D:\\anoconda\\DLLs', 'D:\\anoconda\\lib', 'D:\\anoconda', 'D:\\anoconda\\lib\\site-packages', 'D:\\anoconda\\lib\\site-packages\\win32', 'D:\\anoconda\\lib\\site-packages\\win32\\lib', 'D:\\anoconda\\lib\\site-packages\\Pythonwin', 'D:\\系统工具\\pycharm\\PyCharm 2018.3.2\\helpers\\pycharm_matplotlib_backend'] Process finished with exit code 0
注意import 模块名,本质上就是执行了这个模块里面的.py文件。
import glance glance.api.policy.get()#结果报错
那是因为在包里面,当执行import glance的时候,实际上执行的是glnce文件里面唯一的一个.py文件,__init__.py。
如何修改代码来避免报错
首先在glance下面的__init__.py文件中输入:
from glance import api from glance import cmd from glance import db
然后在api下面的__init__.py文件中输入:
from glance.api import policy from glance.api import versions
这样再进行运行的时候就不会报错。
同理在cmd包下的__init__.py文件中输入:加上首先在glance下面的__init__.py文件中的代码。
from glance.cmd import manage
就可以实现
import glance glance.cmd.manage.main()
代码的正常运行。
绝对导入和相对导入(这里没怎么看明白)
绝对路径:不能移动,但是直观;不管在包内部还是在外部,导入就可以使用
相对路径:可以随意移动包,只要能够找到包的位置,就可以使用包里的模块;包里的模块如果想使用其他模块的内容只能使用相对路径,使用了相对路径就不能在包内直接使用了.
关于包的作业在day26,自己没看懂这里,先掠过