Python 基础课程第五天
第四章:控制语句
循环结构
for循环和可迭代对象遍历
for 循环通常用于可迭代对象的遍历。for 循环的语法格式如下:
for 变量 in 可迭代对象:
循环体语句
【操作】遍历一个元组或列表
for x in (10,20,30):
print(x*30)
可迭代对象
Python 包含以下几种可迭代对象:
- 序列。包含:字符串、列表、元组
- 字典
- 迭代器对象(iterator)
- 生成器函数(generator)
- 文件对象
我们已经在前面学习了序列、字典等知识,迭代器对象和生成器函数将在后面进行详解。接下来,我们通过循环来遍历这几种类型的数据:
【操作】遍历字符串中的字符
for y in "abcdefg":
print(y)
【操作】遍历字典
d={"name":"gaoqi","age":18,"job":"programmer"}
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)
for x in range(5):
print(x)
range 对象
range 对象是一个迭代器对象,用来产生指定范围的数字序列。格式为:
range(start, end [,step])
生成的数值序列从start 开始到end 结束(不包含end)。若没有填写start,则默认从0开始。step 是可选的步长,默认为1。
【操作】利用for 循环,计算1-100 之间数字的累加和;计算1-100 之间偶数的累加和,计算1-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))
运算结果:
1-100以内的累加总和为5050,奇数和为2500,偶数和为2550
嵌套循环和综合练习
一个循环体内可以嵌入另一个循环,一般称为“嵌套循环”,或者“多重循环”。
【操作】打印如下图案
0 0 0 0 0
1 1 1 1 1
2 2 2 2 2
3 3 3 3 3
4 4 4 4 4
for x in range(5):
for y in range(5):
print(x,end="\t")
print() #仅用于换行
【操作】利用嵌套循环打印九九乘法表
for m in range(1,10):
for n in range(1,m+1):
print("{0}*{1}={2}".format(m,n,(m*n)),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)
运行结果:
{'name': '高小一', 'age': 18, 'salary': 30000, 'city': '北京'}
{'name': '高小二', 'age': 19, 'salary': 20000, 'city': '上海'}
break 语句
break 语句可用于while 和for 循环,用来结束整个循环。
当有嵌套循环时,break 语句只能跳出最近一层的循环。
【操作】使用break 语句结束循环
#测试Break语句
while True:
a=input("请输入一个字符(输入Q或者q时结束):")
if a=="Q" or a=="q":
print("循环结束,退出")
break
else:
print(a)
continue 语句
continue 语句用于结束本次循环,继续下一次。多个循环嵌套时,continue 也是应用于最近的一层循环。
【操作】要求输入员工的薪资,若薪资小于0 则重新输入。最后打印出录入员工的数量和薪资明细,以及平均薪资
# 测试continue语句
empNum=0
salarySum=0
salarys=[]
while True:
s=input("请输入员工的薪资(按Q或q结束)")
if s.upper()=="Q":
print("录入完成,退出")
break
if float(s)<0:
continue
else:
empNum +=1
salarys.append(float(s))
salarySum +=float(s)
print("员工数{0}".format(empNum))
print("录入薪资:",salarys)
print("平均薪资{0}".format(salarySum/empNum))
运行结果:
请输入员工的薪资(按Q或q结束)2000
请输入员工的薪资(按Q或q结束)3000
请输入员工的薪资(按Q或q结束)4000
请输入员工的薪资(按Q或q结束)5000
请输入员工的薪资(按Q或q结束)-1000
请输入员工的薪资(按Q或q结束)q
录入完成,退出
员工数4
录入薪资: [2000.0, 3000.0, 4000.0, 5000.0]
平均薪资3500.0
else 语句
while、for 循环可以附带一个else 语句(可选)。如果for、while 语句没有被break 语句结束,则会执行else 子句,否则不执行。语法格式如下:
【操作】员工一共4 人。录入这4 位员工的薪资。全部录入后,打印提示“您已经全部录入4 名员工的薪资”。最后,打印输出录入的薪资和平均薪资。
#测试else语句
salarySum=0
salarys=[]
for i in range(4):
s=input("请输入一共四名员工的薪资:")
if s.upper()=="Q":
print("录入完成,退出")
break
if float(s)<0:
continue
salarys.append(float(s))
salarySum +=float(s)
else:
print("您已经全部录入四名员工的薪资")
print("录入薪资:",salarys)
print("平均薪资{0}".format(salarySum/4))
运行结果:
请输入一共四名员工的薪资:1000
请输入一共四名员工的薪资:2000
请输入一共四名员工的薪资:3000
请输入一共四名员工的薪资:4000
您已经全部录入四名员工的薪资
录入薪资: [1000.0, 2000.0, 3000.0, 4000.0]
平均薪资2500.0
循环代码优化
虽然计算机越来越快,空间也越来越大,我们仍然要在性能问题上“斤斤计较”。编写循环时,遵守下面三个原则可以大大提高运行效率,避免不必要的低效计算:
- 尽量减少循环内部不必要的计算
- 嵌套循环中,尽量减少内层循环的计算,尽可能向外提。
- 局部变量查询较快,尽量使用局部变量
#循环代码优化测试
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("耗时:{}".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("耗时:{}".format(end2-start2))
运行结果:
耗时:3.3328819274902344
耗时:2.9132978916168213
其他优化手段
- 连接多个字符串,使用join()而不使用+
- 列表进行元素插入和删除,尽量在列表尾部操作
使用zip()并行迭代
我们可以通过zip()函数对多个序列进行并行迭代,zip()函数在最短序列“用完”时就会停止。
【操作】测试zip()并行迭代
for i in [1,2,3]:
print(i)
names = ("高淇","高老二","高老三","高老四")
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]))
运行结果:
1
2
3
高淇--18--老师
高老二--16--程序员
高老三--20--公务员
高淇--18--老师
高老二--16--程序员
高老三--20--公务员
推导式创建序列
推导式是从一个或者多个迭代器快速创建序列的一种方法。它可以将循环和条件判断结合,从而避免冗长的代码。推导式是典型的Python 风格,会使用它代表你已经超过Python 初学者的水平。
列表推导式
字典推导式
集合推导式
生成器推导式(生成元组)
很多同学可能会问:“都有推导式,元组有没有?”,能不能用小括号呢?
>>> (x for x in range(1,100) if x%9==0)
<generator object <genexpr> at 0x0000000002BD3048>
我们发现提示的是“一个生成器对象”。显然,元组是没有推导式的。一个生成器只能运行一次。第一次迭代可以得到数据,第二次迭代发现数据已经没有了。
操作:
#测试推导式
#列表推导式: 生成列表对象,可以加if条件判断
y=[x*2 for x in range(1,5)]
print(y)
y=[]
for x in range(1,5):
y.append(x*2)
print(y)
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,10) for col in range(1,10)] #可以使用两个循环
for cell in cells:
print(cells)
#字典推导式:统计文本中字符出现的次数:
my_text="i love you, i love sxt, i love gaoqi"
char_count={c:my_text.count(c) for c in my_text}
print(char_count)
#课下作业,使用普通循环实现上面的字典推导式统计文本中字符出现的次数
my_text="i love you, i love sxt, i love gaoqi"
k=[]
v=[]
for c in my_text:
k.append(c)
v.append(my_text.count(c))
char_count= dict(zip(k,v))
print(char_count)
#集合推导式:生成集合
b={x for x in range(1,100) if x%9==0}
print(b)
#生成器推导式:
gnt=(x for x in range(4))
print(tuple(gnt))
gnt=(t for t in range(4))
for t in gnt: #gnt是生成器对象,生成器是可迭代的对象,只能使用一次
print(t,end=",")
print(tuple(gnt))
综合练习
- 【操作】绘制多个同心圆
import turtle
t=turtle.Pen()
my_colors=('red','green','yellow','black')
t.speed(0.5)
t.width(4)
for i in range(10):
t.penup()
t.goto(0, -i*20)
t.pendown()
t.color(my_colors[i%len(my_colors)])
t.circle(30+i*20)
turtle.done() #程序执行完,窗口仍然在
运行结果:
- 【操作】绘制18*18 棋盘
import turtle
t=turtle.Pen()
t.speed(5)
for i in range(18): #0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17
t.penup()
t.goto(20*i,0)
t.pendown()
t.goto(20*i,170*2)
for i in range(18): #0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17
t.penup()
t.goto(0,20*i)
t.pendown()
t.goto(170*2,20*i)
t.hideturtle() #隐藏画笔
turtle.done() #窗口运行后不自动关闭
运行结果:
第五章:函数用法和底层分析
函数是可重用的程序代码块。函数的作用,不仅可以实现代码的复用,更能实现代码的一致性。一致性指的是,只要修改函数的代码,则所有调用该函数的地方都能得到体现。
在编写函数时,函数体中的代码写法和我们前面讲述的基本一致,只是对代码实现了封装,并增加了函数调用、传递参数、返回计算结果等内容。
为了让大家更容易理解,掌握的更深刻。我们也要深入内存底层进行分析。绝大多数语言内存底层都是高度相似的,这样大家掌握了这些内容也便于以后学习其他语言。
函数简介
函数的基本概念
- 一个程序由一个个任务组成;函数就是代表一个任务或者一个功能。
- 函数是代码复用的通用机制。
Python 函数的分类
Python 中函数分为如下几类:
- 内置函数
我们前面使用的str()、list()、len()等这些都是内置函数,我们可以拿来直接使用。 - 标准库函数
我们可以通过import 语句导入库,然后使用其中定义的函数 - 第三方库函数
Python 社区也提供了很多高质量的库。下载安装这些库后,也是通过import 语句导入,然后可以使用这些第三方库的函数 - 用户自定义函数
用户自己定义的函数,显然也是开发中适应用户自身需求定义的函数。今天我们学习的就是如何自定义函数。
函数的定义和调用
核心要点
Python 中,定义函数的语法如下:
def 函数名([参数列表]) :
‘’‘文档字符串’’’
函数体/若干语句
要点:
- 我们使用def 来定义函数,然后就是一个空格和函数名称;
(1) Python 执行def 时,会创建一个函数对象,并绑定到函数名变量上。 - 参数列表
(1) 圆括号内是形式参数列表,有多个参数则使用逗号隔开
(2) 形式参数不需要声明类型,也不需要指定函数返回值类型
(3) 无参数,也必须保留空的圆括号
(4) 实参列表必须与形参列表一一对应 - return 返回值
(1) 如果函数体中包含return 语句,则结束函数执行并返回值;
(2) 如果函数体中不包含return 语句,则返回None 值。 - 调用函数之前,必须要先定义函数,即先调用def 创建函数对象
(1) 内置函数对象会自动创建
(2) 标准库和第三方库函数,通过import 导入模块时,会执行模块中的def 语句
#测试函数的定义和调用
def test01():
print("*"*10)
print("@"*10)
test01()
for i in range(3):
test01()
print(id(test01()))
print(type(test01))
print(test01)
运行结果:
**********
@@@@@@@@@@
**********
@@@@@@@@@@
**********
@@@@@@@@@@
**********
@@@@@@@@@@
**********
@@@@@@@@@@
140715927297248
<class 'function'>
<function test01 at 0x000002091421D798>
形参和实参
【操作】定义一个函数,实现两个数的比较,并返回较大的值。
def printMax(a,b):
'''用于比较两个数的大小,打印较大值'''
if a>b:
print(a,"较大值")
else:
print(b,"较大值")
printMax(100,200)
printMax(10,30)
help(printMax.__doc__)
运行结果:
200 较大值
30 较大值
No Python documentation found for '用于比较两个数的大小,打印较大值'.
Use help() to get the interactive help utility.
Use help(str) for help on the str class.
上面的printMax 函数中,在定义时写的printMax(a,b)。a 和b
称为“形式参数”,简称“形参”。也就是说,形式参数是在定义函数时使用的。形式参数的命名只要符合“标识符”命名规则即可。
在调用函数时,传递的参数称为“实际参数”,简称“实参”。上面代码中, printMax(10,30),10 和30就是实际参数。
文档字符串(函数的注释)
程序的可读性最重要,一般建议在函数体开始的部分附上函数定义说明,这就是“文档字符串”,也有人成为“函数的注释”。我们通过三个单引号或者三个双引号来实现,中间可以加入多行文字进行说明。
返回值
return 返回值要点:
- 如果函数体中包含return 语句,则结束函数执行并返回值;
- 如果函数体中不包含return 语句,则返回None 值。
- 要返回多个返回值,使用列表、元组、字典、集合将多个值“存起来”即可。
操作:
#函数返回值的基本用法
def add(a,b):
print("计算两个数的和:{0}+{1}={2}".format(a,b,a+b))
return a+b
def test02():
print("sxt")
print("gao")
return # return两个作用 1.返回值2.结束函数的执行
print("hello") #因为return, 该语句不执行
def test03(x,y,z):
return[x*10,y*10,z*10]
c=add(30,40)
print(c)
d=test02()
print(d)
print(test03(4,6,7))
运行结果:
计算两个数的和:30+40=70
70
sxt
gao
None
[40, 60, 70]