前言
又是我,一年一度的写博客(我度日如年),尽我所能地记录下我对模块的理解,给我自己,也是给他人的。
模块
什么是模块呢?
模块 -> 程序文件
文件就像我们祖国母亲,她包含了python的所有东西,我们可以随时调用她。
命名空间
命名空间因为对象的不同,也有所区别,可以分为如下几种:
- 内置命名空间(Built-in Namespaces):Python 运行起来,它们就存在了。内置函数的命名空间都属于内置命名空间,所以,我们可以在任何程序中直接运行它们,比如
id()
,不需要做什么操作,拿过来就直接使用了。 - 全局命名空间(Module:Global Namespaces):每个模块创建它自己所拥有的全局命名空间,不同模块的全局命名空间彼此独立,不同模块中相同名称的命名空间,也会因为模块的不同而不相互干扰。
(就是说中国的哈利和美国的哈利不是同一个人哦,互相没有交集互不认识,没有关系) - 本地命名空间(Function & Class:Local Namespaces):模块中有函数或者类,每个函数或者类所定义的命名空间就是本地命名空间。如果函数返回了结果或者抛出异常,则本地命名空间也结束了。
如图所示,3个空间是处于包含关系的,老大是内置命名空间,当程序开始运行,就有内置命名空间了;老二是全局命名空间;老三是本地命名空间。虽然老三最小,但是老大老二都喜欢护小,当查询命名空间的时候,都是从老三开始。
import hello
hello.hi() # Hi everyone, I love lsgogroup!
hi() # NameError: name 'hi' is not defined
导入模块
在导入模块前,我们必须先创建一个模块,不然哪里有模块给我们导入呢?对吧。
# TemperatureConversion.py
def c2f(cel):
fah = cel * 1.8 + 32
return fah
def f2c(fah):
cel = (fah - 32) / 1.8
return cel
导入模块分别有三种方法:
第一种方法:import 模块名
import TemperatureConversion
print('32摄氏度 = %.2f华氏度' % TemperatureConversion.c2f(32))
print('99华氏度 = %.2f摄氏度' % TemperatureConversion.f2c(99))
# 32摄氏度 = 89.60华氏度
# 99华氏度 = 37.22摄氏度
第二种方法:from 模块名 import 函数名(这种方法并不推荐,我们还是乖乖地用第一种好。)
from TemperatureConversion import c2f, f2c
print('32摄氏度 = %.2f华氏度' % c2f(32))
print('99华氏度 = %.2f摄氏度' % f2c(99))
# 32摄氏度 = 89.60华氏度
# 99华氏度 = 37.22摄氏度
除了想上面这样直接导入模块中的函数,也有一种模模糊糊,含糊不清地导入函数名:
from TemperatureConversion import *
print('32摄氏度 = %.2f华氏度' % c2f(32))
print('99华氏度 = %.2f摄氏度' % f2c(99))
# 32摄氏度 = 89.60华氏度
# 99华氏度 = 37.22摄氏度
第三种方法:import 模块名 as 新名字
这种就是将模块名转换成简写方式,再调用时不需打太多字
import TemperatureConversion as tc
print('32摄氏度 = %.2f华氏度' % tc.c2f(32))
print('99华氏度 = %.2f摄氏度' % tc.f2c(99))
# 32摄氏度 = 89.60华氏度
# 99华氏度 = 37.22摄氏度
if __name__ == '__main__'
假设我们有一个 const.py 文件,内容如下:
PI = 3.14
def main():
print("PI:", PI)
main() # PI: 3.14
现在,我们写一个用于计算圆面积的 area.py 文件,area.py 文件需要用到 const.py 文件中的 PI
变量。从 const.py 中,我们把 PI
变量导入 area.py:
from const import PI
def calc_round_area(radius):
return PI * (radius ** 2)
def main():
print("round area: ", calc_round_area(2))
main()
'''
PI: 3.14
round area: 12.56
'''
我们看到 const.py 中的 main 函数也被运行了,实际上我们不希望它被运行,因为 const.py 提供的 main 函数只是为了测试常量定义。这时if __name__ == '__main__'
派上了用场,我们把 const.py 改一下,添加if __name__ == "__main__"
:
PI = 3.14
def main():
print("PI:", PI)
if __name__ == "__main__":
main()
name:是内置变量,可用于表示当前模块的名字。
import const
print(__name__)
# __main__
print(const.__name__) #这是被 导入 了,所以__name__不会时__main__,而是模块名称,这时if语句不成立就不会运行被导入模块中的main函数。
# const
由此我们可知:如果一个 .py 文件(模块)被直接运行时,其__name__
值为__main__
,即模块名为__main__
。
所以,if __name__ == '__main__'
的意思是:当 .py 文件被直接运行时,if __name__ == '__main__'
之下的代码块将被运行;当 .py 文件以模块形式被导入时,if __name__ == '__main__'
之下的代码块不被运行。
以我的理解来说,就是情侣之间喊宝宝(main),分手之后,相视如仇人,喊得就是你的名字(const)了。
搜索路径
当解释器遇到 import 语句,如果模块在当前的搜索路径就会被导入。
搜索路径被存储在 sys
模块中的 path
变量中。
通过这种方法来查找路径:
import sys
print(sys.path)
包(package)
包是一种管理 Python 模块命名空间的形式,采用"点模块名称"。
创建包分为三个步骤:
- 创建一个文件夹,用于存放相关的模块,文件夹的名字即包的名字。
- 在文件夹中创建一个
__init__
的模块文件,内容可以为空。 - 将相关的模块放入文件夹中。
这里给出了一种可能的包结构(在分层的文件系统中):(每个子包 的开头都有一个初始化函数,必须的)
sound/ 顶层包
__init__.py 初始化 sound 包
formats/ 文件格式转换子包
__init__.py
wavread.py
wavwrite.py
aiffread.py
aiffwrite.py
auread.py
auwrite.py
...
effects/ 声音效果子包
__init__.py
echo.py
surround.py
reverse.py
...
filters/ filters 子包
__init__.py
equalizer.py
vocoder.py
karaoke.py
...
在导入一个包的时候,Python 会根据 sys.path
中的目录来寻找这个包中包含的子目录。
目录只有包含一个叫做 __init__.py
的文件才会被认作是一个包,最简单的情况,放一个空的 __init__.py
就可以了。
import sound.effects.echo
这将会导入子模块 sound.effects.echo
。 他必须使用全名去访问:
sound.effects.echo.echofilter(input, output, delay=0.7, atten=4)
还有一种导入子模块的方法是:
from sound.effects import echo
这同样会导入子模块: echo,并且他不需要那些冗长的前缀,所以他可以这样使用:
echo.echofilter(input, output, delay=0.7, atten=4)
还有一种变化就是直接导入一个函数或者变量:
from sound.effects.echo import echofilter
同样的,这种方法会导入子模块: echo,并且可以直接使用他的 echofilter() 函数:
echofilter(input, output, delay=0.7, atten=4)
注意当使用 from package import item
这种形式的时候,对应的 item
既可以是包里面的子模块(子包),或者包里面定义的其他名称,比如函数,类或者变量。
设想一下,如果我们使用 from sound.effects import *
会发生什么?
Python 会进入文件系统,找到这个包里面所有的子模块,一个一个的把它们都导入进来。
导入语句遵循如下规则:如果包定义文件 __init__.py
存在一个叫做 __all__
的列表变量,那么在使用 from package import *
的时候就把这个列表中的所有名字作为包内容导入。
这里有一个例子,在 sounds/effects/__init__.py
中包含如下代码:
__all__ = ["echo", "surround", "reverse"]
这表示当你使用 from sound.effects import *
这种用法时,你只会导入包里面这三个子模块。
如果 __all__
真的没有定义,那么使用from sound.effects import *
这种语法的时候,就不会导入包 sound.effects
里的任何子模块。他只是把包 sound.effects
和它里面定义的所有内容导入进来(可能运行__init__.py
里定义的初始化代码)。
这会把 __init__.py
里面定义的所有名字导入进来。并且他不会破坏掉我们在这句话之前导入的所有明确指定的模块。
import sound.effects.echo
import sound.effects.surround
from sound.effects import *
这个例子中,在执行 from...import
前,包 sound.effects
中的 echo
和 surround
模块都被导入到当前的命名空间中了。
通常我们并不主张使用 *
这种方法来导入模块,因为这种方法经常会导致代码的可读性降低。
习题
1、怎么查出通过 from xx import xx导入的可以直接调用的方法?
答:使用all方法,查出模块下不带_的所有的方法,可以直接调用。
2、了解Collection模块,编写程序以查询给定列表中最常见的元素。
题目说明:
输入:language = [‘PHP’, ‘PHP’, ‘Python’, ‘PHP’, ‘Python’, ‘JS’, ‘Python’, ‘Python’,‘PHP’, ‘Python’]
输出:Python
"""
Input file
language = ['PHP', 'PHP', 'Python', 'PHP', 'Python', 'JS', 'Python', 'Python','PHP', 'Python']
Output file
Python
"""
def most_element(language):
""" Return a list of lines after inserting a word in a specific line. """
# your code here
解答代码:
from collections import Counter
language = ['PHP', 'PHP', 'Python', 'PHP', 'Python', 'JS', 'Python', 'Python', 'zHP', 'Python']
def most_element(language):
c = Counter(language)
for k, v in c.items():
if max(c.values()) == v:
print(k)
most_element(language)
总结
今天我去吃饭看见饭堂大叔在旁边闷闷不乐地,我过去问了问他怎么了,他指了指旁边那块菜地,说:“!!!我菜死了!!!”