• 本节主要内容:
    1. from xxx import xxx知识点补充
    2. 包

    一、知识点补充。

      我们现在知道可以使⽤import和from xxx import xxx来导入⼀个模块中的内容. 那有⼀种
    特殊的写法: from xxx import * 我们说此时是把模块中的所有内容都导入. 注意, 如果模块中
    没有写出__all__ 则默认所有内容都导入. 如果写了__all__ 此时导入的内容就是在__all__列表
    中列出来的所有名字.

    # haha.py
    __all__ = ["money", "chi"]
    money = 100
    def chi():
         print("我是吃")
    def he():
         print("我是呵呵")
    
    # test.py from haha import * chi() print(money) # he() # 报错

    二、包

      包是一种通过 ".模块"来组织python模块名称空间的方式。那什么样的东西是包呢?我们创建的每个文件夹都可以被称之为包。但是我们要注意,在python2中规定。包内必须处在__init__.py文件。创建包的目的不是为了运行,而是被导入使用。包知识一种形式而已。包的本质就是一种模块

      为何要使用包?包的本质就是一个文件夹,那么文件夹唯一的功能就是将文件组织起来,随着功能越写越多,我们无法将所有功能都放在一个文件中,于是我们使用模块去组织功能,随着模块越来越多,我们需要用文件夹将模块文件组织起来,以此来提高程序的结构性和可维护性

      首先,我们先创建一些包。用来作为接下来的学习。包很好创建。只要有一个文件夹,有__init__.py就可以。

    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/__init__.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)
    

      接下来我们在bao中使用包中的内容。并且,我们导入包的时候可以使用import或者

      from xxx import xxx这种形式

      首先看import

    import glance.db.models
    glance.db.models.register_models('mysql')
    

      没问题, 很简单, 我们还可以使⽤from xxx import xxx 来导入包内的模块

    from glance.api.policy import get
    get()

      也很简单, 但是, 要注意. from xxx import xxx这种形式, import后⾯不可以出现"点" 也
    就是说from a.b import c是ok的. 但是 from a import b.c 是错误的。

      好了, 到⽬前为⽌, 简单的包已经可以使⽤了. 那包⾥的__init__.py是什么⻤? 其实. 不论
    我们使⽤哪种⽅式导入⼀个包, 只要是第⼀次导入包或者是包的任何其他部分, 都会先执⾏
    __init__.py⽂件. 这个⽂件可以是空的. 但也可以存放⼀些初始化的代码. (随意在glance中的
    __init__.py都可以进⾏测试)

      那我们之前⽤的from xxx import *还可以⽤么? 可以. 我们要在__init__.py⽂件中给出
    __all__来确定* 导入的内容.

    print("我是glance的__init__.py⽂件. ")
    x = 10
    def hehe():
         print("我是呵呵")
    def haha():
         print("我是哈哈")
    __all__ = ['x', "hehe"]
    

      test.py

    from glance import *
    print(x) # OK
    hehe() # OK
    haha() # 报错. __all__⾥没有这个⻤东⻄
    

      接下来,我们来看一下绝对导入和相对导入,我们的最顶级包glance是写给别人用的。然后再glance包内部也会有彼此之间相互导入的需求,这时候就有绝对导入和相对导入两种方式了。

      1.绝对导入:以glance作为起始

      2.相对导入:用.或者..作为起始

    例如, 我们在glance/api/version.py中使⽤glance/cmd/manage.py

    # 在glance/api/version.py
    #绝对导⼊
    from glance.cmd import manage
    manage.main()
    #相对导⼊
    # 这种情形不可以在versions中启动程序.
    # attempted relative import beyond top-level package
    from ..cmd import manage
    manage.main()

      测试的时候要注意. python包路径跟运⾏脚本所在的⽬录有关系. 说⽩了. 就是你运⾏的
    py⽂件所在的⽬录. 在python中不允许你运⾏的程序导包的时候超过当前包的范围(相对导
    入). 如果使⽤绝对导入. 没有这个问题. 换个说法. 如果你在包内使⽤了相对导入. 那在使⽤该
    包内信息的时候. 只能在包外⾯导入.

      接下来,我们来看一个大坑。比如:我们向在policy中使用verson中的内容。

    # 在policy.py
    import versions

      如果我们程序的入⼝是policy.py 那此时程序是没有任何问题的. 但是如果我们在glance
    外⾯import了glance中的policy就会报错 原因是如果在外⾯访问policy的时候. sys.path中的
    路径就是外⾯. 所以根本就不能直接找到versions模块. 所以⼀定会报错:

    ModuleNotFoundError: No module named 'versions'
    

      在导包出错的时候. ⼀定要先看sys.path 看⼀下是否真的能获取到包的信息. 

      

      最后, 我们看⼀下如何单独导入⼀个包. 

    # 在test.py中
    import glance

      此时导入的glance什么都做不了. 因为在glance中的__init__.py中并没有关于⼦包的加
    载. 此时我们需要在__init__.py中分别取引入⼦包中的内容.
      1. 使⽤绝对路径
      2. 使⽤相对路径

    包的注意事项:

      1. 关于包相关的导入语句也分为import和from xxx import xxx两种, 但⽆论使⽤哪种,
    ⽆论在什么位置, 在导入时都必须遵循⼀个原则: 凡是在导入时d带点的. 点左边都必须是⼀
    个包. 否则报错. 可以带⼀连串的点. 比如a.b.c

      2. import导入⽂件时. 产⽣名称空间中的名字来源于⽂件, import 包, 产⽣的名称空间
    中的名字同样来源于⽂件, 即包下的__init__,py, 导入包本质就是在导入该⽂件。

      3. 包A和包B下有同名模块也不会冲突, 如A.a和B.a来⾃两个名称空间

  • 相关阅读:
    Python全栈开发记录_第七篇(模块_time_datetime_random_os_sys_hashlib_logging_configparser_re)
    Python全栈开发记录_第六篇(生成器和迭代器)
    Python全栈开发记录_第五篇(装饰器)
    Python全栈开发记录_第四篇(集合、函数等知识点)
    Python全栈开发记录_第三篇(linux(ubuntu)的操作)
    Python全栈开发记录_第二篇(文件操作及三级菜单栏增删改查)
    Python全栈开发记录_第一篇(循环练习及杂碎的知识点)
    NET控件Designer架构设计
    如何把Excel中的单元格等对象保存成图片
    “某某云词典” – 纠结的初体验
  • 原文地址:https://www.cnblogs.com/guchenxu/p/10197609.html
Copyright © 2020-2023  润新知