全面理解Python中的条件控制:if, else与switch的运用解析
导读:条件语句是指根据条件表达式的不同计算结果,使程序流转至不同的代码块。Python中的条件语句有——if语句、if… else…语句。
作者:黄传禄 张克强 赵越
来源:大数据DT(ID:hzdashuju)
01 if条件语句
if语句用于检测某个条件是否成立。如果成立,则执行if语句内的程序;否则,跳过if语句,执行后面的内容。if语句的格式如下。
if(表达式):
语句1
else:
语句2
if语句的执行过程如下:如果表达式的布尔值为真,则执行语句1;否则,执行语句2。其中的else子句可以省略,表达式两侧的括号也可以省略。
在讲解if语句之前,先介绍一下Python中的控制台输入函数。在C语言中,使用scanf()和getchar()捕获用户输入,而Java语言的System.in包提供了控制台输入的方法。Python也提供了类似功能的函数——input(),用于捕获用户的原始输入并将其转为字符串。input()函数的声明如下。
input([prompt]) -> string
参数prompt是控制台中对于输入的提示文字,提示用户输入,返回值为字符串。如果输入的是数字,返回的还是字符串,那么使用前需要调用int()做一下字符串和数字类型的转换。下面这段代码说明了字符串和数字类型的转换。
x = input("x:")
x = int(x)
x = x + 1
如果不调用int()把字符串转换为数字,而直接计算表达式x = x + 1,将提示如下错误。
TypeError: Can't convert 'int' object to str implicitly
下面这段代码演示了if语句的执行流程。
# 执行if语句内的程序
a = input("a:")
a = int(a)
b = input("b:")
b = int(b)
if(a > b):
print (a, " > ", b)
【代码说明】
- 第2行代码定义了变量a。
- 第3行将用户输入的a转换为int类型。
- 第4行代码定义了变量b。
- 第5行将用户输入的b转换为int类型。
- 第6行代码判断变量a、b的大小。
- 对于第7行代码,假设a=2、b=1,输出结果:2>1。
如果不满足if语句内的条件,程序将跳过if语句,执行后面的内容。
# 跳过if语句
a = input("a:")
a = int(a)
b = input("b:")
b = int(b)
if(a > b):
print (a, " > ", b)
print (a, " < ", b)
【代码说明】
- 假设a=1、b=2,第6行代码中变量a的值小于变量b的值,因此,程序跳转执行第8行代码。
- 第8行代码输出结果:1<2。
【例3-1】把上面的代码改写成if… else…结构。
【例3-1.py】
# if… else…语句
a = input("a:")
a = int(a)
b = input("b:")
b = int(b)
if(a > b):
print (a, " > ", b)
else:
print (a, " < ", b)
【代码说明】
- 假设a=1、b=2,第6行代码中变量a的值小于变量b的值。因此,程序跳转到else子句。
- 第9行代码输出结果:1<2。
注意:else子句后需要加一个冒号,使Python解释器能识别出else子句对应的代码块。Java程序员可能会不习惯这种语法,往往会忽略else子句后的冒号。在Python 2中还可用raw_input()函数接收用户输入,其功能与Python 3的input()相同。而Python 2中的input()接收的值不转换为字符串类型,而是保留原始类型,它在Python 3中已经去除。
02 if…elif…else判断语句
if…elif…else语句是对if…else…语句的补充。当程序的条件分支很多时,可以使用这种语句。if…elif…else语句相当于C、Java中的if…elseif…else语句。该语句的格式如下。
if(表达式1): 语句1
elif(表达式2): 语句2
…
elif(表达式n): 语句n
else: 语句m
if…elif…else语句的执行过程:首先判断表达式1的值是否为真。如果为真,则执行语句1。否则,程序流转到elif子句,判断表达式2的值是否为真。如果表达式2的值为真,则执行语句2。否则,程序进入下面一个elif子句,以此类推。如果所有的表达式都不成立,则程序执行else子句的代码。其中的else子句可以省略,表达式两侧的括号也可以省略。
下面【例3-2】中的这段代码通过判断学生的分数,确定学生成绩的等级。
【例3-2.py】
# if elif else语句
score = float( input("score:")) # 接收用户输入并将其转换为float类型,当输入为小数时,使用int转换会报错
if 90 <= score <= 100:
print("A")
elif 80 <= score < 90:
print("B")
elif 60 <= score < 80:
print("C")
else:
print("D")
【代码说明】
- 第2行代码定义了一个变量score,假设输入的值为70。这个变量表示学生的分数。接收用户输入并将其转换为float类型。
- 第3行代码,分数大于等于90并且小于等于100,则等级评定为“A”。
- 第5行代码,分数大于等于80并且小于90,则等级评定为“B”。
- 第7行代码,分数大于等于60并且小于80,则等级评定为“C”。此时条件表达式成立,程序流转到第8行。输出结果为C。
- 第9行代码,当前面的条件表达式都不成立时,程序流转到else子句。
03 if语句也可以嵌套
if语句的嵌套是指if语句中可以包含一个或多个if语句。嵌套的格式如下所示。
if(表达式1):
if(表达式2): 语句1
elif(表达式3): 语句2
…
else: 语句3
elif(表达式n):
…
else:
…
下面【例3-3】中的这段代码是一个嵌套的条件语句。如果x的值大于0,则y的值等于1;如果x的值等于0,则y的值等于0;如果x的值小于0,则y的值等于-1。
【例3-3.py】
x = -1
y = 99
if(x >= 0):
if(x > 0):#嵌套的if语句
y = 1
else:
y = 0
else:
y = -1
print ("y =", y)
【代码说明】
- 第2行代码定义了一个变量y。为了不和最终可能的输出结果1、0、-1重复,设置y的初始值为99。
- 第3行代码判断变量x的值。如果x大于等于0,则执行下面嵌套的if语句。
- 第4行代码,判读x的值是否大于0。如果大于0,则执行第5行代码;否则,执行第7行代码。
- 第8行代码,如果变量x的值小于0,则执行第9行代码。
- 第9行代码,由于变量x的值为-1,因此y的值等于-1。
- 第10行代码的输出结果为-1。
嵌套语句可以组合出很多写法,但是要注意把所有的分支情况都考虑到。下面的这种写法是错误的。
# 错误的嵌套语句
x = -1
y = 99
if(x != 0): # 如果x不等于0
if(x > 0): #嵌套的if语句
y = 1
else:
y = 0
print ("y =", y)
【代码说明】
- 第4行代码判断变量x的值是否等于0。如果不等于0,则执行if语句下面的代码块;否则执行else子句的代码。由于x的值等于-1,程序流转到第5行。
- 第5行代码判断变量x的值是否大于0。如果大于0,则变量y的值设置为1。由于这里没有考虑到变量x小于0的情况,所以程序直接跳转到第9行。
- 第9行代码,变量y的值并没有被改变,程序的分支结构没有考虑到x小于0的情况,所以最终输出的不是期望中的结果。输出结果为99。
注意:编写条件语句时,应该尽可能避免使用嵌套语句。嵌套语句不便于阅读,而且可能会忽略一些可能性。
04 switch语句的替代方案
switch语句用于编写多分支结构的程序,类似于if…elif…else语句。C语言中switch语句的结构如下所示。
switch(表达式) {
case 常量表达式1: 语句1
case 常量表达式2: 语句2
…
case 常量表达式n: 语句n
default: 语句m
}
switch语句表示的分支结构比if…elif…else语句更清晰,代码可读性更高,但是Python并没有提供switch语句,而是可以通过字典实现switch语句的功能。
实现方法分为两步。首先,定义一个字典。字典是由键值对组成的集合。其次,调用字典的get()获取相应的表达式。
下面【例3-4】中的这段代码通过算术运算的符号,获取算术运算表达式。
【例3-4.py】
# 使用字典实现switch语句
from __future__ import division# 导入division模块
x = 1
y = 2
operator = "/"
result = { # 定义字典
"+" : x + y,
"-" : x - y,
"*" : x * y,
"/" : x / y
}
print (result.get(operator))
【代码说明】
- 第3、4行代码定义了两个操作数x、y。
- 第5行代码定义了操作符变量operator,该变量用于存放算术运算符。
- 第6行代码定义了一个字典result。该字典的key值由“+”“-”“*”“/”四则运算符组成。value值由对应的算术表达式组成。
- 第12行代码调用get()方法,get()的参数就是变量operator的值。由于operator的值为“/”,因此将执行除法运算。输出结果为0.5。
另一种使用switch分支语句的方案是创建一个switch类,处理程序的流转。这种实现方法比较复杂,涉及面向对象、for循环、中断语句、遍历等知识,实现步骤分为4步。
- 创建一个switch类,该类继承自Python的祖先类object。调用构造函数__init__()初始化需要匹配的字符串,并定义两个成员变量value和fall。value用于存放需要匹配的字符串。fall用于记录是否匹配成功,初始值为False,表示匹配不成功。如果匹配成功,程序向后执行。
- 定义一个match()方法,该方法用于匹配case子句。这里需要考虑3种情况。首先是匹配成功的情况,其次是匹配失败的默认case子句,最后是case子句中没有使用break中断的情况。
- 重写__iter__()方法,定义了该方法后才能使switch类用于循环语句中。__iter__()调用match()方法进行匹配,通过yield关键字,使函数可以在循环中迭代。此外,调用异常StopIteration中断循环。Python中的循环都是通过异常StopIteration中断的。这样switch类就构造完成了。
- 编写调用代码,在for…in…循环中使用switch类。
下面【例3-5】中的这段代码实现了switch语句的功能。
【例3-5.py】
class switch(object):# 定义switch类
def __init__(self, value):# 初始化需要匹配的值value
self.value = value
self.fall = False # 如果匹配到的case语句中没有break,则fall为True
def __iter__(self):# 定义__iter__()方法
yield self.match # 调用match方法 返回一个生成器
raise StopIteration# 用StopIteration 异常来判断for循环是否结束
def match(self, *args): # 模拟case子句的方法
if self.fall or not args: # 如果fall为True,则继续执行下面的case子句
# 或case子句没有匹配项,则流转到默认分支
return True
elif self.value in args: # 匹配成功
self.fall = True
return True
else: # 匹配失败
return False
operator = "+"
x = 1
y = 2
for case in switch(operator): # switch只能用于for... in...循环中
if case('+'):
print (x + y)
break
if case('-'):
print (x - y)
break
if case('*'):
print (x * y)
break
if case('/'):
print (x / y)
break
if case(): # 默认分支
print ""
【代码说明】
- 第1行到第18行代码定义了switch类,定义了__init__()、__iter__()、match()方法。
- 第23行代码在for…in…循环中调用switch类,变量operator作为switch类的参数传递给构造函数。变量operator的值等于“+”,程序流转到第24行。
- 第25行代码输出x + y的结果。输出结果为3。
- 第26行代码使用break语句中断switch分支结构,程序流转到文件的末尾。
注意:switch语句会造成代码不易维护,使源文件臃肿。面向对象的设计中常常对switch语句进行重构,把switch语句分解为若干个类。当然,对于分支流程简单的switch,可以使用字典来实现。使用字典更容易管理switch,而switch类回到了C、Java的老路上,而且写法更复杂了,不值得推荐。
本文摘编自《零基础学Python》(第2版),经出版方授权发布。
上一篇: conditional sentences type 012
下一篇: but和if能一起用吗
推荐阅读
-
【摩尔线程+Colossal-AI强强联手】MusaBert登上CLUE榜单TOP10:技术细节揭秘 - 技术实力:摩尔线程凭借"软硬兼备"的技术底蕴,让MusaBert得以从底层优化到顶层。其内置多功能GPU配备AI加速和并行计算模块,提供了全面的AI与科学计算支持,为AI推理和低资源条件下的大模型训练等场景带来了高效、经济且环保的算力。 - 算法层面亮点:依托Colossal-AI AI大模型开发系统,MusaBert在训练过程中展现出了卓越的并行性能与易用性,特别在预处理阶段对DataLoader进行了优化,适应低资源环境高效处理海量数据。同时,通过精细的建模优化、领域内数据增强以及Adan优化器等手段,挖掘和展示了预训练语言模型出色的语义理解潜力。基于MusaBert,摩尔线程自主研发的MusaSim通过对比学习方法微调,结合百万对标注数据,MusaSim在多个任务如语义相似度、意图识别和情绪分析中均表现出色。 - 数据资源丰富:MusaBert除了自家高质量语义相似数据外,还融合了悟道开源200GB数据、CLUE社区80GB数据,以及浪潮公司提供的1TB高质量数据,保证模型即便在较小规模下仍具备良好性能。 当前,MusaBert已成功应用于摩尔线程的智能客服与数字人项目,并广泛服务于语义相似度、情绪识别、阅读理解与声韵识别等领域。为了降低大模型开发和应用难度,MusaBert及其相关高质量模型代码已在Colossal-AI仓库开源,可快速训练优质中文BERT模型。同时,通过摩尔线程与潞晨科技的深度合作,仅需一张多功能GPU单卡便能高效训练MusaBert或更大规模的GPT2模型,显著降低预训练成本,进一步推动双方在低资源大模型训练领域的共享目标。 MusaBert荣登CLUE榜单TOP10,象征着摩尔线程与潞晨科技联合研发团队在中文预训练研究领域的领先地位。展望未来,双方将携手探索更大规模的自然语言模型研究,充分运用上游数据资源,产出更为强大的模型并开源。持续强化在摩尔线程多功能GPU上的大模型训练能力,特别是在消费级显卡等低资源环境下,致力于降低使用大模型训练的门槛与成本,推动人工智能更加普惠。而潞晨科技作为重要合作伙伴,将继续发挥关键作用。
-
全面理解Python中的条件控制:if, else与switch的运用解析
-
理解与运用 Java 的条件控制:if...else结构
-
南邮OJ Web任务大揭秘:层层挑战剖析 1. 挑战一:迷宫般的目录探索 题目作者似乎穷举了所有可能的目录组合,最终在404.php中的