1
0

Python:模块和包

This commit is contained in:
周中平 2022-11-28 17:53:30 +08:00
parent 3bb3967bb4
commit 54ce20d066
No known key found for this signature in database
GPG Key ID: B1DF9DD42D8E00DC
3 changed files with 286 additions and 126 deletions

View File

@ -327,3 +327,53 @@ print(f)
for val in f: for val in f:
print(val) print(val)
``` ```
## 全局、局部变量
**定义在函数内部的变量拥有一个局部作用域,定义在函数外的拥有全局作用域。**
局部变量只能在其被声明的函数内部访问,而全局变量可以在整个程序范围内访问。调用函数时,所有在函数内声明的变量名称都将被加入到作用域中。如下实例:
```python
total = 0 # 全局变量
def sum( arg1, arg2 ):
# 返回 2 个参数的和
total = arg1 + arg2 # total在这里是局部变量.
print ("函数内是局部变量 : ", total)
return total
#调用 sum 函数
sum( 10, 20 )
print ("函数外是全局变量 : ", total)
```
## global、nonlocal 关键字
内部作用域修改外部作用域的变量时,需要使用 global 关键字声明。反之要修改嵌套作用域enclosing 作用域,外层非全局作用域)中的变量则需要 nonlocal 关键字。
```python
num = 1
def fun1():
global num # 需要使用 global 关键字声明
print(num)
num = 123
print(num)
fun1()
print(num)
```
```python
def outer():
num = 10
def inner():
nonlocal num # nonlocal关键字声明
num = 100
print(num)
inner()
print(num)
outer()
```

View File

