• 一、什么是包?

    包是模块的一种形式,包的本质就是一个含有.py的文件的文件夹。

    二、为什么要有包?

    模块的第一个版本只有10个功能,但是未来在扩展版本的时候,模块名和用法应该最好不要去修改,但是这只是对使用者友好,而由于版本扩展,文件越来越大,模块设计者对模块的管理、维护会越来越复杂,因此我们可以使用包来扩展模块的功能。

    三、如何使用包?

    3.1 模块和包

    导入模块发生的三件事:

    1. 创建一个包的名称空间
    2. 执行py文件,将执行过程中产生的名字存放于名称空间中。
    3. 在当前执行文件中拿到一个名字aaa,aaa是指向包的名称空间的

    导入包发生的三件事:

    1. 创建一个包的名称空间
    2. 由于包是一个文件夹,无法执行包,因此执行包下的.py文件,将执行过程中产生的名字存放于包名称空间中(即包名称空间中存放的名字都是来自于.py)
    3. 在当前执行文件中拿到一个名字aaa,aaa是指向包的名称空间的

    导入包就是在导入包下的.py,并且可以使用以下两种方式导入:

    1. import ...
    2. from ... import...

    3.2 扩展模块功能

    65包-目录结构.jpg?x-oss-process=style/watermark

    如下我们如果需要扩展aaa.py模块,需要建立一个aaa的目录文件,并且删除aaa.py文件,将aaa.py修改成m1.py和m2.py两个文件,让模块的功能使用方法不改变。

    # aaa.py
    
    def func1():
        pass
    
    def func2():
        pass
    
    def func3():
        pass
    
    def func4():
        pass
    
    def func5():
        pass
    
    def func6():
        pass
    
    # m1.py
    
    def func1():
        pass
    
    def func2():
        pass
    
    def func3():
        pass
    
    # m2.py
    
    def func4():
        pass
    
    def func5():
        pass
    
    def func6():
        pass
    
    # run.py
    
    import aaa
    
    aaa.func1()
    aaa.func2()
    aaa.func3()
    aaa.func4()
    aaa.func5()
    aaa.func6()
    

    3.3 修改__init__.py文件

    # aaa/.py
    
    func1 = 111
    func2 = 222
    func3 = 333
    func4 = 444
    func5 = 555
    func6 = 666
    

    由于在__init__.py中定义了func1,因此我们可以在run.py文件中导入func1,但是这个func1并不是我们想要的func1,因此需要修改__init__.py文件,又由于执行文件run.py的环境变量不为aaa,因此直接使用import导入m1会报错,因此使用from导入。

    # aaa/.py
    
    from aaa.m1 import func1
    from aaa.m2 import func2
    
    # run.py
    
    import aaa
    
    
    print(aaa.func1())
    print(aaa.func2())
    

    3.4 导入包内包

    aaa.bbb指向aaa内部的文件夹bbb包,如果我们需要导入bbb这个包。

    # bbb/.py
    
    from aaa import bbb
    
    # run.py
    
    import aaa
    
    
    print(aaa.bbb)
    

    3.5 导入包内包的模块

    如果bbb包内部有m3.py,我们需要从run.py导入m3模块。

    # bbb/.py
    
    from aaa.bbb import m3
    
    # run.py
    
    import aaa
    
    
    aaa.bbb.m3
    

    3.6 绝对导入和相对导入

    绝对导入:

    # aaa/.py
    from aaa.m1 import func1
    from aaa.m2 import func2
    

    相对导入:

    • .代表当前被导入文件所在的文件夹
    • ..代表当前被导入文件所在的文件夹的上一级
    • ...代表当前被导入文件所在的文件夹的上一级的上一级
    
    from .m1 import func1
    from .m2 import func2
    

    四、总结

    1. 包内所有的文件都是被导入使用的,而不是被直接运行的

    2. 包内部模块之间的导入可以使用绝对路径导入(以包的根目录为基准)与相对导入(以当前被导入的模块所在的目录为基准), 推荐使用相对导入

    3. 当文件是执行文件是,无法在该文件内使用相对导入的语法,只有在文件时,被当作模块导入时,该文件内才能使用相对导入的语法

    4. 凡是在导入时带.的,点的左边必须有一个包import aaa.bbb.m3.f3错误

    5. 包的文件以执行文件的环境变量为基准,包的init只能从包名开始导入

    6. from m1 import * , 运行init这个文件,创建一个init这个名称空间,然后把inti内的变量丢入 init 这个名称空间内

    7. 导包以当前执行文件的环境变量为基准,先从当前环境中导,在__init__中需要注意,尤其在调用的时候

      # __init__
      import  sys
      print(sys.path)
      from m2 import f1
      # 执行不会报错
      
      # 与创建包同一级别的.py文件
      import sys
      print(sys.path)
      from m1 import f1
      
      # 包如下错误
      Traceback (most recent call last):
        File "G:/python学习/python基础/01 包.py", line 5, in <module>
          from m1 import f1
        File "G:python学习python基础m1\__init__.py", line 5, in <module>
          from m2 import f1
      ModuleNotFoundError: No module named 'm2'
      

      报错原因就是因为在__init__中的from m2 import f1导入模块,是从当前执行文件目录中查找m2.py模块,但是当前文件中并没有m2.py这个模块,所以会报错。如果在在导包的同一级别中创建一个m2.py模块就不会报错,从而证明导包是从当前执行文件为基准

    注:导包以当前执行文件的环境变量为基准,先从当前环境中导,在__init__中需要注意,尤其在调用的时候

    在当下的阶段,必将由程序员来主导,甚至比以往更甚。
  • 相关阅读:
    剑指offer 二叉树中和为某一个值的路径
    剑指offer 二叉搜索树的后序遍历序列
    二叉树
    剑指offer 二叉树的层序遍历
    剑指offer 二叉树的镜像
    二叉树的子结构
    牛客网 斐波那契数列
    NMT 机器翻译
    剑指offer 从尾到头打印链表
    剑指offer 链表中倒数第k个节点
  • 原文地址:https://www.cnblogs.com/randysun/p/11368120.html
Copyright © 2020-2023  润新知