Day 5

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

for 变量 in 可迭代对象:

循环体语句

【操作】遍历一个元组或列表   

#测试for循环
for x in (10,20,30):
    print(x,end='\t')

结果如下:

10 20 30

可迭代对象

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

  1. 序列。包含:字符串、列表、元组
  2. 字典
  3. 迭代器对象(iteratior)
  4. 生成器函数(generator)
  5. 文件对象

【操作】遍历字符串中的字符

for y in "abcdef":
    print(y,end='\t')

结果:a  b c d e f

【操作】遍历字典

#字典的遍历
d={'name':'zhang','age':18,'job':'程序员'}
for x in d:   #遍历字典所有的key
    print(x,end='\n')
for x in d.keys():  #遍历字典所有的key
    print(x,end='\t')
for x in d.values():  #遍历字典所有的value
    print(x, end='\t')
for x in d.items():  #遍历字典所有的键值对
    print(x,end='\t')

结果如下:

name

age

job

name age job zhang 18 程序员

('name', 'zhang') ('age', 18) ('job', '程序员')

range对象

range对象是一个迭代器对象,用来产生指定范围的数字序列。格式为:

                     Range(start,end ,[step])

生成的数值序列从start开始到end结束(不包含end)。若没有填写start,则默认从0开始。Step是可选的步长,默认为1.以下是几种示例

for i in range(10)  产生序列:0 1 2 3 4 5 6 7 8 9

for i in range(3,10)  产生序列:3 4 5 6 7 8 9

for i in range(3,10,2)  产生序列:3 5 7 9

【操作】利用for循环,计算1-100之间数字的累加和;计算1-100之间偶数的累加和,计算1-100之间奇数的累加和

sum_all = 0  #1-100所有数累加和
sum_even = 0 #1-100偶数的累加和
sum_odd = 0 #1-100奇数的累加和

for num in range(101):
    sum_all += num
    if num%2 == 0:
        sum_even +=num
    else:
        sum_odd +=num

print('1-100所有数累加和',sum_all)
print('1-100所有偶数累加和',sum_even)
print('1-100所有奇数累加和',sum_odd)

结果:

1-100所有数累加和 5050

1-100所有偶数累加和 2550

1-100所有奇数累加和 2500

嵌套循环

一个循环体内可以嵌入另一个循环,成为“嵌套循环”

【操作】打印下列图案

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('\n') #起到换行的作用

结果:

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

【操作】打印九九乘法表

  1. 文件对象

【操作】遍历字符串中的字符

for y in "abcdef":
    print(y,end='\t')

结果:a  b c d e f

【操作】遍历字典

#字典的遍历
d={'name':'zhang','age':18,'job':'程序员'}
for x in d:   #遍历字典所有的key
    print(x,end='\n')
for x in d.keys():  #遍历字典所有的key
    print(x,end='\t')
for x in d.values():  #遍历字典所有的value
    print(x, end='\t')
for x in d.items():  #遍历字典所有的键值对
    print(x,end='\t')

结果如下:

name

age

job

name age job zhang 18 程序员

('name', 'zhang') ('age', 18) ('job', '程序员')

range对象

range对象是一个迭代器对象,用来产生指定范围的数字序列。格式为:

                     Range(start,end ,[step])

生成的数值序列从start开始到end结束(不包含end)。若没有填写start,则默认从0开始。Step是可选的步长,默认为1.以下是几种示例

for i in range(10)  产生序列:0 1 2 3 4 5 6 7 8 9

for i in range(3,10)  产生序列:3 4 5 6 7 8 9

for i in range(3,10,2)  产生序列:3 5 7 9

【操作】利用for循环,计算1-100之间数字的累加和;计算1-100之间偶数的累加和,计算1-100之间奇数的累加和

um_all = 0  #1-100所有数累加和
sum_even = 0 #1-100偶数的累加和
sum_odd = 0 #1-100奇数的累加和

for num in range(101):
    sum_all += num
    if num%2 == 0:
        sum_even +=num
    else:
        sum_odd +=num

print('1-100所有数累加和',sum_all)
print('1-100所有偶数累加和',sum_even)
print('1-100所有奇数累加和',sum_odd)

结果:

1-100所有数累加和 5050

1-100所有偶数累加和 2550

1-100所有奇数累加和 2500

嵌套循环

一个循环体内可以嵌入另一个循环,成为“嵌套循环”

【操作】打印下列图案

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('\n') #起到换行的作用

结果:

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 m in range(1,10):
    for n in range(1,m+1):
        print('{0}*{1}={2}'.format(m,n,(m*n)),end='\t')
    print('\n')

