iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Python中怎么理解yield from语法
  • 398
分享到

Python中怎么理解yield from语法

2023-06-01 22:06:24 398人浏览 泡泡鱼

Python 官方文档:入门教程 => 点击学习

摘要

本篇内容介绍了“python中怎么理解yield from语法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!. 为什么要使用协程在

本篇内容介绍了“python中怎么理解yield from语法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

. 为什么要使用协程

在上一篇中,我们从生成器的基本认识与使用,成功过渡到了协程。

但一定有许多人,只知道协程是个什么东西,但并不知道为什么要用协程?换句话来说,并不知道在什么情况下用协程?它相比多线程来说,有哪些过人之处呢?

在开始讲yield from 之前,我想先解决一下这个给很多人带来困惑的问题。

举个例子。假如我们做一个爬虫。我们要爬取多个网页,这里简单举例两个网页(两个spider函数),获取html(耗io耗时),然后再对HTML对行解析取得我们感兴趣的数据。

我们的代码结构精简如下:

def spider_01(url):

    html = get_html(url)

    ...

    data = parse_html(html)

def spider_02(url):

    html = get_html(url)

    ...

    data = parse_html(html)

我们都知道,get_html()等待返回网页是非常耗IO的,一个网页还好,如果我们爬取的网页数据极其庞大,这个等待时间就非常惊人,是极大的浪费。

聪明的程序员,当然会想如果能在get_html()这里暂停一下,不用傻乎乎地去等待网页返回,而是去做别的事。等过段时间再回过头来到刚刚暂停的地方,接收返回的html内容,然后还可以接下去解析parse_html(html)。

利用常规的方法,几乎是没办法实现如上我们想要的效果的。所以Python想得很周到,从语言本身给我们实现了这样的功能,这就是yield语法。可以实现在某一函数中暂停的效果。

试着思考一下,假如没有协程,我们要写一个并发程序。可能有以下问题

1)使用最常规的同步编程要实现异步并发效果并不理想,或者难度极高。2)由于GIL的存在,多线程的运行需要频繁的加锁解锁,切换线程,这极大地降低了并发性能;

而协程的出现,刚好可以解决以上的问题。它的特点有

00001. 协程是在单线程里实现任务的切换的

00002. 利用同步的方式去实现异步

00003. 不再需要锁,提高了并发性能

. yield from的用法详解

yield from 是在python3.3才出现的语法。所以这个特性在Python2中是没有的。

yield from 后面需要加的是可迭代对象,它可以是普通的可迭代对象,也可以是迭代器,甚至是生成器。

简单应用:拼接可迭代对象

我们可以用一个使用yield和一个使用yield from的例子来对比看下。

使用yield

# 字符串astr='ABC'# 列表alist=[1,2,3]# 字典adict={"name":"wangbm","age":18}# 生成器agen=(i for i in range(4,8))

def gen(*args, **kw):

    for item in args:

        for i in item:

            yield i

new_list=gen(astr, alist, adict, agen)print(list(new_list))# ['A', 'B', 'C', 1, 2, 3, 'name', 'age', 4, 5, 6, 7]

使用yield from

# 字符串astr='ABC'# 列表alist=[1,2,3]# 字典adict={"name":"wangbm","age":18}# 生成器agen=(i for i in range(4,8))

def gen(*args, **kw):

    for item in args:

        yield from item

new_list=gen(astr, alist, adict, agen)print(list(new_list))# ['A', 'B', 'C', 1, 2, 3, 'name', 'age', 4, 5, 6, 7]

由上面两种方式对比,可以看出,yield from后面加上可迭代对象,他可以把可迭代对象里的每个元素一个一个的yield出来,对比yield来说代码更加简洁,结构更加清晰。

复杂应用:生成器的嵌套

如果你认为只是 yield from 仅仅只有上述的功能的话,那你就太小瞧了它,它的更强大的功能还在后面。

当 yield from 后面加上一个生成器后,就实现了生成的嵌套。

