今日主要内容

一、包

(一)什么是包

  • 只要是含有__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/test.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)
    • 下述所有内容,均以此图结构为基础

(三)__init__.py文件

  • 在我们导入包的时候,其实本质上导入的就是__init__.py文件,解释器会自动执行__init__.py文件中的内容

    # glance/__init__.py
    print("这里是glance下的__init__.py文件") # run.py
    import glance 运行结果:
    这里是glance下的__init__.py文件

(四)包的导入

  • 包的导入,须注意两点内容:

    • 导入时无论是使用import还是使用from xxx import xxx导入,点的前面必须是一个包
    • 使用from xxx import xxx导入时,import前可以出现点,import后不能出现点
  1. 常规导入

    • import导入
    # policy.py
    def policy_func():
    print("这是policy文件中的函数") # run.py
    import glance.api.policy glance.api.policy.policy_func() 运行结果:
    这是policy文件中的函数 # 成功导入
    • from import导入
    # policy.py
    def policy_func():
    print("这是policy文件中的函数") # run.py
    from glance.api.policy import policy_func policy_func() 运行结果:
    这是policy文件中的函数 # 成功导入
  2. 包内部之间的导入:

    • 包内部之间的相互引用,只能用绝对导入和相对导入,不能直接导入,比如policy.py文件需要导入versions.py文件中的函数,如果直接import导入会报找不到该模块的错误

      • 包内部互相引用的时候不能直接导入
      # versions.py
      def versions_func():
      print("这是versions文件中的函数") # policy.py
      import versions # 直接导入versions模块,在外面的run文件中运行,就无法找到versions模块 def policy_func():
      versions.versions_func()
      print("这是policy文件中的函数") # run.py
      from glance.api import policy policy.policy_func() 运行结果:
      ModuleNotFoundError: No module named 'versions'
      • 原因分析:如果直接在policy文件中运行程序,完全没有问题,因为此时policy文件的模块查找路径中包含versions模块,但是**包是用来导入的,如果在包外的run文件中导入policy模块,此时模块查找路径就变为了run文件所在的路径,所以当policy文件导入versions模块时,路径下并没有versions模块,所以找不到该模块
    • 绝对导入:

    # versions.py
    def versions_func():
    print("这是versions文件中的函数") # policy.py
    from glance.api import versions def policy_func():
    versions.versions_func()
    print("这是policy文件中的函数") # run.py
    from glance.api import policy policy.policy_func() 运行结果:
    这是versions文件中的函数
    这是policy文件中的函数
    • 相对导入:
    # versions.py
    def versions_func():
    print("这是versions文件中的函数") # policy.py
    from . import versions def policy_func():
    versions.versions_func()
    print("这是policy文件中的函数") # run.py
    from glance.api import policy policy.policy_func() 运行结果:
    这是versions文件中的函数
    这是policy文件中的函数
    • 只要使用了相对导入的模块,只能作为模块使用,不能作为脚本(终端中直接运行的文件成为脚本)使用,说白了就是不能直接运行
    # versions.py
    def versions_func():
    print("这是versions文件中的函数") # policy.py
    from . import versions def policy_func():
    versions.versions_func()
    print("这是policy文件中的函数") 运行结果:
    ImportError: cannot import name 'versions'
  3. 单独导入包,并使用包内的包或模块

    • 单独导入一个包,其本质上导入的只是__init__.py文件,包中其他的文件并没有被导入
    # policy.py
    def policy_func():
    print("这是policy文件中的函数") # run.py
    import glance # 只导入了__init__.py文件 glance.api.policy.policy_func() # 并不能使用 运行结果:
    AttributeError: module 'glance' has no attribute 'policy'
    • 此时就要利用到__init__.py文件了,__init__.py文件在包中起到了交接管理的作用,可以在__init__.py文件中导入每个子文件,也可以控制其是否能被导入,所以每一个包中必须要有__init__.py文件
    # policy.py
    def policy_func():
    print("这是policy文件中的函数") # glance/__init__.py
    from . import api # glance/api/__init__.py
    from . import policy # run.py
    import glance glance.api.policy.policy_func() 运行结果:
    这是policy文件中的函数
    • 直接调用模块中的方法
    # policy.py
    def policy_func():
    print("这是policy文件中的函数") # glance/__init__.py
    from .api import * # glance/api/__init__.py
    from .policy import * # run.py
    import glance glance.policy_func() 运行结果:
    这是policy文件中的函数
    • 还可以通过__all__来控制模块和模块中函数的导入

      • 控制模块导入
      # glance/__init__.py
      from .api import * # glance/api/__init__.py
      __all__ = ["versions"] # 只能导入versions,其余api下模块都不能被导入 # run.py
      import glance glance.policy 运行结果:
      AttributeError: module 'glance' has no attribute 'policy'
      • 控制方法导入
      # policy.py
      def policy_func():
      print("这是policy文件中的func") def policy_foo():
      print("这是policy文件中的foo") # glance/__init__.py
      from .api import * # glance/api/__init__.py
      from .policy import *
      __all__ = ["policy_foo"] # 控制只能导入foo函数,其余函数都不能被导入 # run.py
      import glance glance.policy_func() 运行结果:
      AttributeError: module 'glance' has no attribute 'policy_func'

