文章目录
循环结构
循环结构用来重复执行一条或多条语句。表达这样的逻辑:如果符合条件,则反 复执行循环体里的语句。在每次执行完后都会判断一次条件是否为 True,如果 为 True 则重复执行循环体里的语句。
循环体里面的语句至少应该包含改变条件表达式的语句,以使循环趋于结束;否则,就会变成一个死循环。
while 循环
while 循环的语法格式如下:
while 条件表达式:
循环体语句
num = 0
while num<=10:
print(num,end="\t")
num += 1
for 循环和可迭代对象遍历
for 循环通常用于可迭代对象的遍历。for 循环的语法格式如下:
for 变量 in 可迭代对象:
循环体语句
Python 包含以下几种可迭代对象:
- 序列。包含:字符串、列表、元组
- 字典
- 迭代器对象(iterator)
- 生成器函数(generator)
- 文件对象
d = {"name":"cat","age":1,"hobby":"fishing"} #字典
for x in d: #默认打印键
print(x)
for x in d.keys():#打印键
print(x)
for x in d.values():#打印值
print(x)
for x in d.items():#打印键值对
print(x)
'''运行结果
name
age
hobby
name
age
hobby
cat
1
fishing
('name', 'cat')
('age', 1)
('hobby', 'fishing')
'''
计算0-100的和,奇数和,偶数和
sum_all = 0
sum_odd = 0 #100以内的奇数和
sum_even = 0 #100以内的偶数和
for x in range(101):
sum_all += x
if x%2==1:
sum_odd += x
else:
sum_even += x
print("1-100累加总和{0},奇数和{1},偶数和{2}".format(sum_all,sum_odd,sum_even))
嵌套循环
一个循环体内可以嵌入另一个循环,一般称为“嵌套循环”,或者“多重循环”。
打印此图:
for x in range(5): #x用来决定打印出的数字
for y in range(5): #y用来决定一个数字打印几次
print(x,end='\t')
print() #print()等价于print('\n'),起到换行的作用
打印九九乘法表:
for x in range(1,10):
for y in range(1,x+1):
print("{0}*{1}={2}".format(x,y,x*y),end="\t")
print() #换行
'''
结果:
1*1=1
2*1=2 2*2=4
3*1=3 3*2=6 3*3=9
4*1=4 4*2=8 4*3=12 4*4=16
5*1=5 5*2=10 5*3=15 5*4=20 5*5=25
6*1=6 6*2=12 6*3=18 6*4=24 6*5=30 6*6=36
7*1=7 7*2=14 7*3=21 7*4=28 7*5=35 7*6=42 7*7=49
8*1=8 8*2=16 8*3=24 8*4=32 8*5=40 8*6=48 8*7=56 8*8=64
9*1=9 9*2=18 9*3=27 9*4=36 9*5=45 9*6=54 9*7=63 9*8=72 9*9=81
'''
用列表和字典存储下表信息,并打印出表中工资高于 15000 的数据
r1= dict(name="高小一",age=18,salary=30000,city="北京")
r2= dict(name="高小二",age=19,salary=20000,city="上海")
r3= dict(name="高小三",age=20,salary=10000,city="深圳")
tb = [r1,r2,r3]
for x in tb:
if x.get("salary")>15000:
print(x)
break 语句
break 语句可用于 while 和 for 循环,用来结束整个循环。当有嵌套循环时,break 语句只能跳出最近一层的循环。
while True:
a = input("请输入一个字符(输入Q或q时退出):")
if a=="q" or a=="Q":
print("循环结束,退出")
break
else:
print(a)
continue 语句
continue 语句用于结束本次循环,继续下一次。多个循环嵌套时,continue 也是应用于最近的一层循环。
empNum = 0
salarySum= 0
salarys = []
while True:
s = input("请输入员工的薪资(按Q或q结束)")
if s.upper()=='Q':
print("录入完成,退出")
break
if float(s)<0:
continue
empNum +=1
salarys.append(float(s))
salarySum += float(s)
print("员工数{0}".format(empNum))
print("录入薪资:",salarys)
print("平均薪资{0}".format(salarySum/empNum))
else 语句
while、for 循环可以附带一个 else 语句(可选)。如果 for、while 语句没有被 break 语句结束,则会执行 else 子句,若出现break则不执行else(其实就是else也是循环体的一部分)。
语法格式如下:
while 条件表达式:
循环体
else:语句块
或者:
for 变量 in 可迭代对象:
循环体
else:语句块
salarySum= 0
salarys = []
for i in range(4):
s = input("请输入一共4名员工的薪资(按Q或q中途结束)")
if s.upper()=='Q':
print("录入完成,退出")
break
if float(s)<0:
continue
salarys.append(float(s))
salarySum += float(s)
else:
print("您已经全部录入4名员工的薪资")
print("录入薪资:",salarys)
'''
结果:
请输入一共4名员工的薪资(按Q或q中途结束)20000
请输入一共4名员工的薪资(按Q或q中途结束)30000
请输入一共4名员工的薪资(按Q或q中途结束)40000
请输入一共4名员工的薪资(按Q或q中途结束)50000
您已经全部录入4名员工的薪资 #没有遇到break语句所以执行了else部分
录入薪资: [20000.0, 30000.0, 40000.0, 50000.0]
请输入一共4名员工的薪资(按Q或q中途结束)30000
请输入一共4名员工的薪资(按Q或q中途结束)q #遇到break所以结束循环且不执行else部分
录入完成,退出
录入薪资: [30000.0]
'''
循环代码优化
虽然计算机越来越快,空间也越来越大,我们仍然要在性能问题上“斤斤计较”。编写循环时,遵守下面三个原则可以大大提高运行效率,避免不必要的低效计算: 1. 尽量减少循环内部不必要的计算 2. 嵌套循环中,尽量减少内层循环的计算,尽可能向外提。 3. 局部变量查询较快,尽量使用局部变量
import time
start = time.time()
for i in range(1000):
result = []
for m in range(10000):
result.append(i*1000+m*100)
end = time.time()
print("耗时:{0}".format((end-start)))
start2 = time.time()
for i in range(1000):
result = []
c = i*1000
for m in range(10000):
result.append(c+m*100)
end2 = time.time()
print("耗时:{0}".format((end2-start2)))
'''
结果:
耗时:2.099863290786743
耗时:1.6899256706237793
'''
其他优化手段
- 连接多个字符串,使用 join()而不使用+
- 列表进行元素插入和删除,尽量在列表尾部操作
empNum = 0
salarySum= 0
salarys = []
while True:
s = input("请输入员工的薪资(按Q或q结束)")
if s.upper()=='Q':
print("录入完成,退出")
break
if float(s)<0:
continue
empNum +=1
salarys.append(float(s))
salarySum += float(s)
print("员工数{0}".format(empNum))
print("录入薪资:",salarys)
print("平均薪资{0}".format(salarySum/empNum))
使用 zip()并行迭代
我们可以通过 zip()函数对多个序列进行并行迭代,zip()函数在最短序列“用完”时就会停止。
names = ("ping","an","xi","le")
ages = (18,16,20,25)
jobs = ("老师","程序员","公务员")
for name,age,job in zip(names,ages,jobs):
print("{0}--{1}--{2}".format(name,age,job))
for i in range(3): #普通方法也可以完成并行迭代
print("{0}--{1}--{2}".format(names[i],ages[i],jobs[i]))
'''
结果:
ping--18--老师
an--16--程序员
xi--20--公务员
ping--18--老师
an--16--程序员
xi--20--公务员
'''
推导式创建序列
推导式是从一个或者多个迭代器快速创建序列的一种方法。它可以将循环和条件判断结合, 从而避免冗长的代码。推导式是典型的 Python 风格
列表推导式
列表推导式生成列表对象,语法如下:
[表达式 for item in 可迭代对象 ] 或者:[表达式 for item in 可迭代对象 if 条件判断]
y = [x*2 for x in range(1,50) if x%5==0]
print(y)
y = []
for x in range(1,50):
if x%5==0:y.append(x*2)
print(y)
cells = [(row,col) for row in range(1,4) for col in range(1,4)]
print(cells)
'''
结果:
[10, 20, 30, 40, 50, 60, 70, 80, 90]
[10, 20, 30, 40, 50, 60, 70, 80, 90]
[(1, 1), (1, 2), (1, 3), (2, 1), (2, 2), (2, 3), (3, 1), (3, 2), (3, 3)]
'''
字典推导式
字典的推导式生成字典对象,格式如下: {key_expression : value_expression for 表达式 in 可迭代对象} 类似于列表推导式,字典推导也可以增加 if 条件判断、多个 for 循环。
my_text = "i love you, i love cat, i love dog"
test={my_text.count(q):q for q in my_text}
print(test)
char_count = { c:my_text.count(c) for c in my_text}
print(char_count)
'''
结果:
{3: 'e', 8: ' ', 5: 'o', 1: 'g', 2: ','} #注意:test的结果与char_count不同
{'i': 3, ' ': 8, 'l': 3, 'o': 5, 'v': 3, 'e': 3, 'y': 1, 'u': 1, ',': 2, 'c': 1, 'a': 1, 't': 1, 'd': 1, 'g': 1}
#注意:这里进行迭代的时候自动忽略了已经出现过的字符,是因为迭代的对象为键,而字典的键不可以重复
'''
注意:上面的代码运行结果显示test的结果与char_count不同,因为test以字符出现的次数为键,而字典的键是唯一的,并且字典中遇到重复的key时,后面出现的键值对会覆盖前面的键值对,最终显示最后出现的那个键值对
qqq={1:'a',2:'b',3:'c',2:'w',2:'z'}
print(qqq)
'''result:
{1: 'a', 2: 'z', 3: 'c'}
'''
集合推导式
集合推导式生成集合,和列表推导式的语法格式类似: {表达式 for item in 可迭代对象 } 或者:{表达式 for item in 可迭代对象 if 条件判断}
b = {x for x in range(1,100) if x%9==0}
print(b)
生成器推导式(生成元组)
很多同学可能会问:“都有推导式,元组有没有?”,能不能用小括号呢?
>>> (x for x in range(1,100) if x%9==0)
<generator object <genexpr> at 0x0000000002BD3048>
我们发现提示的是“一个生成器对象”。显然,元组是没有推导式的。
gnt = (x for x in range(4))
print(gnt)
print(tuple(gnt))
print(tuple(gnt)) #一个生成器只能运行一次。第一次迭代可以得到数据,第二次迭代发现数据已经没有了。
'''
结果:
<generator object <genexpr> at 0x00000203B5F816C8>
(0, 1, 2, 3)
()
'''
gnt = (x for x in range(4))
for x in gnt:
print(x,end=",")
print(tuple(gnt))#gnt是生成器对象,生成器只能使用一次
'''
结果:
0,1,2,3,()
'''
绘制同心圆
import turtle
t = turtle.Pen()
my_colors = ("red","green","yellow","black")
t.width(4)
t.speed(1) #0是最快的,1是最慢的,5、10等等都比较快
for i in range(10): #0 1 2 3 4
t.penup()
t.goto(0,-i*10) #0, -100,-200,-300,-400
t.pendown()
t.color(my_colors[i%len(my_colors)]) #注意:i%4的结果只有0,1,2,3这4个值
t.circle(15+i*10) #100,200,300, 400,, 500
turtle.done() #程序执行完,窗口仍然在
绘制棋盘
import turtle
t=turtle.Pen()
t.speed(0)
for i in range(11): #0,1,2,3..
t.penup()
t.goto(0,-i*10)#(0,0)(0,-10)(0,-20)
t.pendown()
t.goto(100,-i*10)#(100,0)(100,-10)(100,-20)
for j in range(11):
t.penup()
t.goto(j*10,0)
t.pendown()
t.goto(j*10,-100)
函数
函数是可复用的程序代码块。函数不仅可以实现代码的复用,更能实现代码的一致性。一致性指的是,只要修改函数的代码,则所有调用该函数的地方都能得到体现。 在编写函数时,函数体中的代码写法和我们前面讲述的基本一致,只是对代码实现了封 装,并增加了函数调用、传递参数、返回计算结果等内容。 函数是代码复用的通用机制。一个程序由一个个任务组成;函数就是代表一个任务或者一个功能。
Python函数的分类
Python 中函数分为如下几类:
- 内置函数 我们前面使用的 str()、list()、len()等这些都是内置函数,我们可以拿来直接使用。
- 标准库函数 我们可以通过 import 语句导入库,然后使用其中定义的函数
- 第三方库函数 Python 社区也提供了很多高质量的库。下载安装这些库后,也是通过 import 语句导 入,然后可以使用这些第三方库的函数
- 用户自定义函数 用户自己定义的函数,显然也是开发中适应用户自身需求定义的函数。
函数的定义和调用
Python 中,定义函数的语法如下:
def 函数名 ([参数列表]) :
‘’‘文档字符串’’’#其实就是函数的注解说明
函数体/若干语句
要点:
- 我们使用 def 来定义函数,然后就是一个空格和函数名称;Python 执行 def 时,会创建一个函数对象,并绑定到函数名变量上。
- 参数列表 :(1) 圆括号内是形式参数列表,有多个参数则使用逗号隔开 (2) 形式参数不需要声明类型,也不需要指定函数返回值类型 (3) 无参数,也必须保留空的圆括号 (4) 实参列表必须与形参列表一一对应
- return 返回值: (1) 如果函数体中包含 return 语句,则结束函数执行并返回值; (2) 如果函数体中不包含 return 语句,则返回 None 值。(3)return 语句就是将结果返回到调用的地方,并把程序的控制权一起返回,程序运行到所遇到的第一个return即返回(退出def块),不会再运行第二个return。
- 调用函数之前,必须要先定义函数,即先调用 def 创建函数对象 (1) 内置函数对象会自动创建 (2) 标准库和第三方库函数,通过 import 导入模块时,会执行模块中的 def 语句
形参、实参和文档字符串(函数的注释)
简单来说,形式参数是在定义函数时使用的。 形式参数的命名只要符合“标识符”命名规则即可。 在调用函数时,传递的参数称为“实际参数”,简称“实参”。程序的可读性最重要,一般建议在函数体开始的部分附上函数定义说明,这就是“文档字符串”,也有人成为“函数的注释”。我们通过三个单引号或者三个双引号来实现,中间可以加入多行文字进行说明。调用 help(函数名.doc)可以打印输出函数的文档字符串。
def printMax(a,b):
'''用于比较两个数的大小,打印较大的值'''
if a>b:
print(a,"较大值")
else:
print(b,"较大值")
printMax(10,20)
printMax(200,300)
help(printMax.__doc__)
'''
结果:
20 较大值
300 较大值
No Python documentation found for '用于比较两个数的大小,打印较大的值'.
Use help() to get the interactive help utility.
Use help(str) for help on the str class.
'''
返回值
return 返回值要点:
- 如果函数体中包含 return 语句,则结束函数执行并返回值;
- 如果函数体中不包含 return 语句,则返回 None 值。
- 要返回多个返回值,使用列表、元组、字典、集合将多个值“存起来”即可。
- return 语句就是将结果返回到调用的地方,并把程序的控制权一起返回,程序运行到所遇到的第一个return即返回(退出def块),不会再运行第二个return。
def add(a,b):
print("计算两个数的和:{0},{1}".format(a,b,))
return a+b
def test02():
print("aaa")
print("bbb")
return #注意:return两个作用:1.返回值 2. 结束函数的执行。有return则函数在此结束执行,即使调用test02()也不会打印hello
print("hello")
def test03():
print("aaa")
print("bbb")
print("hello") #没有return,也返回None
def test04(x,y,z):
return [x*10,y*10,z*10]
c = add(30,40)
print(c)
d = test02()
print(d)
e = test03()
print(e)
print(test04(4,3,2))
'''
结果:
计算两个数的和:30,40
70
aaa
bbb
None
aaa
bbb
hello
None
[40, 30, 20]
'''
函数对象的内存示意
Python 中,“一切都是对象”。实际上,执行 def 定义函数后,系统就创建了相应的函数对象。
def print_star(n):
print("*"*n)
print(print_star)
print(id(print_star))
c = print_star
c(3)
'''
结果:
<function print_star at 0x0000000002BB8620>
45844000
***
'''
上面代码执行 def 时,系统中会创建函数对象,并通过 print_star 这个变量进行引用:
我们执行“c=print_star”后,显然将 print_star 变量的值赋给了变量 c,内存图变成了:
显然,我们可以看出变量 c 和 print_star 都是指向了同一个函数对象。因此,执行 c(3)和执 行 print_star(3)的效果是完全一致的。 Python 中,圆括号意味着调用函数。在没有圆括号的情况下,Python 会把函数当做普通对象。
与此核心原理类似,我们也可以做如下操作:
zhengshu = int
zhengshu(“234”)
显然,我们将内置函数对象 int()赋值给了变量 zhengshu,这样 zhengshu 和 int 都是指向了同一个内置函数对象。当然,此处仅限于原理性讲解,实际开发中没必要这么做。
def test01():
print("asdasd")
test01()
c = test01
c()
print(id(test01))
print(id(c))
print(type(c))
print(c() is test01())
'''
结果:
asdasd
asdasd
45844000
45844000
<class 'function'>
asdasd
asdasd
True
'''