结果:

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的数据

姓名

年龄

薪资

城市

高小一

18

30000

北京

高小二

19

20000

上海

高小山

20

10000

深圳

#使用列表和字典存储表格的数据
r1= {'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)

结果:

请输入一个字符(输入Q或者q时推出):d

d

请输入一个字符(输入Q或者q时推出):a

a

请输入一个字符(输入Q或者q时推出):q

循环结束,退出

continue语句

continue语句用于结束本次循环,继续下一次。多个循环嵌套时,continue也是用于最近的一层循环

【操作】要求输入员工的薪资,若薪资小于0则重新输入。最后打印出录入员工的数量和薪资明细,以及平均薪资

#测试break
empNum=0
salarySum=0
salary=[] #列表表示(因为列表是可变序列)

while True:
    s = input('请输入员工的薪资(按Q或q退出)')

    if s.upper()=='Q' or s.upper()=='q':
        print('录入完成,退出')
        break
    if float(s)<0:
        continue
    empNum +=1
    salary.append(float(s))
    salarySum+=float(s)

print('员工数{0}'.format(empNum))
print('录入薪资:',salary)
print('平均薪资{0}'.format(salarySum/empNum))

结果:

请输入员工的薪资(按Q或q退出)1000

请输入员工的薪资(按Q或q退出)2000

请输入员工的薪资(按Q或q退出)5000

请输入员工的薪资(按Q或q退出)3500

请输入员工的薪资(按Q或q退出)q

录入完成,退出

员工数4

录入薪资: [1000.0, 2000.0, 5000.0, 3500.0]

平均薪资2875.0

else语句

while、for循环可以附带一个else语句(可选)。如果for、while语句没有被break语句结束,则会执行else子句,否则不执行。语法格式如下:

while 条件表达式

循环体

else:

语句块

或者:

for 变量 in 可迭代对象:

循环体

else:

语句块

【操作】员工一共四人,录入这4位员工的薪资,全部录入后,打印提示“您已经全部录入4名员工的薪资”。最后,打印输出录入的薪资和平均水平

salarySum = 0
salaries =[]
for i in range(4):
    s =input(('请输入4名员工的薪资(按Q或q中途退出)'))

    if s.upper()=='Q' or s.upper()=='q':
        print('录入完成,退出')
        break
    if float(s)<0:
        continue

    salaries.append(float(s))
    salarySum+=float(s)

else:
    print('您已经全部录入4名员工的薪资')

print('录入薪资',salaries)
print('平均薪资:',salarySum/4)

结果:

请输入4名员工的薪资(按Q或q退出)5000

请输入4名员工的薪资(按Q或q退出)4000

请输入4名员工的薪资(按Q或q退出)3600

请输入4名员工的薪资(按Q或q退出)2900

您已经全部录入4名员工的薪资

录入薪资 [5000.0, 4000.0, 3600.0, 2900.0]

平均薪资: 3875.0

循环代码优化

虽然计算机越来越快,空间也越来越大,我们仍然要在性能问题上“斤斤计较”。

编写循环时,遵循下面三个原则能大大提高运行效率,避免不必要的低效计算:

  1. 尽量减少循环内部不必要的计算
  2. 嵌套循环中,尽量减少内层循环的计算,尽可能向外提
  3. 局部变量查询较快,尽量使用局部变量

示例:

#循环代码优化测试
import time
#方法1:
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))

#方法2:把i乘1000提到内循环外面
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)))

结果:

耗时1.911698341369629

耗时:1.7906055450439453

其他优化手段

  1. 连接多个字符串,使用join()而不适用‘+’(因为+会产生新的字符串对象)
  2. 列表进行元素插入和删除,尽量在列表尾部操作(中间操作的话,后面所有数据都需要重新移位拷贝,浪费时间)

使用zip()进行迭代

我们可以通过zip()函数对多个序列进行并行迭代,zip()函数在最短序列“用完”时就会停止

【操作】测试zip()并进行迭代

names=('张一','张二','张三','张四')
ages=(18,20,30,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]))

结果:

张一--18--老师

张二--20--程序员

张三--30--医生

张一--18--老师

张二--20--程序员

张三--30--医生

推导式创建序列

推导式是从一个或者多个迭代器快速创建序列的一种方法。它可以将循环和条件判断相结合,从而避免冗长的代码。推导式是典型的Python风格,会使用它代表你已经超过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)

结果如下:

[10, 20, 30, 40, 50, 60, 70, 80, 90]

[10, 20, 30, 40, 50, 60, 70, 80, 90]

cells = [(row,col) for row in range(1,10) for col in range(1,10)]
print(cells)

