python之模块与包

一、什么是模块?

  简单概括一下模块就是一组功能的集合体

  查看一下内置的模块:

    由此可见模块可以是一个py文件 ,模块的名字就是os 

  

  

二、模块的使用

  1.模块的导入

    模块的导入有两种方法,第一种用来导入默认路径(os.system())下的模块使用import,第二种用来导入其他路径下模块需要from import关联使用

1 import os   
2 # 本质上就是调用os.py文件,在模块与包的概念中称之为os模块
3 from 课程练习 import my_module 
4 # 什么时候会用到from import 关联来使用呢? 
5 # import 导入的是一个路径,在这个路径下找到py文件,而当我们需要找的py文件不在这个路径下的时候,
6 # 我们就需要通过from 找到 这个py文件所在的路径, 然后在从这个路径下找到py文件,这里from 与improt关联来使用

  2.为模块起别名

    就是为了方便使用,将长或者不容易理解的模块名字重新命名。

1 import os as cz
2 cz.remove("")
3 # 通过上述代码可以观察到,模块可以重新赋值一个全新的名字,并且其中内部的函数都可以被使用,
4 
5 import asjdlasjdoaisud as of
6 # 我们什么时候会用到模块的重名名? 当然如果我们需要用到一个模块,而模块的名字又非常长,这时候我们可以为模块起别名

  3.模块循环导入问题

    模块循环/嵌套导入抛出异常的根本原因是由于在python中模块被导入一次之后,就不会重新导入,只会在第一次导入时执行模块内代码

  在我们的项目中应该尽量避免出现循环/嵌套导入,如果出现多个模块都需要共享的数据,可以将共享的数据集中存放到某一个地方

  在程序出现了循环/嵌套导入后的异常分析、解决方法

  

#示范文件内容如下
#m1.py
print('正在导入m1')
from m2 import y

x='m1'

#m2.py
print('正在导入m2')
from m1 import x

y='m2'

#run.py
import m1

#测试一
执行run.py会抛出异常
正在导入m1
正在导入m2
Traceback (most recent call last):
  File "/Users/linhaifeng/PycharmProjects/pro01/1 aaaa练习目录/aa.py", line 1, in <module>
    import m1
  File "/Users/linhaifeng/PycharmProjects/pro01/1 aaaa练习目录/m1.py", line 2, in <module>
    from m2 import y
  File "/Users/linhaifeng/PycharmProjects/pro01/1 aaaa练习目录/m2.py", line 2, in <module>
    from m1 import x
ImportError: cannot import name 'x'

#测试一结果分析
先执行run.py--->执行import m1,开始导入m1并运行其内部代码--->打印内容"正在导入m1"
--->执行from m2 import y 开始导入m2并运行其内部代码--->打印内容“正在导入m2”--->执行from m1 import x,由于m1已经被导入过了,所以不会重新导入,所以直接去m1中拿x,然而x此时并没有存在于m1中,所以报错


#测试二:执行文件不等于导入文件,比如执行m1.py不等于导入了m1
直接执行m1.py抛出异常
正在导入m1
正在导入m2
正在导入m1
Traceback (most recent call last):
  File "/Users/linhaifeng/PycharmProjects/pro01/1 aaaa练习目录/m1.py", line 2, in <module>
    from m2 import y
  File "/Users/linhaifeng/PycharmProjects/pro01/1 aaaa练习目录/m2.py", line 2, in <module>
    from m1 import x
  File "/Users/linhaifeng/PycharmProjects/pro01/1 aaaa练习目录/m1.py", line 2, in <module>
    from m2 import y
ImportError: cannot import name 'y'


#测试二分析
执行m1.py,打印“正在导入m1”,执行from m2 import y ,导入m2进而执行m2.py内部代码--->打印"正在导入m2",执行from m1 import x,此时m1是第一次被导入,执行m1.py并不等于导入了m1,于是开始导入m1并执行其内部代码--->打印"正在导入m1",执行from m1 import y,由于m1已经被导入过了,所以无需继续导入而直接问m2要y,然而y此时并没有存在于m2中所以报错



# 解决方法:
方法一:导入语句放到最后
#m1.py
print('正在导入m1')

x='m1'

from m2 import y

#m2.py
print('正在导入m2')
y='m2'

from m1 import x

方法二:导入语句放到函数中
#m1.py
print('正在导入m1')

def f1():
    from m2 import y
    print(x,y)

x = 'm1'

# f1()

#m2.py
print('正在导入m2')

def f2():
    from m1 import x
    print(x,y)

y = 'm2'

#run.py
import m1

m1.f1()

示例文件

  

三、什么是包?

  包只是模块的一种形式而已,包的本质就是一种模块

  

  

  包就是一个包含有__init__.py文件的文件夹,所以其实我们创建包的目的就是为了用文件夹将文件/模块组织起来

 四、包的使用

  1.为什么会用到包?  

    包的本质就是一个文件夹,那么文件夹唯一的功能就是将文件组织起来随着功能越写越多,我们无法将所以功能都放到一个文件中,于是我们使用模块去组织功能,而随
  着模块越来越多,我们就需要用文件夹将模块文件组织起来,以此来提高程序的结构性和可维护性
 2.包的导入
   
 1 绝对导入与相对导入
 2 
 3 # 绝对导入: 以执行文件的sys.path为起始点开始导入,称之为绝对导入
 4 #        优点: 执行文件与被导入的模块中都可以使用
 5 #        缺点: 所有导入都是以sys.path为起始点,导入麻烦
 6 
 7 # 相对导入: 参照当前所在文件的文件夹为起始开始查找,称之为相对导入
 8 #        符号: .代表当前所在文件的文件加,..代表上一级文件夹,...代表上一级的上一级文件夹
 9 #        优点: 导入更加简单
10 #        缺点: 只能在导入包中的模块时才能使用
11       #注意:
12         1. 相对导入只能用于包内部模块之间的相互导入,导入者与被导入者都必须存在于一个包内
13         2. attempted relative import beyond top-level package # 试图在顶级包之外使用相对导入是错误的,言外之意,必须在顶级包内使用相对导入,每增加一个.代表跳到上一级文件夹,而上一级不应该超出顶级包

 

 
posted @ 2018-08-23 19:57  sado  阅读(190)  评论(0编辑  收藏  举报