Python基础(十八)的更多相关文章

  1. Bootstrap <基础十八>面包屑导航(Breadcrumbs)

    面包屑导航(Breadcrumbs)是一种基于网站层次信息的显示方式.以博客为例,面包屑导航可以显示发布日期.类别或标签.它们表示当前页面在导航层次结构内的位置. Bootstrap 中的面包屑导航( ...

  2. python 基础(十) 面向对象

    面向对象 一.概念 类(class): 用来描述具有相同属性和方法的对象的集合 对象是类的实例化 类变量:类变量在整个实例化的对象中是共用的.定义在类中 并且是函数体外的 实例变量:只能作用于 当前类 ...

  3. python基础(八种数据类型)

    Python的八种数据类型 八种数据类型分别是: number(数字).string(字符串).Boolean(布尔值).None(空值) list(列表).tuple(元组).dict(字典).se ...

  4. python基础-第八篇-8.1初识Socket

    socket基础 socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求. so ...

  5. Python基础(八)装饰器

    今天我们来介绍一下可以提升python代码逼格的东西——装饰器.在学习装饰器之前我们先来复习一下函数的几个小点,方便更好的理解装饰器的含义. 一.知识点复习 1, 在函数中f1和f1()有什么不同,f ...

  6. python基础知识八

    当你的程序中出现某些 异常的 状况的时候,异常就发生了.例如,当你想要读某个文件的时候,而那个文件不存在.或者在程序运行的时候,你不小心把它删除了.上述这些情况可以使用异常来处理. 如你的程序中有一些 ...

  7. python基础(八)-迭代器与生成器

    一.迭代器 li=[1,2,3] f=li.__iter__() print(f) print(f.__next__()) print(f.__next__()) print(f.__next__() ...

  8. Python爬虫(十八)_多线程糗事百科案例

    多线程糗事百科案例 案例要求参考上一个糗事百科单进程案例:http://www.cnblogs.com/miqi1992/p/8081929.html Queue(队列对象) Queue是python ...

  9. Python基础篇(八)

    key words:私有变量,类静态变量,生成器,导入Python模块,r查看模块可以使用的函数,查看帮助信息,启动外部程序,集合,堆,时间模块,random模块,shelve模块,文件读取等 > ...

  10. 小甲鱼Python第十八讲课后习题

    笔记: 1.函数与过程:过程(procedure)是简单的,特殊且没有返回值的:函数(Function)有返回值 Python严格来说只有函数没有过程 2.局部变量:在局部生效如在函数中定义的变量 3 ...

随机推荐

  1. ImportError: cannot import name '_obtain_input_shape' from 'keras.applications.imagenet_utils'

    报错 Using TensorFlow backend. Traceback (most recent call last): File "D:/PyCharm 5.0.3/WorkSpac ...

  2. ZKW线段树 非递归版本的线段树

    学习和参考 下面是支持区间修改和区间查询的zkw线段树模板,先记下来. #include <algorithm> #include <iterator> #include &l ...

  3. UVA1486 Transportation 费用流 拆边。

    #include <iostream> #include <cstdio> #include <cmath> #include <queue> #inc ...

  4. 牛客暑假多校 F RIKKA with Line Graph

    题意: 现在有一副完全图, 将他转化成线图. 线图就是 把原来的图上的边都变成点, 然后如果原来的任意2条边存在公共点, 他们就会有一条边, 边权为原来的2条边的和. 最后求出线图中的任意2点的最短路 ...

  5. uiautomator2 实现App九宫格解锁

    App九宫格解锁 之前在testerhome社区看见codeskyblue大佬写过一种方法,但是这种办法存在一个弊端,那就是多个点的坐标是写死的,也就是说要是换了部手机,九宫格解锁就行不通了,于是就想 ...

  6. 除了FastJson,你还有选择: Gson简易指南

    前言 这个周末被几个技术博主的同一篇公众号文章 fastjson又被发现漏洞,这次危害可导致服务瘫痪! 刷屏,离之前漏洞事件没多久,fastjson 又出现严重 Bug.目前项目中不少使用了 fast ...

  7. vuex-class用法

    vuex-class可以包装vuex的写法,使代码简化 Installation $ npm install --save vuex-class Example import Vue from 'vu ...

  8. SpringBoot自定义过滤器的两种方式及过滤器执行顺序

    第一种 @WebFilter + @ServletComponentScan 注解 1.首先自定义过滤器 如下自定义过滤器 ReqResFilter 必须实现  javax.servlet.Filte ...

  9. 大数据平台搭建 - cdh5.11.1 - hadoop集群安装

    一.前言 由于线下测试的需要,需要在公司线下(测试)环境搭建大数据集群. 那么CDH是什么? hadoop是一个开源项目,所以很多公司再这个基础上进行商业化,不收费的hadoop版本主要有三个,分别是 ...

  10. (附源码gitHub下载地址)spring boot -jta-atomikos分布式事务

    应用场景:双数据源,就是某些项目会涉及到两个数据源或者两个以上的数据源,这个多数据源的项目一般是数据同步,也就是把数据从另一个系统中,保存到另一个系统,两边的 数据库又不一样,比如一个Mysql.一个 ...