结果:

[(1, 1), (1, 2), (1, 3), (1, 4), (1, 5), (1, 6), (1, 7), (1, 8), (1, 9), (2, 1), (2, 2), (2, 3), (2, 4), (2, 5), (2, 6), (2, 7), (2, 8), (2, 9), (3, 1), (3, 2), (3, 3), (3, 4), (3, 5), (3, 6), (3, 7), (3, 8), (3, 9), (4, 1), (4, 2), (4, 3), (4, 4), (4, 5), (4, 6), (4, 7), (4, 8), (4, 9), (5, 1), (5, 2), (5, 3), (5, 4), (5, 5), (5, 6), (5, 7), (5, 8), (5, 9), (6, 1), (6, 2), (6, 3), (6, 4), (6, 5), (6, 6), (6, 7), (6, 8), (6, 9), (7, 1), (7, 2), (7, 3), (7, 4), (7, 5), (7, 6), (7, 7), (7, 8), (7, 9), (8, 1), (8, 2), (8, 3), (8, 4), (8, 5), (8, 6), (8, 7), (8, 8), (8, 9), (9, 1), (9, 2), (9, 3), (9, 4), (9, 5), (9, 6), (9, 7), (9, 8), (9, 9)]

字典推导式

字典的推导式生成字典对象,格式如下:

     {key_expression : value_expression  for 表达式 in 可迭代对象}

类似于列表推导式,字典推导也可以增加if条件判断、多个for循环

统计文本中字符出现的次数:

#字典推导式
my_text='i love you,i love tx,i love zzm'
#定义统计列表,和字典推导式 key是c,value是my_text.count(c)统计出现的次数
char_count = {c:my_text.count(c) for c in my_text}
print(char_count)

结果:

{'i': 3, ' ': 6, 'l': 3, 'o': 4, 'v': 3, 'e': 3, 'y': 1, 'u': 1, ',': 2, 't': 1, 'x': 1, 'z': 2, 'm': 1}

集合推导式

集合推导式生成集合,和列表推导式的语法格式类似:

                 {表达式 for  item   in 可迭代对象}

          或者:{表达式   for   item  in 可迭代对象   if  条件判断}

#集合推导式
b ={x for x in range(1,100) if x%9==0}
print(b)

结果:

{99, 36, 72, 9, 45, 81, 18, 54, 90, 27, 63}

生成器推导式(生成元组)

a=(x for x in range(1,100) if x%9 ==0)
print(a)

结果:

<generator object <genexpr> at 0x000002D5C24AB248>

我们发现提示的是“一个生成器对象”。显然,元组是没有推导式的

一个生成器只能运行一次。第一次迭代可以得到数据,第二次迭代数据发现数据已经没有了

#生成器推导式(生成元组)
gnt = (x for x in range(1,4))
print(tuple(gnt))  #tuple()是生成元组
print(tuple(gnt))  #第二次遍历就看不了了

gnt1 = (x for x in range(1,4))
for x in gnt1:   #gnt1是生成器对象,生成器是一个可迭代的对象
    print(x,end=',')
print(tuple(gnt1))

结果如下:

(1, 2, 3)

()

1,2,3,()

【绘制】不同颜色的多个同心圆

#绘制不同颜色的多个同心圆
import turtle

t = turtle.Pen()
my_colors = ('red','green','blue','purple','yellow')
t.width(4)
t.speed(10)  #0是最快,1是最慢的,5,10都比较快

for i in range(10):  #0,1,2,3,4,5,6,7,8,9
    t.penup()
    t.goto(0,-i*10)  #0,-10,-20,-30,-40...-90
    t.pendown()
    t.color(my_colors[i%len(my_colors)])
    t.circle(10*(i+1))  #10,20,30,40,50..90

turtle.done()  #程序执行完,窗口仍然在

结果:

Python如何输出Javaobject的遍历 python 遍历对象_迭代

 【绘制】棋盘

Python如何输出Javaobject的遍历 python 遍历对象_python_02

import  turtle
#获取画笔
t=turtle.Pen()

#声明棋盘的单元格宽度为30,数量为18
width = 20
num = 200 // 20*2+1  #//整数除法
#200//20=10*2+1 =21
t.speed(0)


#声明列表来存取坐标

#循环绘制横向线
for i in range(num):
    t.penup()
    t.goto(-200,200-width*i)
    t.pendown()
    t.goto(200,200-width*i)
#循环绘制纵向线
for i in range(num):
    t.penup()
    t.goto(-200+i*width,200)
    t.pendown()
    t.goto(-200+i*width,-200)

turtle.done()

