浅谈 Python 中的 yield——生成器对象与函数调用的区别

我们来看这么一个例子:

python 复制代码
def greeter():
    name = yield "你是谁?"
    yield f"你好,{name}"

g = greeter()
print(next(g))        # → "你是谁?"
print(g.send("张三")) # → "你好,张三"

执行流程:

复制代码
next(g):执行到 yield,返回 "你是谁?";

send("张三"):将 "张三" 送入上次 yield 表达式的返回值,继续执行。

send() 是构建协程通信、任务调度器的核心机制之一(如 asyncio、trio、LangGraph 内核)。

g = greeter()是greeter() 执行的结果赋值给 g这个结果是一个生成器对象(generator) ,而不是把函数本身赋值给 g


🔍 详细分析

当我们写:

python 复制代码
def greeter():
    name = yield "你是谁?"
    yield f"你好,{name}"

此时,greeter 是一个函数对象(还没执行)。

当我们调用 greeter()

python 复制代码
g = greeter()

Python 会:

  • 执行函数定义,但并不运行函数体
  • 因为函数中含有 yield,所以它会返回一个 生成器对象 ,即 g 是一个 generator

我们可以验证:

python 复制代码
print(type(g))  # <class 'generator'>

这就意味着:

python 复制代码
g = greeter()  # g 现在是生成器对象,不是函数本身

✅ 所以两种情况的对比:

写法 含义 类型
g = greeter 把函数本身赋值给变量 g,不执行 <function>
g = greeter() 执行函数,返回一个生成器对象 <generator>

✅ 举个例子验证一下

python 复制代码
def example():
    yield 42

print(example)      # <function example at 0x...>
print(example())    # <generator object example at 0x...>

✅ 总结

greeter() 是函数调用表达式,执行时返回一个生成器对象 ;所以 g = greeter() 是在 将生成器对象赋值给 g ,而不是将函数本身赋值给 g

相关推荐
TF男孩9 小时前
ARQ:一款低成本的消息队列,实现每秒万级吞吐
后端·python·消息队列
该用户已不存在14 小时前
Mojo vs Python vs Rust: 2025年搞AI,该学哪个?
后端·python·rust
站大爷IP16 小时前
Java调用Python的5种实用方案:从简单到进阶的全场景解析
python
用户8356290780511 天前
从手动编辑到代码生成:Python 助你高效创建 Word 文档
后端·python
侃侃_天下1 天前
最终的信号类
开发语言·c++·算法
c8i1 天前
python中类的基本结构、特殊属性于MRO理解
python
echoarts1 天前
Rayon Rust中的数据并行库入门教程
开发语言·其他·算法·rust
liwulin05061 天前
【ESP32-CAM】HELLO WORLD
python
Aomnitrix1 天前
知识管理新范式——cpolar+Wiki.js打造企业级分布式知识库
开发语言·javascript·分布式
Doris_20231 天前
Python条件判断语句 if、elif 、else
前端·后端·python