python之模块与包

一模块

二包

一模块

常见的场景:一个模块就是一个包含了python定义和声明的文件,文件名就是模块名字加上.py的后缀。

   但其实import加载的模块分为四个通用类别: 

  1 使用python编写的代码(.py文件)

  2 已被编译为共享库或DLL的C或C++扩展

  3 包好一组模块的包

  4 使用C编写并链接到python解释器的内置模块

为什么要使用模块

 如果你退出python解释器然后重新进入,那么你之前定义的函数或者变量都将丢失,因此我们通常将程序写到文件中以便永久保存下来,需要时就通过python test.py方式去执行,此时test.py被称为脚本script。

    随着程序的发展,功能越来越多,为了方便管理,我们通常将程序分成一个个的文件,这样做程序的结构更清晰,方便管理。这时我们不仅仅可以把这些文件当做脚本去执行,还可以把他们当做模块来导入到其他的模块中,实现了功能的重复利用。

模块的导入

直接导入文件的名字,不需要带着后缀。

在模块中函数的调用:模块名.函数名()

模块导入做3件事:

(1)首先,开辟了一个新的命名空间名字是模块名

(2)执行里面的代码

(3)将模块名里面的名字和命名空间绑定在一起

注:模块在一个程序中只会被导入一次,不会重复导入 (为了节约资源)

如何实现模块在程序中只会被导入一次的呢?

导入一个模块后,会将模块存储在内存中

当再次导入时,就到内存中去查看是否导入了这个模块,如果导入就不继续导了。是通过sys里面的module。

导入的模块有自己的命名空间。

可以给导入的模块起一个别名这个时候,就产生了一个命名空间,这个命名空间只和别名相关。

1 import my_moudle as sm
2 print(sm.money)
#mysql.py
def sqlparse():
    print('from mysql sqlparse')
#oracle.py
def sqlparse():
    print('from oracle sqlparse')

#test.py
db_type=input('>>: ')
if db_type == 'mysql':
    import mysql as db
elif db_type == 'oracle':
    import oracle as db

db.sqlparse()

from .....import....

这种形式,导入啥可以用啥,不导入一律不能用

这个被import导入的名字属于全局变量

参数问题和返回值问题都和函数一样

from...import *

*与all 一起用的  首先会把模块中所有不是——开头的内容导入进来

可以通过——all——来控制导入的内容。但是只和*有关

大部分情况下我们的python程序不应该使用这种导入方式,因为*你不知道你导入什么名字,很有可能会覆盖掉你之前已经定义的名字。而且可读性极其的差,在交互式环境中导入时没有问题。

把模块当做脚本使用运行

python解释器在启动时会自动加载一些模块,可以使用sys.modules查看

模块的查找顺序是:内存中已经加载的模块->内置模块->sys.path路径中包含的模块

编译文件:为了提高加载模块的速度,强调强调强调:提高的是加载速度而绝非运行速度。python解释器会在__pycache__目录中下缓存每个模块编译后的版本,格式为:module.version.pyc。通常会包含python的版本号。

二包

包是一种通过使用‘.模块名’来组织python模块名称空间的方式。

1. 无论是import形式还是from...import形式,凡是在导入语句中(而不是在使用时)遇到带点的,都要第一时间提高警觉:这是关于包才有的导入语法

2. 包是目录级的(文件夹级),文件夹是用来组成py文件(包的本质就是一个包含__init__.py文件的目录)

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

强调:

  1. 在python3中,即使包下没有__init__.py文件,import 包仍然不会报错,而在python2中,包下一定要有该文件,否则import 包报错

  2. 创建包的目的不是为了运行,而是被导入使用,记住,包只是模块的一种形式而已,包即模块

注意事项:

1.关于包相关的导入语句也分为import和from ... import ...两种,但是无论哪种,无论在什么位置,在导入时都必须遵循一个原则:凡是在导入时带点的,点的左边都必须是一个包,否则非法。可以带有一连串的点,如item.subitem.subsubitem,但都必须遵循这个原则。

2.对于导入后,在使用时就没有这种限制了,点的左边可以是包,模块,函数,类(它们都可以用点的方式调用自己的属性)。

3.对比import item 和from item import name的应用场景:
如果我们想直接使用name那必须使用后者。

需要注意的是from后import导入的模块,必须是明确的一个不能带点,否则会有语法错误,如:from a import b.c是错误语法

绝对导入相对导入(待续中。。。。。)

原文地址:https://www.cnblogs.com/1a2a/p/7325700.html