结果:

Python如何输出Javaobject的遍历 python 遍历对象_python_03

函数用法和底层分析

函数是可重用的程序代码块。函数的作用,不仅可以实现代码的复用,更能实现代码的一致性。一致性指的是,只要修改函数的代码,则所有调用该函数的地方都得到体现。

在编写函数时,函数体中的代码写法和我们前面讲述的基本一致,只是对代码实现了封装,并增加了函数调用,传递参数、返回计算结果等内容。

为了让大家更容易理解,掌握的更深刻。我们也要深入内存进行分析。绝大多数语言内存底层都是高度相似的,也方便学习其他的语言

函数简介

函数的基本功能

  1. 一个程序由一个个任务组成,函数就是代表一个任务或者一个功能
  2. 函数是代码复用的通用机制

Python函数的分类

Python中函数分为如下几类:

  1. 内置函数

   我们前面使用的wtr(),list(),len()等这些都是内置函数,可以拿来直接使用

     2.标准库函数

   我们可以通过import语句导入库,然后使用其中定义的函数

     3.第三方库函数

   Python社区也提供了很多高质量的库。下载安装这些库后,也是通过import语句导入,然后可以使用这些第三方库的函数

     4.用户自己定于函数

   用户自己定义的函数,显然也是开发中适应用户自身需求定义的函数。今天我们学习的就是如何自己定义函数

函数的定义和调用

核心要点

Python中,定义函数的语法如下:

def 函数名([参数列表]):

  ‘’’文档字符串’’’

  函数体/若干语句

要点:;

  1. 我们使用def来定义函数,然后就是一个空格和函数名称:                                                      (1)Python执行def时,会创建一个函数对象(函数也是对象),并绑定到函数名变量上
  2. 参数列表                                                                                                                           (1)圆括号内是形式参数列表,有多个参数则使用逗号隔开(2)形式参数不需要声明类型,也不需要指定函数的返回值类型(3)无参数,也必须保留空的圆括号 (4)实参列表必须与形参列表一一对应
  3. return返回值                                                                                                                     (1)如果函数体中包含return语句,则结束函数执行并返回值(2)如果函数体中不包含return语句,则返回None值
  4. 调用函数之前,必须要先定义函数,即先调用def创建函数对象                                          (1)内置函数对象会自动创建(2)标准库和第三方库函数,通过import导入模块时,会执行模块中的def语句

形参和实参

【操作】定义一个函数,实现两个数的比较,并返回较大的值

#测试形参、实参的基本用法

def printMax(a,b):
    if a>b:
        print(a,'较大值')
    else:
        print(b,'较大值')

printMax(10,20)
printMax(200,300)

20 较大值

300 较大值

上面的printMax函数中,在定义时写的printMax(a,b)。a和b称为“形式参数”,简称“形参”。也就是说,形式参数是在定义函数时使用的。形式参数的命名只要符合“标识符”命名规律即可

     在调用函数中时,传递的参数称为“实际参数”,简称“实参”。上述代码中,printMax(10,20),10和20就是实际参数

文档字符串(函数的注释)

程序的可读性最重要,一般建议在函数体开始的部分附上函数定义说明,这就是“文档字符串”,也有人成为“函数的注释”。我们通过三个单引号或者三个双引号来实现,中间可以加入多行文字进行说明

【操作】测试文档字符串的使用

def print_star(n):
    '''根据传入n,打印多个星号'''
    print('*'*n)

help(print_star)

我们调用help(函数名.__doc__)可以打印出函数的文档字符串。执行结果如下:

结果:

Help on function print_star in module __main__:

print_star(n)

    根据传入n,打印多个星号

返回值

return返回值要点:

  1. 如果函数体中包含return语句,则结束函数执行并返回值
  2. 如果函数体中不包含return语句,则返回None值
  3. 要返回多个返回值,使用列表、元组、字典、集合将多个值“存起来”即可

【操作】定义一个打印N个星号的无返回值的函数

def print_star(n):
    '''根据传入n,打印多个星号'''
    print('*'*n)

print_star(5)

结果:

*****

【操作】定义一个返回两个数平均值的函数

def my_avg(a,b):
    return (a+b)/2

#如下是函数的调用
c = my_avg(20,30)
print(c)

结果:

25.0

def test02():
    print('abc')
    print("edf")

    return #return的两个作用:1.返回值  2.结束函数的执行
    print('hello') #不执行

d = test02()
print(d)

结果:

abc

edf

None

返回多个参数:

def test03(x,y,z):
    return [x*10,y*10,z*10]

print(test03(3,4,2))

结果:[30, 40, 20]