python2基础难点_python 基础知识点二

深浅copy

1对于赋值运算来说,l1与l2指向的是同一个内存地址,所以他们是完全一样的。

l1 = [1,2,3,['barry','alex']]

l2 = l1

l1[0] = 111

print(l1,id(l1)) # [111, 2, 3, ['barry', 'alex']] 112431152

print(l2,id(l2)) # [111, 2, 3, ['barry', 'alex']] 112431152

l1[3][0] = 'wusir'

print(l1,id(l1)) # [1, 2, 3, ['wusir', 'alex']] 112431152

print(l2,id(l2)) # [1, 2, 3, ['wusir', 'alex']] 112431152

2对于浅copy(copy)来说,只是在内存中重新创建了开辟了一个空间存放一个新列表(内存地址不同),列表中的元素的内存地址是一样的。

列表中可变的数据类型修改就会跟着修改,但是列表中的不可变的数据类型修改就不会跟着修改。

解释:copy.copy 浅拷贝 只拷贝父对象,不会拷贝对象的内部的子对象。子对象(数组,字典等可变数据类型)修改,也会修改

l1 = [1, '太白', True, (1,2,3), [22, 33]]

l2 = l1.copy()

print(id(l1), id(l2)) # 112496688 112475312

print(id(l1[-2]), id(l2[-2])) #35243432 35243432

print(id(l1[-1]),id(l2[-1])) #112410096 112410096

3.对于深copy(deepcopy)来说,列表是在内存中重新创建的,是原始的对象。不会随着修改而修改。

copy.deepcopy 深拷贝 拷贝对象及其子对象(原始对象)

import copy

l1 = [1, 'alex', True, (1,2,3), [22, 33]]

l2 = copy.deepcopy(l1)

# print(id(l1), id(l2)) # 112627760 112606384

print(id(l1[0]),id(l2[0])) # 1521669168 1521669168

print(id(l1[-1]),id(l2[-1])) # 112999960 112999880 #元素为列表 为不可哈希(可变类型)

print(id(l1[-2]),id(l2[-2])) # 6735272 6735272

例子:

import copy

a = [1, 2, [3, 4], {'a': 1}] # 原始对象

b = a # 赋值,传对象的引用

c = copy.copy(a) # 对象拷贝,浅拷贝

d = copy.deepcopy(a) # 对象拷贝,深拷贝

e = a[:] # 能复制序列,浅拷贝

a.append('add1')

# 修改对象a

a[2].append('add2')

# 修改对象a中的[3,4]数组对象

a[3]['a'] = 666

print('a:', a)

print('b:', b)

print('c:', c)

print('d:', d)

print('e:', e)

结果;

a: [1, 2, [3, 4, 'add2'], {'a': 666}, 'add1']

b: [1, 2, [3, 4, 'add2'], {'a': 666}, 'add1']

c: [1, 2, [3, 4, 'add2'], {'a': 666}]

d: [1, 2, [3, 4], {'a': 1}]

e: [1, 2, [3, 4, 'add2'], {'a': 666}]

解释:copy.copy 浅拷贝 只拷贝父对象,不会拷贝对象的内部的子对象。子对象(数组)修改,也会修改

copy.deepcopy 深拷贝 拷贝对象及其子对象(原始对象)

文件的操作

文件句柄 = open(‘文件路径’,‘模式’,'编码')

#1. 打开文件的模式有(默认为文本模式):

r ,只读模式【默认模式,文件必须存在,不存在则抛出异常】 #注意读是读光标后的内容

w,只写模式【不可读;不存在则创建;存在则清空内容】

a, 只追加写模式【不可读;不存在则创建;存在则只追加内容】

#2. 对于非文本文件,我们只能使用b模式,"b"表示以字节的方式操作(而所有文件也都是以字节的形式存储的,使用这种模式无需考虑文本文件的字符编码、

图片文件的jgp格式、视频文件的avi格式)

rb

wb

ab

注:以b方式打开时,读取到的内容是字节类型,写入时也需要提供字节类型,不能指定编码

#3,‘+’模式(就是增加了一个功能)

r+, 读写【可读,可写】

w+,写读【可写,可读】

a+, 写读【可写,可读】

#4,以bytes类型操作的读写,写读,写读模式

