python yield send 用法简明理解

前提

首先可以简单理解一下yield关键字,包含了关键字yield的函数,可以被视为一个生成器,这个生成器有更丰富的功能,并且这个生成器是随用随生成的,下面的例子说明了这点:

def get_next():
    for i in range(10):
        yield i

if __name__ == "__main__":

    g = get_next()

    # 输出结果完全一致,都是0-10的数字
    print("yield生成结果:")
    print([x for x in g])
    print("传统生成器结果:")
    print([x for x in range(10)])

两次print生成的结果完全一样,说明get_next函数就是一个生成器。

按照网上其他人说法,假设生成100万个数字,传统生成器需要实现把这一百万个数字准备好,而yield则是随用随取;假设一下,你现在只用了前5个数据,后面的直接忽略了,那么传统生成器生成100万个数字就有浪费了。这一点可以看其他人的验证。

用处

因此,可以认为yield有两个用处,一是提供功能更丰富的的生成器,二是提供更节约资源的生成器。

下面来看一下实际效果。

效果

参照别人的代码,进行了部分修改,主要对inner和outer两个值进行了比较,以此来帮助理解yield与传统生成器的区别。

其中结合业务逻辑,可以假设outer是真正业务中需要的值,这个值由call(i)函数生成,分别是0,2,4,6,8

关于yield,它就是一个类似于return的关键词。类似于生成器,调用__next__或者send()方法时,yield会立即返回一个值并停止执行,当下一次调用__next__或者send()方法的时候,会从停止执行的位置继续执行

最需要额外注意的是,如果yield函数与其他变量结合(例如代码中的inner)并被外界调用,如果是通过__next__调用的,那么inner会为空,如果是通过send()调用的,inner会被设置为相应的值;二者的区别在于返回之后,如何填补原来的空位。

接下来,通过inner变量,可以看出next和send的区别,可以认为send包含了next。

inner分析

在get_next()函数中打印“inner”的值,这个值在step1中不会被打印,

在step2和step3中,“inner”的值为none,这是因为,在"yield call(i)"时,直接返回了call(i),并且将none赋给了“inner”;在外界是通过__next__调用的.

在step4中,因为调用了send方法,因此在在"yield call(i)"时,不仅仅返回了call(i),还将将外部send发送进来的值赋给了“inner”,

在step5中,send()函数发送进来的值可以是任意值,只要满足业务要求,也就是说这个send进来的值将作用于inner;

代码如下:

def call(i):
    return i * 2


def get_next():
    for i in range(100):
        inner = yield call(i)
        print("inner----", inner)

if __name__ == "__main__":

    g = get_next()
    
    print("**********step 1**********")
    print("outer----", g.__next__()) 
    
    
    print("**********step 2**********")
    print("outer----", g.__next__())
    print("**********step 3**********")
    value_to_send = g.__next__()
    print("outer----", value_to_send)
    
    
    print("**********step 4**********")
    value_to_send = g.send(value_to_send)
    print("outer----", value_to_send)
    
    
    print("**********step 5**********")
    value_to_send = g.send("达到最大步骤5,需要提前终止")
    print("outer----", value_to_send)

	# 运行结果:
	# **********step 1**********
	# outer---- 0
	# **********step 2**********
	# inner---- None
	# outer---- 2
	# **********step 3**********
	# inner---- None
	# outer---- 4
	# **********step 4**********
	# inner---- 4
	# outer---- 6

对比

当不考虑inner变量,仅仅考虑外部的outer变量时,情况就变得简单了;注意,此时仍然能够通过send()函数继续进行数字生成,并且效果和__next__是相似的:

def call(i):
    return i * 2

def get_next():
    for i in range(100):
        yield call(i)

if __name__ == "__main__":

    g = get_next()
    
    print("**********step 1**********")
    print("outer----", g.__next__())
    print("**********step 2**********")
    print("outer----", g.__next__())


    print("**********step 3**********")
    print("outer----",g.send("输入不会影响yield"))
    
	# 输出结果
	# **********step 1**********
	# outer---- 0
	# **********step 2**********
	# outer---- 2
	# **********step 3**********
	# outer---- 4
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
yield函数是一个生成器(generator),它可以用于迭代。在函数中,yield类似于return,不同的是,yield返回一个值并且记住这个返回值的位置,下次迭代就从记住的位置开始执行,并且从上一次迭代遇到的yield后面的代码开始执行。 yield函数的优点在于它可以减少内存的消耗,因为它不直接生成返回值。相比使用return返回值或者迭代大量数据,yield函数能够更好地减少内存消耗。然而,由于yield函数只能读取一次,所以它的特点是只能迭代一次。 yield函数不仅可以用于for循环,还可以作为函数参数。例如,可以将yield函数作为其他函数的参数传递,实现更加灵活的功能。 除了next()函数之外,还有一个特别的函数叫做send()函数。send()函数的特点是可以携带参数,并且可以修改上一个yield表达式的值。使用方式与next()函数有很多相似之处。<span class="em">1</span><span class="em">2</span><span class="em">3</span><span class="em">4</span> #### 引用[.reference_title] - *1* *2* *3* *4* [Python|yield的解析及用法](https://blog.csdn.net/gschen_cn/article/details/107293784)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值