当然实现生成器的嵌套,并不是一定必须要使用yield from,而是使用yield from可以让我们避免让我们自己处理各种料想不到的异常,而让我们专注于业务代码的实现。如果自己用yield去实现,那只会加大代码的编写难度,降低开发效率,降低代码的可读性。既然Python已经想得这么周到,我们当然要好好利用起来。

讲解它之前,首先要知道这个几个概念

调用方:调用委派生成器的客户端(调用方)代码2、委托生成器:包含yield from表达式的生成器函数3、子生成器:yield from后面加的生成器函数

你可能不知道他们都是什么意思,没关系,来看下这个例子。

这个例子,是实现实时计算平均值的。比如,第一次传入10,那返回平均数自然是10.第二次传入20,那返回平均数是(10+20)/2=15第三次传入30,那返回平均数(10+20+30)/3=20

# 子生成器def average_gen():

    total = 0

    count = 0

    average = 0

    while True:

        new_num = yield average

        count += 1

        total += new_num

        average = total/count

# 委托生成器def proxy_gen():

    while True:

        yield from average_gen()

# 调用方def main():

    calc_average = proxy_gen()

    next(calc_average)            # 预激下生成器

    print(calc_average.send(10))  # 打印:10.0

    print(calc_average.send(20))  # 打印:15.0

    print(calc_average.send(30))  # 打印:20.0

if __name__ == '__main__':

    main()

认真阅读以上代码,你应该很容易能理解,调用方、委托生成器、子生成器之间的关系。我就不多说了

委托生成器的作用是:在调用方与子生成器之间建立一个双向通道。

所谓的双向通道是什么意思呢?调用方可以通过send()直接发送消息给子生成器,而子生成器yield的值,也是直接返回给调用方。

你可能会经常看到有些代码,可以在yield from前面看到可以赋值。这是什么用法?

你可能会以为,子生成器yield回来的值,被委托生成器给拦截了。你可以亲自写个demo运行试验一下,并不是你想的那样。因为我们之前说了,委托生成器,只起一个桥梁作用,它建立的是一个双向通道,它并没有权利也没有办法,对子生成器yield回来的内容做拦截。

为了解释这个用法,我还是用上述的例子,并对其进行了一些改造。添加了一些注释,希望你能看得明白。

按照惯例,我们还是举个例子。

# 子生成器def average_gen():

    total = 0

    count = 0

    average = 0

    while True:

        new_num = yield average

        if new_num is None:

            break

        count += 1

        total += new_num

        average = total/count

    # 每一次return,都意味着当前协程结束。

    return total,count,average

# 委托生成器def proxy_gen():

    while True:

        # 只有子生成器要结束(return)了,yield from左边的变量才会被赋值,后面的代码才会执行。

        total, count, average = yield from average_gen()

        print("计算完毕!!\n总共传入 {} 个数值, 总和:{},平均数:{}".fORMat(count, total, average))

# 调用方def main():

    calc_average = proxy_gen()

    next(calc_average)            # 预激协程

    print(calc_average.send(10))  # 打印:10.0

    print(calc_average.send(20))  # 打印:15.0

    print(calc_average.send(30))  # 打印:20.0

    calc_average.send(None)      # 结束协程

    # 如果此处再调用calc_average.send(10),由于上一协程已经结束,将重开一协程

if __name__ == '__main__':

    main()

运行后,输出

015.020.0计算完毕!!总共传入 3 个数值, 总和:60,平均数:20.0

. 为什么要使用yield from

学到这里,我相信你肯定要问,既然委托生成器,起到的只是一个双向通道的作用,我还需要委托生成器做什么?我调用方直接调用子生成器不就好啦?

高能预警~~~

下面我们来一起探讨一下,到底yield from 有什么过人之处,让我们非要用它不可。

因为它可以帮我们处理异常

如果我们去掉委托生成器,而直接调用子生成器。那我们就需要把代码改成像下面这样,我们需要自己捕获异常并处理。而不像使yield from那样省心。