r+b, 读写【可读,可写】

w+b,写读【可写,可读】

a+b, 写读【可写,可读】

#在读的时候,bytes类型在读的时候内置函数自动转变为字符串了

# with open('01.txt','r',encoding='UTF-8') as f:

# # f.write('曾辉123456')

# print(f.read()) bytes ---->str

read:

1. 文件打开方式为文本模式时,代表读取3个字符

2. 文件打开方式为b模式时,代表读取3个字节

其余的文件内光标移动都是以字节为单位的如:seek,tell,truncate

注意:

1. seek有三种移动方式0,1,2,其中1和2必须在b模式下进行,但无论哪种模式,都是以bytes为单位移动的

2. truncate是截断文件,所以文件的打开方式必须可写,但是不能用w或w+等方式打开,因为那样直接清空文件了,所以truncate要在r+或a或a+等模式下使用。

with open('log',mode='a+',encoding='utf-8') as f:

# f.write('佳琪')

count = f.tell() #tell 是获取光标的位置,也是以一个字节为单位的。

f.seek(0) #seek 光标的移动是以一个字节为单位的

print(f.read()) #读取光标后的文本

f.read(2) #读取几个字符

f.truncate(5) #截断原文件,对原文件进行变化,

print(f.readline()) #只读取文件光标后的一行内容

coun = f.readlines() #readlines读取文件的每行内容,并且把每行内容

#作为List中的元素储存起来。

f.writelines(['1','23','3']) #将一个字符串列表(列表中的元素为字符串)写入文件

with open('log',mode='a+',encoding='utf-8') as f,\

open('log',mode='a+',encoding='utf-8') as f1,\

open('log',mode='a+',encoding='utf-8') as f2: #用 with 可以同时打开多个文件,

#同时不用在最后关闭文件

文件的修改,实际上文件不能直接进行修改,可以通过间接修改。修改的过程为:先打开2个文件,然后在写入新的内容放在一个文件里面,然后在删除原来的文件,在将修改内容的文件重命名为原来的文件名。

#函数的文件修改

def txt_func(fliename,old,new): #打开文件

with open(fliename,encoding='utf-8') as f,open('%s.bak'%(fliename),'w',encoding='utf-8') as f2:

for line in f:

if old in line:

line = line.replace(old,new)

#写文件

f2.write(line)

import os

os.remove(fliename) #删除文件

os.rename('%s.bak'%(fliename),fliename) #重命名文件

txt_func('123.txt','123','aaa')

函数的返回值

函数:可读性强  复用性强

return 关键字的作用:

1,结束函数的执行(return 下面的语句都不执行)

2,返回值

没有返回值。

不写return的情况下,会默认返回一个None;

有返回值包括一个值或者是多个值。

def fun(a,b):

c= a+b

return c,a

sun,b = fun(1,1) #返回多个值,如果用一个变量去接收,变量代表一个元组;

sun1 = fun(1,1)

print(sun1) #用相对应的个数的变量去接收,每一个变量代表一个值.

print(sun,b)

函数的参数

实参:调用函数传递的参数(实际的参数);     形参:定义函数时传递的参数;

站在实参的角度上:

1,按照位置传参;

2, 按照关键字传参

#位置传参必须在关键字传参之前;

形参里面的:

默认参数(不可变数据的类型);

#参数陷阱:默认参数是一个可变数据的类型;

# 如果默认参数的值是一个可变数据类型,

# 那么每一次调用函数的时候,

# 如果不传值就公用这个数据类型的资源

def defult_param(a,l = []):

l.append(a)

print(l)

defult_param('alex')

defult_param('egon') #每次调用公用一个列表

#['alex'] #默认参数l里面只一个元素

#['alex', 'egon'] #l里面变成2个元素来;

站在形参的角度上:

1,按照位置传参;

2,按照动态传参(*args)

3,按照关键字传参(默认参数);

4,**kwargs

所有参数的传参的顺序是 位置参数,*args,关键字参数,**kwargs。

#参数的位置: 位置参数 ,*args, 默认参数,**kargs

#*args :接收多个位置参数,组织成一个元组

#**kwargs可以接收多个关键字参数,组成成一个字典