@ -0,0 +1,236 @@
---
title: 模块和包
description: Python 模块和包
keywords:
- Python
- 模块和包
tags:
- Python
sidebar_position: 7
author: 7Wate
date: 2022-11-28
---
在编程语言中,代码块、函数、类、模块,一直到包,逐级封装,层层调用。**在 Python 中,一个`.py`文件就是一个模块,模块是比类更高一级的封装。**在其他语言,被导入的模块也通常称为库。
## 模块
**模块可以分为自定义模块、内置模块和第三方模块**。使用模块有什么好处?
- 首先,提高了代码的可维护性。
- 其次,编写代码不必从零开始。当一个模块编写完毕,就可以被其他的模块引用。不要重复造轮子,我们简简单单地使用已经有的模块就好了。
- 使用模块还可以避免类名、函数名和变量名发生冲突。相同名字的类、函数和变量完全可以分别存在不同的模块中。但是也要注意尽量不要与内置函数名(类名)冲突。
### 自定义模块
自己开发完成复用的模块。
### 标准模块
Python 拥有一个强大的标准库。Python语言的核心只包含数值、字符串、列表、字典、文件等常见类型和函数而由 Python 标准库提供了系统管理、网络通信、文本处理、数据库接口、图形系统、XML 处理等额外的功能。
Python 标准库的主要功能有:
- 文本处理包含文本格式化、正则表达式、文本差异计算与合并、Unicode 支援,二进制数据处理等功能。
- 文件系统功能,包含文件和目录操作、建立临时文件、文件压缩与归档、操作配置文件等功能。
- 操作系统功能包含线程与进程支持、IO 复用、日期与时间处理、调用系统函数、日志logging等功能。
- 网络通信包含网络套接字SSL 加密通信、异步网络通信等功能。支持 HTTPFTPSMTPPOPIMAPNNTPXMLRPC 等多种网络协议,并提供了编写网络服务器的框架。
- W3C 格式支持,包含 HTMLSGMLXML 的处理。
- 其它功能包括国际化支持、数学运算、HASH、Tkinter 等。
### 第三方模块
Python 拥有大量的第三方模块,这也是其核心优点之一。基本上,所有的第三方模块都会在[PyPI - the Python Package Index](https://pypi.python.org/)上注册,只要找到对应的模块名字,即可用 pip 安装。
## 包
Python 为了避免模块名冲突又引入了按目录来组织模块的方法称为包Package**包是模块的集合,比模块又高一级的封装。**包是一个分层次的文件目录结构,它定义了一个由模块及子包,和子包下的子包等组成的 Python 的应用环境。**包名通常为全部小写,避免使用下划线。**
简单来说,包就是文件夹,但该文件夹下必须存在 `__init__.py` 文件, 该文件的内容可以为空。`__init__.py` 用于标识当前文件夹是一个包。
```markdown
test.py
package_runoob
|-- __init__.py
|-- runoob1.py
|-- runoob2.py
```
```python
# package_runoob/runoob1.py
def runoob1():
print "I'm in runoob1"
# package_runoob/runoob2.py
def runoob2():
print "I'm in runoob2"
# package_runoob/__init__.py
if __name__ == '__main__':
print '作为主程序运行'
else:
print 'package_runoob 初始化'
# test.py
from package_runoob.runoob1 import runoob1
from package_runoob.runoob2 import runoob2
runoob1()
runoob2()
# 输出
# package_runoob 初始化
# I'm in runoob1
# I'm in runoob2
```
## import
Python 中,模块(包、类、函数)的导入方式有以下四种:
```python
import xx.xx
from xx.xx import xx
from xx.xx import xx as rename
from xx.xx import *
```
### import xx.xx
将对象(这里的对象指的是包、模块、类或者函数,下同)中的所有内容导入。如果该对象是个模块,那么调用对象内的类、函数或变量时,需要以`module.xxx`的方式。
```python
# Module_a.py
def func():
print("this is module A!")
# Main.py
import module_a
module_a.func() # 调用方法
```
### from xx.xx import xx.xx
从某个对象内导入某个指定的部分到当前命名空间中,不会将整个对象导入。这**种方式可以节省写长串导入路径的代码,但要小心名字冲突**。
```python
# Main.py
from module_a import func
module_a.func() # 错误的调用方式
func() # 这时需要直接调用 func
```
### from xx.xx import xx as rename
为了避免命名冲突,在导入的时候,可以给导入的对象重命名。
```python
# Main.py
from module_a import func as f
def func(): # main 模块内部已经有了 func 函数
print("this is main module!")
func()
f()
```
### from xx.xx import *
将对象内的所有内容全部导入。非常容易发生命名冲突,请慎用!
```python
# Main.py
from module_a import *
def func():
print("this is main module!")
func() # 从 module 导入的 func 被 main 的 func 覆盖了
```
### 模块搜索路径
**不管在程序中执行了多少次import一个模块只会被导入一次。**导入一个模块Python 解析器对模块位置的搜索顺序是:
1. Python 项目当前目录
2. Python 搜索在 shell 变量 PYTHONPATH 下的每个目录。
3. Python 默认搜索路径。UNIX下默认路径一般为 /usr/local/lib/python/。
模块搜索路径存储在 system 模块的 sys.path 变量中。变量里包含当前目录PYTHONPATH 由安装过程决定的默认目录。
```python
import sys
print(sys.path)
# ['/workspace/PythonStudy', '/usr/local/lib/python310.zip', '/usr/local/lib/python3.10', '/usr/local/lib/python3.10/lib-dynload', '/home/user/.local/lib/python3.10/site-packages']
```
## 命名空间
命名空间Namespace是从名称到对象的映射大部分的命名空间都是通过 Python 字典来实现的。
命名空间提供了在项目中避免名字冲突的一种方法。各个命名空间是独立的,没有任何关系的,所以一个命名空间中不能有重名,但不同的命名空间是可以重名而没有任何影响。
一般有三种命名空间:
- **内置名称built-in names**Python 语言内置的名称,比如函数名 abs、char 和异常名称 BaseException、Exception 等等。
- **全局名称global names**:模块中定义的名称,记录了模块的变量,包括函数、类、其它导入的模块、模块级的变量和常量。
- **局部名称local names**:函数中定义的名称,记录了函数的变量,包括函数的参数和局部定义的变量。(类中定义的也是)
![img](https://static.7wate.com/img/2022/11/20/7ee3813629181.png)
假设我们要使用变量 runoob则 **Python 的查找顺序为:局部的命名空间 -> 全局命名空间 -> 内置命名空间。**
如果找不到变量 runoob它将放弃查找并引发一个 NameError 异常:`NameError: name 'runoob' is not defined。`
### 生命周期
命名空间的生命周期取决于对象的作用域,如果对象执行完成,则该命名空间的生命周期就结束。因此,我们**无法从外部命名空间访问内部命名空间的对象。**
![img](https://static.7wate.com/img/2022/11/20/8beaf2d3e4567.png)
```python
# var1 是全局名称
var1 = 5
def some_func():
# var2 是局部名称
var2 = 6
def some_inner_func():
# var3 是内嵌的局部名称
var3 = 7
```
## 作用域
作用域就是一个 Python 程序可以直接访问命名空间的正文区域。
在一个 python 程序中,直接访问一个变量,会从内到外依次访问所有的作用域直到找到,否则会报未定义的错误。
Python 中,程序的变量并不是在哪个位置都可以访问的,访问权限决定于这个变量是在哪里赋值的。
变量的作用域决定了在哪一部分程序可以访问哪个特定的变量名称。Python 的作用域一共有4种分别是
- **LLocal**:最内层,包含局部变量,比如一个函数/方法内部。
- **EEnclosing**包含了非局部non-local也非全局non-global的变量。比如两个嵌套函数一个函数或类 A 里面又包含了一个函数 B ,那么对于 B 中的名称来说 A 中的作用域就为 nonlocal。
- **GGlobal**:当前脚本的最外层,比如当前模块的全局变量。
- **BBuilt-in** 包含了内建的变量 / 关键字等,最后被搜索。
![img](https://static.7wate.com/img/2022/11/20/1e3af056f1ac0.png)
规则顺序: **L > E > G > B**
```python
g_count = 0 # 全局作用域
def outer():
o_count = 1 # 闭包函数外的函数中
def inner():
i_count = 2 # 局部作用域
```

View File

@ -1,126 +0,0 @@
---
title: 命名空间和作用域
description: Python 命名空间和作用域
keywords:
- Python
- 命名空间和作用域
tags:
- Python
sidebar_position: 1
author: 7Wate
date: 2022-11-20
---
## 命名空间
命名空间Namespace是从名称到对象的映射大部分的命名空间都是通过 Python 字典来实现的。
命名空间提供了在项目中避免名字冲突的一种方法。各个命名空间是独立的,没有任何关系的,所以一个命名空间中不能有重名,但不同的命名空间是可以重名而没有任何影响。
一般有三种命名空间:
- **内置名称built-in names**Python 语言内置的名称,比如函数名 abs、char 和异常名称 BaseException、Exception 等等。
- **全局名称global names**:模块中定义的名称,记录了模块的变量,包括函数、类、其它导入的模块、模块级的变量和常量。
- **局部名称local names**:函数中定义的名称,记录了函数的变量,包括函数的参数和局部定义的变量。(类中定义的也是)
![img](https://static.7wate.com/img/2022/11/20/7ee3813629181.png)
假设我们要使用变量 runoob则 **Python 的查找顺序为:局部的命名空间 -> 全局命名空间 -> 内置命名空间。**
如果找不到变量 runoob它将放弃查找并引发一个 NameError 异常:`NameError: name 'runoob' is not defined。`
### 生命周期
命名空间的生命周期取决于对象的作用域,如果对象执行完成,则该命名空间的生命周期就结束。因此,我们**无法从外部命名空间访问内部命名空间的对象。**
![img](https://static.7wate.com/img/2022/11/20/8beaf2d3e4567.png)
```python
# var1 是全局名称
var1 = 5
def some_func():
# var2 是局部名称
var2 = 6
def some_inner_func():
# var3 是内嵌的局部名称
var3 = 7
```
## 作用域
作用域就是一个 Python 程序可以直接访问命名空间的正文区域。
在一个 python 程序中,直接访问一个变量,会从内到外依次访问所有的作用域直到找到,否则会报未定义的错误。
Python 中,程序的变量并不是在哪个位置都可以访问的,访问权限决定于这个变量是在哪里赋值的。
变量的作用域决定了在哪一部分程序可以访问哪个特定的变量名称。Python 的作用域一共有4种分别是
- **LLocal**:最内层,包含局部变量,比如一个函数/方法内部。
- **EEnclosing**包含了非局部non-local也非全局non-global的变量。比如两个嵌套函数一个函数或类 A 里面又包含了一个函数 B ,那么对于 B 中的名称来说 A 中的作用域就为 nonlocal。
- **GGlobal**:当前脚本的最外层,比如当前模块的全局变量。
- **BBuilt-in** 包含了内建的变量 / 关键字等,最后被搜索。
![img](https://static.7wate.com/img/2022/11/20/1e3af056f1ac0.png)
规则顺序: **L > E > G > B**
```python
g_count = 0 # 全局作用域
def outer():
o_count = 1 # 闭包函数外的函数中
def inner():
i_count = 2 # 局部作用域
```
## 全局、局部变量
**定义在函数内部的变量拥有一个局部作用域,定义在函数外的拥有全局作用域。**
局部变量只能在其被声明的函数内部访问,而全局变量可以在整个程序范围内访问。调用函数时,所有在函数内声明的变量名称都将被加入到作用域中。如下实例:
```python
total = 0 # 全局变量
def sum( arg1, arg2 ):
# 返回 2 个参数的和
total = arg1 + arg2 # total在这里是局部变量.
print ("函数内是局部变量 : ", total)
return total
#调用 sum 函数
sum( 10, 20 )
print ("函数外是全局变量 : ", total)
```
## global、nonlocal 关键字
内部作用域修改外部作用域的变量时,需要使用 global 关键字声明。反之要修改嵌套作用域enclosing 作用域,外层非全局作用域)中的变量则需要 nonlocal 关键字。
```python
num = 1
def fun1():
global num # 需要使用 global 关键字声明
print(num)
num = 123
print(num)
fun1()
print(num)
```
```python
def outer():
num = 10
def inner():
nonlocal num # nonlocal关键字声明
num = 100
print(num)
inner()
print(num)
outer()
```