# 子生成器# 子生成器def average_gen():

    total = 0

    count = 0

    average = 0

    while True:

        new_num = yield average

        if new_num is None:

            break

        count += 1

        total += new_num

        average = total/count

    return total,count,average

# 调用方def main():

    calc_average = average_gen()

    next(calc_average)            # 预激协程

    print(calc_average.send(10))  # 打印:10.0

    print(calc_average.send(20))  # 打印:15.0

    print(calc_average.send(30))  # 打印:20.0

    # ----------------注意-----------------

    try:

        calc_average.send(None)

    except StopIteration as e:

        total, count, average = e.value

        print("计算完毕!!\n总共传入 {} 个数值, 总和:{},平均数:{}".format(count, total, average))

    # ----------------注意-----------------

if __name__ == '__main__':

    main()

此时的你,可能会说,不就一个StopIteration的异常吗?自己捕获也没什么大不了的。

你要是知道yield from在背后为我们默默无闻地做了哪些事,你就不会这样说了。

具体yield from为我们做了哪些事,可以参考如下这段代码。

#一些说明"""_i:子生成器,同时也是一个迭代器_y:子生成器生产的值_r:yield from 表达式最终的值_s:调用方通过send()发送的值_e:异常对象"""

_i = iter(EXPR)

try:

    _y = next(_i)except StopIteration as _e:

    _r = _e.value

else:

    while 1:

        try:

            _s = yield _y

        except GeneratorExit as _e:

            try:

                _m = _i.close

            except AttributeError:

                pass

            else:

                _m()

            raise _e

        except BaseException as _e:

            _x = sys.exc_info()

            try:

                _m = _i.throw

            except AttributeError:

                raise _e

            else:

                try:

                    _y = _m(*_x)

                except StopIteration as _e:

                    _r = _e.value

                    break

        else:

            try:

                if _s is None:

                    _y = next(_i)

                else:

                    _y = _i.send(_s)

            except StopIteration as _e:

                _r = _e.value

                breakRESULT = _r

以上的代码,稍微有点复杂,有兴趣的同学可以结合以下说明去研究看看。

00001. 迭代器(即可指子生成器)产生的值直接返还给调用者

00002. 任何使用send()方法发给委派生产器(即外部生产器)的值被直接传递给迭代器。如果send值是None,则调用迭代器next()方法;如果不为None,则调用迭代器的send()方法。如果对迭代器的调用产生StopIteration异常,委派生产器恢复继续执行yield from后面的语句;若迭代器产生其他任何异常,则都传递给委派生产器。

00003. 子生成器可能只是一个迭代器,并不是一个作为协程的生成器,所以它不支持.throw()和.close()方法,即可能会产生AttributeError 异常。

00004. 除了GeneratorExit 异常外的其他抛给委派生产器的异常,将会被传递到迭代器的throw()方法。如果迭代器throw()调用产生了StopIteration异常,委派生产器恢复并继续执行,其他异常则传递给委派生产器。

00005. 如果GeneratorExit异常被抛给委派生产器,或者委派生产器的close()方法被调用,如果迭代器有close()的话也将被调用。如果close()调用产生异常,异常将传递给委派生产器。否则,委派生产器将抛出GeneratorExit 异常。

00006. 当迭代器结束并抛出异常时,yield from表达式的值是其StopIteration 异常中的第一个参数。

00007. 一个生成器中的return expr语句将会从生成器退出并抛出 StopIteration(expr)异常。

“Python中怎么理解yield from语法”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: Python中怎么理解yield from语法