# def func(*args ,**kwargs): #站在形参的角度上(定义函数的时候),给变量加上*;**,就是组合所有传来的值.

# print (args,kwargs)

# # return ('{},{}'.format(name,age))

# l = [1,2,3,'abc1']

# # l='12331231'

# a={'name':'zenghui','age':'22'}

#

# # func(1,2,3,name='zenghui',age = '22'

# func(*l,**a) #**就是将a的字典按照位置的顺序打散;

#站在实参的角度上(调用函数的时候),给一个序列加上*;**,就是将这个序列按照顺利打散.

函数的注释

def fun(a,b):

'''

函数的功能

param a(参数):

param b(参数):

return(返回值):

'''

c= a+b #函数的主体

return c,a

参数的总结:

7af52dcee755d44d300f9fb802c2f530.png

命名空间和作用域python

python代码运行的时候碰到函数的运行过程:

第一步是python解释器先执行,然后在内存中开辟了一个空间,每当遇到一个变量的时候,就把变量名和值之间的对应的关系记录下来。但是当遇到函数的定义的时候,解释器只是将函数名读入内存中,表示这个函数的存在。但是不关心函数内部的变量以及名字。等到执行函数调用的时候,python解释器会再开辟一块内存开存储这个函数里面的内容,这个时候才关注函数里面有哪些变量,而函数中的变量会存储在新开辟出来的内存中。函数中的变量只能在函数的内部使用,并且会随着函数执行完毕,这块内存中的所有内容也会被清空。

我们给这个“存放名字与值的关系”的空间起了一个名字——叫做命名空间

代码在运行开始,创建的存储“变量名与值的关系”的空间叫做全局命名空间,在函数的运行中开辟的临时的空间叫做局部命名空间

有三种命名空间;分别为内置的命名空间;全局的命名空间;局部的命名空间;

#内置命名空间 —— python解释器

就是python解释器一启动就可以使用的名字存储在内置命名空间中

内置的名字在启动解释器的时候被加载进内存里

#全局命名空间 —— 我们写的代码但不是函数中的代码

是在程序从上到下被执行的过程中依次加载进内存的

放置了我们设置的所有变量名和函数名

#局部命名空间 —— 函数

就是函数内部定义的名字

当调用函数的时候 才会产生这个名称空间 随着函数执行的结束 这个命名空间就又消失了

在局部:可以使用全局、内置命名空间中的名字

在全局:可以使用内置命名空间中的名字,但是不能用局部中使用

在内置:不能使用局部和全局的名字的

在正常情况下,直接使用内置的名字

当我们在全局定义了和内置名字空间中同名的名字时,会使用全局的名字

当我自己有的时候 我就不找我的上级要了

如果自己没有 就找上一级要 上一级没有再找上一级 如果内置的名字空间都没有 就报错

多个函数应该拥有多个独立的局部名字空间,不互相共享

三种命名空间之间的加载与取值顺序:

加载顺序:内置命名空间(程序运行前加载)->全局命名空间(程序运行中:从上到下加载)->局部命名空间(程序运行中:调用时才加载)

取值:

从底层开始往上寻找并且取值;

在局部调用的时候: 局部命名空间————全局命名空间————内置的命名空间

在全局调用的时间:全局命名空间————内置的命名空间

作用域

作用域就是作用范围,按照生效范围可以分为全局作用域和局部作用域。

全局作用域:包含内置名称空间、全局名称空间,在整个文件的任意位置都能被引用、全局有效

局部作用域:局部名称空间,只能在局部范围内生效

global 关键字:声明全局变量

对于不可变数据类型 在局部可是查看全局作用域中的变量

但是不能直接修改

如果想要修改,需要在程序的一开始添加global声明

如果在一个局部(函数)内声明了一个global变量,那么这个变量在局部的所有操作将对全局的变量有效

a=2

def fi():

a = 1

def f2():

global a

a+=1

print(a)

f2()

# print("--a---",a)

fi()

print(a)

globals和locals方法

#globals 永远打印全局的名字

#locals 输出什么 根据locals所在的位置

def func():

a = 12

b = 20

print(locals())

print(globals())

func()

#print(globals())打印的内置空间和局部空间的名字 #{'__name__': '__main__', '__doc__':.....

