Python(5)循环结构、推导式创建序列、函数

循环结构

循环结构用来重复执行一条或多条语句。表达这样的逻辑:如果符合条件,则反 复执行循环体里的语句。在每次执行完后都会判断一次条件是否为 True,如果 为 True 则重复执行循环体里的语句。

循环体里面的语句至少应该包含改变条件表达式的语句,以使循环趋于结束;否则,就会变成一个死循环。

while 循环

while 循环的语法格式如下:
while 条件表达式:
循环体语句

num = 0
while num<=10:
    print(num,end="\t")
    num += 1

for 循环和可迭代对象遍历

for 循环通常用于可迭代对象的遍历。for 循环的语法格式如下:
for 变量 in 可迭代对象:
循环体语句

Python 包含以下几种可迭代对象:

  1. 序列。包含:字符串、列表、元组
  2. 字典
  3. 迭代器对象(iterator)
  4. 生成器函数(generator)
  5. 文件对象
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
'''

其他优化手段

  1. 连接多个字符串,使用 join()而不使用+
  2. 列表进行元素插入和删除,尽量在列表尾部操作
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 中函数分为如下几类:

  1. 内置函数 我们前面使用的 str()、list()、len()等这些都是内置函数,我们可以拿来直接使用。
  2. 标准库函数 我们可以通过 import 语句导入库,然后使用其中定义的函数
  3. 第三方库函数 Python 社区也提供了很多高质量的库。下载安装这些库后,也是通过 import 语句导 入,然后可以使用这些第三方库的函数
  4. 用户自定义函数 用户自己定义的函数,显然也是开发中适应用户自身需求定义的函数。

函数的定义和调用

Python 中,定义函数的语法如下:
def 函数名 ([参数列表]) :
‘’‘文档字符串’’’#其实就是函数的注解说明
函数体/若干语句
要点:

  1. 我们使用 def 来定义函数,然后就是一个空格和函数名称;Python 执行 def 时,会创建一个函数对象,并绑定到函数名变量上。
  2. 参数列表 :(1) 圆括号内是形式参数列表,有多个参数则使用逗号隔开 (2) 形式参数不需要声明类型,也不需要指定函数返回值类型 (3) 无参数,也必须保留空的圆括号 (4) 实参列表必须与形参列表一一对应
  3. return 返回值: (1) 如果函数体中包含 return 语句,则结束函数执行并返回值; (2) 如果函数体中不包含 return 语句,则返回 None 值。(3)return 语句就是将结果返回到调用的地方,并把程序的控制权一起返回,程序运行到所遇到的第一个return即返回(退出def块),不会再运行第二个return。
  4. 调用函数之前,必须要先定义函数,即先调用 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 返回值要点:

  1. 如果函数体中包含 return 语句,则结束函数执行并返回值;
  2. 如果函数体中不包含 return 语句,则返回 None 值。
  3. 要返回多个返回值,使用列表、元组、字典、集合将多个值“存起来”即可。
  4. 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
'''
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值