本文链接: https://www.lsjlt.com/news/227964.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Python中怎么理解yield from语法
    本篇内容介绍了“Python中怎么理解yield from语法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!. 为什么要使用协程在...
    99+
    2023-06-01
  • python中yield from的使用方法
    这篇文章主要介绍了python中yield from的使用方法,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。python的五大特点是什么python的五大特点:1.简单易学,...
    99+
    2023-06-14
  • Python中Yield语法怎么用
    这篇文章主要介绍了Python中Yield语法怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。Python中Yield语法比如在这里,我们定义了一个fibonacci()...
    99+
    2023-06-27
  • python中yield的用法详解—
    此文转载,侵删,原文地址:https://blog.csdn.net/mieleiz... 首先,如果你还没有对yield有个初步分认识,那么你先把yield看做“return”,这个是直观的,它首先是个return,普通的return是...
    99+
    2023-01-31
    详解 python yield
  • python中yield函数的用法详解
    首先,如果你还没有对yield有个初步分认识,那么你先把yield看做“return”,这个是直观的,它首先是个return,普通的return是什么意思,就...
    99+
    2024-04-02
  • Python中Yield的基本用法及Yield与return的区别解析
    目录一、yield的功能1. 使用next方法迭代生成器2. 使用send()方法与生成器函数通信3. Yield的好处二、总结yield的功能三、yield VS return一、...
    99+
    2024-04-02
  • Python语法怎么理解
    这篇文章主要介绍“Python语法怎么理解”,在日常操作中,相信很多人在Python语法怎么理解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python语法怎么理解”的疑惑有所帮助!接下来,请跟着小编一起来...
    99+
    2023-06-17
  • Python中的yield关键词怎么用
    这篇文章将为大家详细讲解有关Python中的yield关键词怎么用,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。要理解yield的作用,你必须理解生成器是什么。在理解生成器之前,必须先理解迭代器。迭代器当...
    99+
    2023-06-22
  • python语法中导入import与from时的代码怎么编写
    今天就跟大家聊聊有关python语法中导入import与from时的代码怎么编写,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。import cgi  i...
    99+
    2023-06-17
  • JavaScript中from()方法怎么用
    这篇文章主要介绍了JavaScript中from()方法怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。from()以对象创建数组。co...
    99+
    2024-04-02
  • python中from import的用法是什么
    在Python中,`from import`语句用于从模块中导入特定的函数、类或变量。它有以下几种使用方式:1. 导入模块中的特定函...
    99+
    2023-10-12
    python
  • Python中yield返回生成器的方法是什么
    这篇文章主要讲解了“Python中yield返回生成器的方法是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Python中yield返回生成器的方法是什么”吧!一、迭代器逐项读取列表,称...
    99+
    2023-06-25
  • pytest解读fixtures之Teardown怎么处理yield和addfinalizer
    本文小编为大家详细介绍“pytest解读fixtures之Teardown怎么处理yield和addfinalizer”,内容详细,步骤清晰,细节处理妥当,希望这篇“pytest解读fixtures之Teardown怎么处理yield和ad...
    99+
    2023-06-30
  • LINQ语法怎么理解
    这篇文章主要介绍“LINQ语法怎么理解”,在日常操作中,相信很多人在LINQ语法怎么理解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”LINQ语法怎么理解”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!程序...
    99+
    2023-06-17
  • 怎么理解Python动态语言
    本篇文章给大家分享的是有关怎么理解Python动态语言,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。有人说Python动态语言是一种解释性的语言,但是这种说法是不正确的,其实P...
    99+
    2023-06-17
  • sql更新语句中update set from怎么用
    这篇文章主要介绍了sql更新语句中update set from怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。  &n...
    99+
    2023-06-29
  • 怎么理解PostgreSQL语法分析中的上下文无关语法
    本篇内容介绍了“怎么理解PostgreSQL语法分析中的上下文无关语法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够...
    99+
    2024-04-02
  • Python语法中的Module模块该如何理解
    Python语法中的Module模块该如何理解,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、定义包含一系列数据,函数,类的文件,通常以.py结尾二、作用让一些相关的数据,函...
    99+
    2023-06-22
  • 怎么理解Python中Django
    本篇内容介绍了“怎么理解Python中Django”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!  Django相较与其他web框架的优势在...
    99+
    2023-06-01
  • JavaScript高级语法中的模块化怎么理解
    这篇文章将为大家详细讲解有关JavaScript高级语法中的模块化怎么理解,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。什么是模块化?到底什么是模块化、模块...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作