#print(locals()) 打印的是本地的空间的名字(此时的本地为函数内部即局部命名空间)即{'b': 20, 'a': 12}

def func():

a = 12

b = 20

# print(locals())

print(globals())

func()

print(locals())

#都是打印的内置空间和局部空间的名字 #{'__name__': '__main__', '__doc__':.....

#因为此时locals()的本地为全局

函数的嵌套和作用域链

#函数的嵌套定义

#内部函数可以使用外部函数的变量

def max2(x,y):

m = x if x>y else y

return m

def max4(a,b,c,d):

res1 = max2(a,b)

res2 = max2(res1,c)

res3 = max2(res2,d)

return res3

# max4(23,-7,31,11)

函数的作用域链

def f1():

a = 1

def f2():

a = 2

f2()

print('a in f1 : ',a)

f1()

nonlocal关键字

#nonlocal 只能用于局部变量 找上层中离当前函数最近一层的局部变量

#声明了nonlocal的内部函数的变量修改会影响到 离当前函数最近一层的局部变量

# 对全局无效

# 对局部 也只是对 最近的 一层 有影响

a=1

def outer():

a=1

def inner():

b=1

def inner2():

nonlocal a

# global a

a+=1

# print('小小局部',a)

inner2()

# print('小局部',a)

inner()

print('局部:',a)

outer()

print("全局:",a)

#局部: 2

#全局: 1

函数名的本质

函数名本质上就是函数的内存地址

1.可以被引用

2.可以被当作容器类型(可变数据类型)的元素

3.可以当作函数的参数和返回值

def func():

print(123)

# func() #函数名就是内存地址

func2 = func #函数名可以赋值

func2()

l = [func,func2] #函数名可以作为容器类型的元素

print(l)

for i in l:

i()

'''

123

[, ]

123

123

'''

def func():

print(123)

def wahaha(f):

f()

return f #函数名可以作为函数的返回值

qqxing = wahaha(func) # 函数名可以作为函数的参数

qqxing()

#123

#123

闭包:

闭包需要2个条件:嵌套函数;内部函数调用外部函数的变量

def outer():

a = 1

def inner():

print(a)

inner()

outer()

闭包常用的方法:在外面直接执行内部的函数

def outer():

a = 1

def inner():

print(a)

return inner

inn = outer()

inn()

判断闭包函数的方法__closure__

def outer():

a = 1

def inner():

print(a)

print(inner.__closure__)

inner()

# print(outer.__closure__)

outer()

#(,)

#输出中有

def outer():

a = 1

def inner():

print(1)

print(inner.__closure__)

inner()

outer()

#不是闭包函数,就输出为None

总结:

命名空间:

一共有三种命名空间从大范围到小范围的顺序:内置命名空间、全局命名空间、局部命名空间

作用域(包括函数的作用域链):

小范围的可以用大范围的

但是大范围的不能用小范围的

范围从大到小(图)

a359f9dfe481ec2936a7aef7926b2c87.png

在小范围内,如果要用一个变量,是当前这个小范围有的,就用自己的

如果在小范围内没有,就用上一级的,上一级没有就用上上一级的,以此类推。

如果都没有,报错

函数的嵌套:

嵌套调用

嵌套定义:定义在内部的函数无法直接在全局被调用

函数名的本质:

就是一个变量,保存了函数所在的内存地址

闭包:

内部函数包含对外部作用域而非全剧作用域名字的引用,该内部函数称为闭包函数

装饰器函数

装饰器形成的过程 : 最简单的装饰器 有返回值的 有一个参数 万能参数

装饰器的作用:在不改变函数调用方式的基础上,在函数的前、后添加功能。

原则 :开放封闭原则

语法糖 :@

装饰器的固定模式

def timmer(f): #装饰器函数

def inner():

start = time.time()

ret = f() #被装饰的函数

end = time.time()

print(end - start)

return ret

return inner

@timmer #语法糖 @装饰器函数名

def func(): #被装饰的函数

time.sleep(0.01)

print('老板好同事好大家好')

return '新年好'

# func = timmer(func)

ret = func() #inner()

print(ret)

# 装饰器的作用 —— 不想修改函数的调用方式 但是还想在原来的函数前后添加功能

# timmer就是一个装饰器函数,只是对一个函数 有一些装饰作用

