1
0
wiki/Tech/programming-language/Python/入门/控制语句.md

357 lines
12 KiB
Markdown
Raw Normal View History

2022-02-15 17:27:43 +08:00
---
2022-04-28 11:20:41 +08:00
title: 控制语句
2022-11-19 21:18:21 +08:00
description: Python 控制语句
keywords:
2023-11-09 17:30:33 +08:00
- Python
- 控制语句
2022-11-19 21:18:21 +08:00
tags:
2023-11-09 17:30:33 +08:00
- Python/入门
2022-11-19 21:18:21 +08:00
sidebar_position: 3
author: 7Wate
2023-08-03 20:12:42 +08:00
date: 2023-08-03
2022-02-15 17:27:43 +08:00
---
2023-08-03 20:12:42 +08:00
Python 中的控制语句有条件语句,循环语句,异常处理,以及其他一些特殊的控制语句。
2022-12-03 18:18:23 +08:00
2023-08-03 20:12:42 +08:00
## 条件语句
2023-11-09 17:30:33 +08:00
Python 中的条件语句是通过一条或多条语句的执行结果(即 True 或 False来决定执行的代码块。
2023-08-03 20:12:42 +08:00
判断的值可以分为:
2022-12-03 18:18:23 +08:00
- 假值 None、空列表、空集合、空字典空元组、空字符串、0、False 等。
- 真值 :非空列表、非空集合、非空字典,非空元组、非空字符串、非 0 数值、True 等。
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### If
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
在 Python 中,要构造分支结构可以使用 `if`、`elif` 和 `else` 关键字。`elif` 和 `else` 都是可选的,可以根据需要进行使用。
2022-02-15 17:27:43 +08:00
```python
2022-11-20 19:10:55 +08:00
# 示例
2022-11-19 21:18:21 +08:00
>>> x = int(input("Please enter an integer: "))
Please enter an integer: 42
>>> if x < 0:
x = 0
print('Negative changed to zero')
elif x == 0:
print('Zero')
elif x == 1:
print('Single')
else:
print('More')
2022-02-15 17:27:43 +08:00
```
2023-11-09 17:30:33 +08:00
当然如果要构造出更多的分支,可以使用 `if...elif...else...` 结构或者嵌套的 `if...else...` 结构。
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### Match
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
Python 3.10 引入了新的 `match` 语句,它是模式匹配的一种形式。`match` 语句接受一个表达式并将它的值与一系列的模式进行比较。 每个模式都关联到一个代码块,当模式与表达式的值匹配时,该代码块将被执行。这在某种程度上类似于其他语言中的 switch 语句。
2022-02-15 17:27:43 +08:00
```python
def http_error(status):
match status:
case 400:
return "Bad request"
case 404:
return "Not found"
case 418:
return "I'm a teapot"
2023-08-03 20:12:42 +08:00
#"变量名" `_` 被作为 通配符 并必定会匹配成功
case _:
2022-02-15 17:27:43 +08:00
return "Something's wrong with the internet"
```
你可以使用 `|` (“ or ”)在一个模式中组合几个字面值:
```python
case 401 | 403 | 404:
return "Not allowed"
```
2023-11-09 17:30:33 +08:00
*注意,`match` 语句和模式匹配的概念是 Python 3.10 中新增的特性,可能在更早版本的 Python 中无法使用。*
2023-08-03 20:12:42 +08:00
## 循环语句
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### For
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
Python 的 for 语句用于遍历任何序列的项目,如列表或字符串。
2022-11-19 21:18:21 +08:00
Python 的 for 语句与 C 或 Pascal 中的不同。**Python 的 for 语句不迭代算术递增数值**(如 Pascal或是给予用户定义迭代步骤和暂停条件的能力如 C而是迭代列表或字符串等任意序列元素的迭代顺序与在序列中出现的顺序一致。
2022-02-15 17:27:43 +08:00
```python
"""
用for循环实现1~100求和
"""
sum = 0
for x in range(101):
sum += x
print(sum)
```
2022-11-19 21:18:21 +08:00
**内置函数 range() 表示不可变的数字序列,通常用于在 for 循环中循环指定的次数。**
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
在 Python 中,`range()` 是一个内置函数,用于生成一个不可变的数字序列。通常,这个函数在 for 循环中使用,用于指定循环的次数。`range(101)` 可以用来生成一个包含 0 到 100不包含 101的整数序列。
2023-08-03 20:12:42 +08:00
2023-11-09 17:30:33 +08:00
你也可以根据需要更改 `range()` 函数的参数,例如:
2022-11-19 21:18:21 +08:00
2023-11-09 17:30:33 +08:00
- `range(1, 101)`:生成一个包含 1 到 100不包含 101的整数序列。
- `range(1, 101, 2)`:生成一个包含 1 到 100 的奇数序列,其中 2 是步长。
- `range(100, 0, -2)`:生成一个包含 100 到 1 的偶数序列,其中 -2 是步长。
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### While
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
`while` 循环语句用于在条件满足的情况下重复执行一个代码块。条件表达式的结果为 `True` 时,继续循环;结果为 `False` 时,结束循环。
2022-02-15 17:27:43 +08:00
```python
"""
猜数字游戏
"""
import random
answer = random.randint(1, 100)
counter = 0
while True:
counter += 1
number = int(input('请输入: '))
if number < answer:
print('大一点')
elif number > answer:
print('小一点')
else:
print('恭喜你猜对了!')
break
print('你总共猜了%d次' % counter)
if counter > 7:
print('你的智商余额明显不足')
```
2023-11-09 17:30:33 +08:00
### Break
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
`break` 语句可以提前退出循环。具体来说,**`break` 用于完全结束一个循环,并跳出该循环体。**
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### Continue
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
`continue` 语句用于**跳过当前循环的剩余语句,然后继续进行下一轮循环。**
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### Else
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
在 Python 中,`else` 子句可以与 `for` 循环和 `while` 循环一起使用。当循环正常完成(即没有碰到 `break` 语句)时,`else` 块的内容会被执行。这个特性在很多其他语言中都没有,因此对于初学者来说可能会感到有些不熟悉。
2022-02-15 17:27:43 +08:00
```python
2023-08-03 20:12:42 +08:00
# 使用 else 的 for 循环的示例
for i in range(5):
if i == 10:
break
else:
print("循环正常完成")
2022-02-15 17:27:43 +08:00
```
2023-08-03 20:12:42 +08:00
在这个示例中,循环会正常完成,因为没有任何一个元素使得 `i == 10``True`,所以 `break` 语句不会被执行,`else` 块的内容会被打印出来。
如果我们修改 `if` 语句的条件使得 `break` 语句被执行,那么 `else` 块的内容就不会被打印出来:
```Python
for i in range(5):
if i == 3:
break
else:
print("循环正常完成")
```
2023-11-09 17:30:33 +08:00
在这个示例中,当 `i` 等于 3 时,`break` 语句就会被执行,所以循环没有正常完成,`else` 块的内容不会被打印出来。
2023-08-03 20:12:42 +08:00
同样的,`else` 也可以与 `while` 循环一起使用,当 `while` 循环的条件变为 `False` 时,`else` 块的内容会被执行。
```python
i = 0
while i < 5:
if i == 3:
break
i += 1
else:
print("循环正常完成")
```
2023-11-09 17:30:33 +08:00
在这个示例中,当 `i` 等于 3 时,`break` 语句就会被执行,所以循环没有正常完成,`else` 块的内容不会被打印出来。
2023-08-03 20:12:42 +08:00
## 异常语句
2022-02-15 17:27:43 +08:00
### try、except、finally
2023-08-03 20:12:42 +08:00
```mermaid
graph TD
A(开始)
B[try 块]
C[except 块1]
D[except 块2]
E[else 块]
F[finally 块]
G(结束)
A --> B
B -- 异常1 --> C
B -- 异常2 --> D
B -- 无异常 --> E
C --> F
D --> F
E --> F
F --> G
```
2022-11-20 19:10:55 +08:00
2023-08-03 20:12:42 +08:00
1. **确定需要的异常处理结构**:在开始编写异常处理结构时,首先确定你需要 `try`, `except`, `else` 和/或 `finally` 语句块。要记住的是,`except` 和 `finally` 块至少需要一个,否则 `try` 将失去其意义。
2. **编写 `try` 语句块**:将可能抛出异常的代码放入 `try` 块。如果在 `try` 块中发生异常Python 将停止执行 `try` 块的其余部分,并转到 `except` 块。
3. **编写 `except` 语句块**:为每种可能抛出的异常类型编写一个 `except` 块。Python 会按照它们在代码中出现的顺序来检查这些 `except` 块。如果匹配到异常Python 将执行相应的 `except` 块并停止查找。
4. **处理多个异常**:你可以将多个异常类型放入一个元组中,然后使用一个 `except` 块来处理它们。例如:`except (TypeError, ValueError):`。
5. **处理所有异常**:如果 `except` 块后面没有指定异常类型,那么这个 `except` 块将处理所有异常。你可以通过 `logging``sys` 模块获取异常的详细信息。
6. **重新抛出异常**:如果你在 `except` 块中捕获了一个异常,然后想要再次抛出它,你可以使用 `raise` 语句而不需要附加任何参数或信息。
7. **编写 `else` 语句块**`else` 块中的代码只有在 `try` 块没有发生任何异常时才会被执行。
8. **编写 `finally` 语句块**:无论是否发生异常,`finally` 块中的代码总是会被执行。这对于清理(例如关闭文件或网络连接)非常有用。
9. **重构代码**:考虑使用 `with` 语句或 `getattr()` 方法等内置的异常处理语句,而不是 `try/except`。此外,如果可能,尽量避免在同一个 `except` 块中捕获和抛出相同的异常。最后,除非你确定需要处理所有可能的异常,否则不应该捕获所有异常,因为这可能会隐藏严重的问题。
2022-02-15 17:27:43 +08:00
```python
2023-08-03 20:12:42 +08:00
# 示例
try:
# 这里可能会抛出异常的代码
do_something()
except SomeException as e:
# 当捕获到SomeException时的处理代码
handle_exception(e)
# 实例
2022-02-15 17:27:43 +08:00
def div(a, b):
try:
print(a / b)
except ZeroDivisionError:
2023-08-03 20:12:42 +08:00
print("错误b 不应为 0 !!")
2022-02-15 17:27:43 +08:00
except Exception as e:
2023-08-03 20:12:42 +08:00
print("意外错误:{}".format(e))
2022-02-15 17:27:43 +08:00
else:
2023-08-03 20:12:42 +08:00
print('只有当一切正常时,才会运行 else')
2022-02-15 17:27:43 +08:00
finally:
2023-08-03 20:12:42 +08:00
print('始终运行 finally 块。')
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
# 测试
2022-02-15 17:27:43 +08:00
div(2, 0)
2023-08-03 20:12:42 +08:00
div(2, '错误的类型')
2022-02-15 17:27:43 +08:00
div(1, 2)
2023-08-03 20:12:42 +08:00
# 在一行中捕获多个异常
2022-02-15 17:27:43 +08:00
try:
print(a / b)
except (ZeroDivisionError, TypeError) as e:
print(e)
2023-08-03 20:12:42 +08:00
# 当存在 finally 时except 是可选的
2022-02-15 17:27:43 +08:00
try:
open(database)
finally:
close(database)
2023-08-03 20:12:42 +08:00
# 捕获所有错误并记录下来
2022-02-15 17:27:43 +08:00
try:
do_work()
except:
2023-08-03 20:12:42 +08:00
# 从 logging 模块获取详细信息
logging.exception('捕获到异常!')
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
# 从 sys.exc_info() 方法获取详细信息
2022-02-15 17:27:43 +08:00
error_type, error_value, trace_back = sys.exc_info()
print(error_value)
raise
```
2023-11-09 17:30:33 +08:00
### With
2022-11-19 21:18:21 +08:00
2023-08-03 20:12:42 +08:00
Python 的 with 语句支持通过上下文管理器所定义的运行时上下文这一概念。
2022-11-19 21:18:21 +08:00
2023-11-09 17:30:33 +08:00
`with` 语句是一种处理上下文管理器的语句,上下文管理器通常包含 `__enter__``__exit__` 这两个方法。在 `with` 语句的代码块被执行前,会首先执行 `__enter__` 方法,在执行完毕后,会调用 `__exit__` 方法。这在你需要管理资源,如文件,网络连接或锁定等情况非常有用,因为它可以**保证在任何情况下都会执行必要的清理操作。**
2022-11-19 21:18:21 +08:00
```python
2023-08-03 20:12:42 +08:00
# 在这里,文件已经被关闭,无需再次手动关闭
with open("file.txt", "r") as file:
for line in file:
print(line)
2022-11-19 21:18:21 +08:00
```
```python
# 基本思想是 with 所求值的对象必须有一个 enter() 方法,一个 exit() 方法。
# 紧跟 with 后面的语句被求值后,返回对象的 enter() 方法被调用,并将返回值赋值给 as 后面的变量。
# 当 with 的代码块全部被执行完之后,将调用前面返回对象的 exit() 方法。
2023-08-03 20:12:42 +08:00
class ManagedFile:
def __init__(self, filename):
self.filename = filename
2022-11-19 21:18:21 +08:00
2023-08-03 20:12:42 +08:00
def __enter__(self):
self.file = open(self.filename, 'r')
return self.file
2022-11-19 21:18:21 +08:00
2023-08-03 20:12:42 +08:00
def __exit__(self, exc_type, exc_val, exc_tb):
if self.file:
self.file.close()
2022-11-19 21:18:21 +08:00
2023-08-03 20:12:42 +08:00
# 使用自定义的上下文管理器
with ManagedFile("file.txt") as file:
print(file.read())
2022-11-19 21:18:21 +08:00
```
2023-11-09 17:30:33 +08:00
### Raise
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
`raise` 语句用于引发特定的异常。你可以定义异常类型并附加一个错误消息:
2022-02-15 17:27:43 +08:00
```python
2023-08-03 20:12:42 +08:00
raise ValueError("这是一个无效的值!")
```
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
如果你在 `except` 块中使用 `raise` 语句,而不提供任何参数,它将默认重新引发最近的异常。
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
```python
try:
print(5/0)
except ZeroDivisionError as e:
print("发生了一个错误!")
raise # 重新引发最近的异常
2022-02-15 17:27:43 +08:00
```
2023-08-03 20:12:42 +08:00
## 其他语句
2022-02-15 17:27:43 +08:00
2023-11-09 17:30:33 +08:00
### Assert
2022-11-19 21:18:21 +08:00
2023-08-03 20:12:42 +08:00
`assert` 语句用于断言某个条件为真,如果条件为假,则会抛出 `AssertionError` 异常。它常常用于调试代码,确认代码的某些方面满足预期,例如:
2022-11-19 21:18:21 +08:00
```python
2023-08-03 20:12:42 +08:00
x = 1
assert x == 1 # 条件为真,没有问题
2022-11-19 21:18:21 +08:00
2023-08-03 20:12:42 +08:00
# 条件为假,抛出 AssertionError并附带错误信息
assert x == 2, "x should be 2 but is actually " + str(x)
2022-11-19 21:18:21 +08:00
```
2023-08-03 20:12:42 +08:00
*注意,`assert` 语句在优化模式下(使用 `-O` 参数启动 Python 时)会被全局禁用。*
2023-11-09 17:30:33 +08:00
### Pass
2022-02-15 17:27:43 +08:00
2023-08-03 20:12:42 +08:00
`pass` 语句是 Python 中的空语句,用于在需要语句的地方保持语法的完整性,但是**实际上不做任何事情**。通常,我们使用它作为未完成代码的占位符:
2022-02-15 17:27:43 +08:00
```python
2023-08-03 20:12:42 +08:00
def my_function():
pass # TODO: implement this function
2022-02-15 17:27:43 +08:00
class MyEmptyClass:
pass
2022-02-21 15:50:42 +08:00
```
2023-11-09 17:30:33 +08:00
### Return
2022-02-21 15:50:42 +08:00
2023-08-03 20:12:42 +08:00
`return` 语句用于从函数返回一个值。所有函数都会返回一个值:如果函数执行到结尾而没有遇到 `return` 语句,它将返回特殊值 `None`
2022-02-21 15:50:42 +08:00
```python
2023-08-03 20:12:42 +08:00
def add(a, b):
return a + b
print(add(1, 2)) # 输出3
2022-02-21 15:50:42 +08:00
```