原则: 开放封闭原则

开放 : 对扩展是开放的

封闭 : 对修改是封闭的

带参数的装饰器

# def wrapper(f): #装饰器函数,f是被装饰的函数

# def inner(*args,**kwargs):

# '''在被装饰函数之前要做的事'''

# ret = f(*args,**kwargs) #被装饰的函数

# '''在被装饰函数之后要做的事'''

# return ret

# return inner

#

# @wrapper #语法糖 @装饰器函数名

# def func(a,b): #被装饰的函数

# time.sleep(0.01)

# print('老板好同事好大家好',a,b)

# return '新年好'

装饰器的固定模式

def wrapper(func): #qqxing

def inner(*args,**kwargs):

ret = func(*args,**kwargs) #被装饰的函数

return ret

return inner

@wrapper #qqxing = wrapper(qqxing)

def qqxing():

print(123)

ret = qqxing() #inner

时间功能的装饰器

import time

def wrapper(func):

def inner(*args,**kwargs):

start=time.time()

ret = func(*args,**kwargs)

end = time.time()

print(end - start)

return ret

return inner

@wrapper #qqxing = wrapper(qqxing)

def qqxing(a,b):

ret=a+b

time.sleep(0.1)

return ret

print(qqxing(1,2))

from functools import wraps

wraps(func)是python提供的给装饰器函数专门用来恢复被装饰函数性状的机制

import time

def wrapper(func):

# @wraps(func) #带参数的装饰器

def inner(*args,**kwargs):

start=time.time()

ret = func(*args,**kwargs)

end = time.time()

print(end - start)

return ret

return inner

@wrapper #qqxing = wrapper(qqxing)

def qqxing(a,b):

'''

这是一个记录运行时间的代码

:param a:

:param b:

:return:

'''

ret=a+b

time.sleep(0.1)

return ret

print(qqxing(1,2))

print(qqxing.__name__)

print(qqxing.__doc__)

#

#

# #结果为 0.10000038146972656

# #3

# #inner #实际上被装饰器的函数为inner

# #None

import time

from functools import wraps

def wrapper(func):

@wraps(func) #带参数的装饰器

def inner(*args,**kwargs):

start=time.time()

ret = func(*args,**kwargs)

end = time.time()

print(end - start)

return ret

return inner

@wrapper #qqxing = wrapper(qqxing)

def qqxing(a,b):

'''

这是一个记录运行时间的代码

:param a:

:param b:

:return:

'''

ret=a+b

time.sleep(0.1)

return ret

print(qqxing(1,2))

print(qqxing.__name__)

print(qqxing.__doc__) #函数的注释

#结果为:0.09999990463256836

#3

#qqxing #在装饰器内部的函数加上了一个wraps带参数的装饰器,这样被装饰函数就是自己了

'''

这是一个记录运行时间的代码

:param a:

:param b:

:return:

'''

带参数的装饰器可以方便的控制装饰器的取消.

import time

from functools import wraps

flag = True

def outer(flag):

def wrapper(func):

@wraps(func) #带参数的装饰器

def inner(*args,**kwargs):

if flag:

start=time.time()

ret = func(*args,**kwargs)

end = time.time()

print(end - start)

return ret

else:

ret = func(*args, **kwargs)

return ret

return inner

return wrapper

@outer(flag) #带参数的装饰器 #qqxing = wrapper(qqxing)

def qqxing(a,b):

ret=a+b

time.sleep(0.1)

return ret

print(qqxing(1,2))

多个装饰器控制一个函数

#多个装饰器控制一个函数

#离函数最近的装饰器函数先执行

from functools import wraps

flag = True

def outer1(flag):

def wrapper1(func): #func ---> func

@wraps(func)

def inner1(*args,**kwargs):

if flag:

print('函数之前执行w1')

ret1 =func(*args,**kwargs) #func()

print('函数之后结束w1')

return ret1 #ret1 = ret

else:

ret1 = func(*args, **kwargs)

return ret1

return inner1

return wrapper1

def outer2(flag):

def wrapper2(func): #func ----> inner1

@wraps(func)

def inner2(*args,**kwargs):

if flag:

print('函数之前执行w2')

ret2 =func(*args,**kwargs) #inner1()

print('函数之后结束w2')

return ret2 # ret2 = ret1= ret

else:

ret2 = func(*args, **kwargs)

return ret2

return inner2

return wrapper2

# wrapper1 = outer1(flag)

# wrapper2 = outer2(flag)

#

# @wrapper2 #func = w2(func) = w2(inner1) --->inner2

# @wrapper1 #func = w1(func) ---> inner1

# def func():

# print('123')

# ret ='hahha'

# return ret

@outer2(flag) #func = w2(func) = w2(inner1) --->inner2

@outer1(flag) #先执行 outer1装饰器函数 #func = w1(func) ---> inner1

def func():

print('123')

ret ='hahha'

return ret

ret = func() #func() ----> innner2()

print(ret)

print(func.__name__)

迭代器(iterator)

可以被for循环的就是可迭代的(iterable),所以说str,list,dic,tuple,set,文件句柄(f=open()),range(),enumerate都是可迭代的,同时他们中的双下方法都含有__iter__,因此只要是能被for循环的数据类型 就一定拥有__iter__方法,就一定是可迭代的。

[].__iter__()则返回的就是一个迭代器。,即可迭代的对象.__iter__就返回一个迭代器。迭代器里面重要的2个双下方法就是__iter__和__next__。所以只要有这2个内置方法就是一个迭代器。

Iterable 可迭代的 -- > __iter__   ;只要含有__iter__方法的都是可迭代的.

可迭代的对象.__iter__() 迭代器 -- > __next__ ;通过next就可以从迭代器中一个一个的取值。

只要含有__iter__方法的都是可迭代的 —— 可迭代协议

迭代器的概念:

迭代器协议 —— 内部含有__next__和__iter__方法的就是迭代器

迭代器协议和可迭代协议

可以被for循环的都是可迭代的

可迭代的内部都有__iter__方法

只要是迭代器 一定可迭代

可迭代的对象.__iter__()方法就可以得到一个迭代器。

迭代器中的__next__()方法可以一个一个的获取值;

for循环,就是在内部帮助可迭代对象变成一个迭代器,然后调用__next__方法来取值;其实就是在使用迭代器iterator可迭代对象直接给你内存地址;

只有可迭代对象才能for循环,因此在我们遇到一个新的变量的时候,不确定能不能用for循环的时候,就判断它是否可迭代.(通过dir(数据类型(变量))查看内部是否含有__iter__方法)

# for i in l:

# pass

#iterator = l.__iter__()

#iterator.__next__()

迭代器的好处:

从容器类型中一个一个的取值,会把所有的值都取到。

从而节省内存空间。

迭代器并不会在内存中再占用一大块内存,

而是随着循环,每次生成一个

每次用next方法每次给我一个。

生成器------>迭代器(generator)

生成器表达形式:

1.生成器函数——本质上就是我们自己写的函数,只不过返回值要用yield 来返回;每__next__一次执行一次yield。

生成器函数()调用之后不执行函数里面的代码,而是返回 一个迭代器 ()

2.生成器表达式

Python中提供的生成器:

1.生成器函数:常规函数定义,但是,使用yield语句而不是return语句返回结果。yield语句一次返回一个结果,在每个结果中间,挂起函数的状态,不会结束,以便下次重它离开的地方继续执行

def func():

for i in range(10):

yield '娃哈哈%s'%i

a = func()

print(a.__next__()) #娃哈哈0

print(a.__next__()) #娃哈哈1

print(a.__next__()) #娃哈哈2

2.生成器表达式:类似于列表推导,但是,生成器返回按需产生结果的一个对象,而不是一次构建一个结果列表

生成器Generator:

本质:迭代器(所以自带了__iter__方法和__next__方法,不需要我们去实现)

特点:惰性运算,开发者自定义

生成器的2个特点:

1.一个生成器中的数据只能从头到尾读取一次,读取完了就没有了。

2.惰性运算是指不找它取值,它就不工作,而且找它要一个,它就给你一个。

生成器函数的列子(监听文件输入的列子)

def tail(filename):

f = open(filename,encoding='utf-8')

while True:

line = f.readline()

if line.strip('\r\n'):

yield line.strip() #把文件返回用作过滤文件的内容,不用return ,用了函数就执行结束了。

data=tail('file') #用生成器来过滤

for i in data:

if 'python' in i :

print(i)

send方法

send 获取下一个值的效果和next基本一致;

只是在获取下一个值的时候,给上一yield的位置传递一个数据。

使用send的注意事项:第一次使用生成器的时候 是用next获取下一个值;最后一个yield不能接受外部的值

def generator():

print('123')

ret = yield 1

print('#####',ret)

print('456')

yield 2

print('789')

ret = yield 3

print(ret)

'''

'''

yield

g = generator()

ret = g.__next__()

print(ret)

ret1 =g.send('marry') #send的效果和__next__效果一样,只不过send可以给上一个yield传一个值。

print(ret1)

ret = g.__next__()

print(ret)

计算移动平均值的生成器,只不过每次第一步的时候要使用next方法;

def average():

sum = 0

count = 0

avg = 0

while True:

ret = yield avg

sum += ret

count += 1

avg = sum/count

a = average()

a.__next__()

print(a.send(10))

print(a.send(20))

因此有了预激生成器的装饰器,为了方便多个函数的时候,不用再外面写next方法‘

def gener_waper(func):

def inner(*args,**kwargs):

ret = func(*args,**kwargs)

ret.__next__()

return ret

return inner

@gener_waper

def average():

sum = 0

count = 0

avg = 0

while True:

ret = yield avg

sum += ret

count += 1

avg = sum/count

a = average()

print(a.send(10))

print(a.send(20))

yield from 返回最小的字符,即拆分

def generator():

a = ['123456','aaaaa','aa']

b = {'name':'zenghui','age':24}

b = {'name': 'zenghui', 'age': 24}.items()

yield from a #单个返回

yield from b #若为字典,则返回k

g = generator()

for i in g:

print(i)

列表推导式和生成器表达式

1.把列表解析的[]换成()得到的就是生成器表达式

2.列表解析与生成器表达式都是一种便利的编程方式,只不过生成器表达式更节省内存,不占用内存.

a = (i*i for i in range(10)) #生成器表达式

list = [ i for i in range(10)] #列表推导式

print(list) #[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

print(a) # at 0x0000000009D01DB0>

for i in a:

print(i)

#1 4 9 16 25 36 49 64 81

完整的列表推导式:

1.遍历取所有的值

2.筛选功能

[   每一个元素或者是和元素的相关操作   for 元素 in  可迭代的数据类型 ]   # 遍历挨个取值

[   每一个元素或者是和元素的相关操作   for 元素 in  可迭代的数据类型  if  元素相关的条件 ]   #筛选功能

#输出30以内被3整除的数

ret = [ i*i for i in range(30) if i%3==0]

print(ret)

生成器相关的面试题:

#生成器的面试题1

# def genger():

# for i in range(4):

# yield i

#

# g = genger()

#

# g1 = ( i for i in g)

# # print(list(g1))

# # print(g1)

# g2 = ( i for i in g1)

# print(list(g2))

#面试题2

def add(n,i):

return n+i

def gen():

for i in range(4):

yield i

g = gen()

for n in [1,10]:

g = (add(n,i) for i in g)

print(list(g))

# n = 1

# g = (add(n,i) for i in gen()) #g = [0,1,2,3]

# # print(list(g)) #一个生成器只能取一次,取完就没了

# n= 10 # i :10,11,12,13 n :10

# g = (add(n,i) for i in (add(n,i) for i in gen())) #没有取值还是一个生成器;

# # for i in (0,1,2,3)

# # (10,11,12,13)

# # (20,21,22,23)

# # print(list(g))

# print(list(g)) #取值的时候才执行调用生成器 n = 10 ,g = [10,11,12,13], for i in g

for n in [1,10,5]: #碰到循环生成器的时候需要拆开分析

g = (add(n, i) for i in g)

print(list(g))

# n = 1

# g = (add(n, i) for i in gen())

# n = 10

# g = (add(n, i) for i in (add(n, i) for i in gen()))

# n = 5

# g = (add(n, i) for i in (add(n, i) for i in (add(n, i) for i in gen())))

# # (0,1,2,3)

# # (5,6,7,8)

# # (10,11,12,13)

# # (15,16,17,18)

# print(list(g)) #取值,生成器里面的代码才执行

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值