微信扫码
添加专属顾问
我要投稿
Google开源LangGraph项目实战指南,手把手教你打造可深度定制的AI Agent。 核心内容: 1. LangGraph项目的核心价值与适用场景 2. 项目架构解析与技术栈选择 3. 从零搭建AI Agent的完整流程演示
最近在带着同事一起做智能 Agent 相关的内部项目,发现很多人对 LangGraph 非常感兴趣,但又不太清楚如何从零开始搭建一个完整的 AI Agent,我于是在 github 上找,看看有没有好的开源项目给他们学习,偶然间发现了 google-gemini 开源的这个项目[1],正好拿来给他们讲讲,学习学习。发现整理的材料又正好可以出一期公众号文章,就作为我公众号的一篇番外,来讲讲。
题外话:我十分不主张,一遇到新需求,就好像对应的知识就得全部掌握,得去花大力气看看基础教程,了解全貌之后,再开始行动。如果真是这样的话,这个项目大概率就得黄了。应该是要根据目标去找,去按需学习。
你可能已经用过市面上的一些“低代码/零代码”智能体平台,比如 Coze、dify 这类产品。它们就像搭积木一样,让你拖拽组件、配置流程,快速拼出一个属于自己的 AI Agent。这种方式对于业务快速上线、简单场景非常友好。
但你有没有遇到过这样的问题:
这时候,你就需要跳出“应用端拼装”,进入“代码层级的自定义”。本项目就是为此而生——它不仅让你看到一个完整的“AI 研究员”是怎么从零到一搭建起来的,还能让你随时插拔、扩展、魔改每一个环节。
它适合:
核心亮点:
在接下来的内容中,我会像一位带你实战的讲师,带你从全局到细节,逐步拆解这个项目的每一处关键实现。
在正式读代码前,我们先快速了解一下项目的整体结构和技术选型。
目录结构简述:
frontend/
:React 前端,负责 UI 展示、与后端 API 通信backend/
:Python 后端,核心为 LangGraph 代理src/agent/
:代理主流程、工具、状态定义examples/cli_research.py
:命令行调用代理的最小示例技术栈:
核心流程:
你可以把它想象成一个“AI 研究员”,自动帮你查资料、归纳、补充、引用,最后给你一份有理有据的答案。
在项目根目录下:
setup.bat install
或分别:
setup.bat install-backend
setup.bat install-frontend
setup.bat dev
setup.bat cli-example "今天武汉的天气怎么样?"
你可以先随便问一个问题,感受一下“processing...”之后,AI 是如何给你一份带引用的研究报告的。
接下来,让我们像课堂实战一样,带着问题、带着好奇心,一步步走进项目的核心实现。
我们先从最简单的命令行入口开始。
文件:backend/examples/cli_research.py
if __name__ == "__main__": main()
(第42行)main()
(第5行)你可以打开这个文件,看到如下关键代码:
from agent.graph import graph # 第3行
...
result = graph.invoke(state) # 第36行
这里的 graph.invoke(state)
,就是整个“AI 研究员”流程的起点。你输入一个问题,所有的自动研究、搜索、推理、引用,都是从这里开始的。
小贴士:如果你想快速体验后端的效果,可以直接在命令行运行:
python backend/examples/cli_research.py "今天武汉的天气怎么样?"
接下来,我们顺着 from agent.graph import graph
跳到核心代理的定义。
文件:backend/src/agent/graph.py
graph = builder.compile(name="pro-search-agent")
(第293行)你会看到一段类似“流程图”的代码:
builder.add_node("generate_query", generate_query) # 第272行
builder.add_node("web_research", web_research) # 第273行
builder.add_node("reflection", reflection) # 第274行
builder.add_node("finalize_answer", finalize_answer) # 第275行
...
builder.add_edge(START, "generate_query") # 第279行
builder.add_conditional_edges("generate_query", continue_to_web_research, ["web_research"]) # 第281行
builder.add_edge("web_research", "reflection") # 第285行
builder.add_conditional_edges("reflection", evaluate_research, ["web_research", "finalize_answer"]) # 第287行
builder.add_edge("finalize_answer", END) # 第291行
林生点评:
现在我们带着“AI 研究员”的视角,逐个节点来看它们都做了什么。
generate_query
节点def generate_query(state: OverallState, config: RunnableConfig) -> QueryGenerationState:
(第44行)llm = ChatGoogleGenerativeAI(...)
structured_llm = llm.with_structured_output(SearchQueryList)
formatted_prompt = query_writer_instructions.format(...)
result = structured_llm.invoke(formatted_prompt)
return {"search_query": result.query}
SearchQueryList
是在 tools_and_schemas.py 里定义的结构,专门用来描述“搜索词列表+理由”。continue_to_web_research
节点def continue_to_web_research(state: QueryGenerationState):
(第84行)return [
Send("web_research", {"search_query": search_query, "id": int(idx)})
for idx, search_query in enumerate(state["search_query"])
]
web_research
节点def web_research(state: WebSearchState, config: RunnableConfig) -> OverallState:
(第95行)response = genai_client.models.generate_content(...)
resolved_urls = resolve_urls(...)
citations = get_citations(response, resolved_urls)
modified_text = insert_citation_markers(response.text, citations)
sources_gathered = [item for citation in citations for item in citation["segments"]]
return {
"sources_gathered": sources_gathered,
"search_query": [state["search_query"]],
"web_research_result": [modified_text],
}
reflection
节点def reflection(state: OverallState, config: RunnableConfig) -> ReflectionState:
(第139行)formatted_prompt = reflection_instructions.format(...)
llm = ChatGoogleGenerativeAI(...)
result = llm.with_structured_output(Reflection).invoke(formatted_prompt)
return {
"is_sufficient": result.is_sufficient,
"knowledge_gap": result.knowledge_gap,
"follow_up_queries": result.follow_up_queries,
...
}
Reflection
结构体也是在 tools_and_schemas.py 里定义的,描述“是否足够、知识盲区、补充查询”。evaluate_research
节点def evaluate_research(state: ReflectionState, config: RunnableConfig) -> OverallState:
(第183行)if state["is_sufficient"] or state["research_loop_count"] >= max_research_loops:
return "finalize_answer"
else:
return [
Send("web_research", {...})
for idx, follow_up_query in enumerate(state["follow_up_queries"])
]
finalize_answer
节点def finalize_answer(state: OverallState, config: RunnableConfig):
(第220行)formatted_prompt = answer_instructions.format(...)
llm = ChatGoogleGenerativeAI(...)
result = llm.invoke(formatted_prompt)
# 替换短链为原始链接
for source in state["sources_gathered"]:
if source["short_url"] in result.content:
result.content = result.content.replace(
source["short_url"], source["value"]
)
unique_sources.append(source)
return {
"messages": [AIMessage(content=result.content)],
"sources_gathered": unique_sources,
}
在读节点实现时,你可能会好奇:SearchQueryList
、Reflection
这些结构体是怎么定义的?
文件:backend/src/agent/tools_and_schemas.py
SearchQueryList
(第5行):定义了搜索词列表和 rationale。Reflection
(第14行):定义了反思节点的输出结构。你可以随时跳到这个文件,看看每个字段的含义和注释。
到这里,我们已经带着“AI 研究员”的视角,完整走了一遍从命令行输入到最终答案输出的全链路。
cli_research.py
第36行 graph.invoke(state)
graph.py
第269-291行(StateGraph 构建)graph.py
第36-265行(每个节点的具体逻辑)tools_and_schemas.py
(第5-24行)你可以跟着这些跳转,边看边实验,体会每一步的数据流转和设计巧思。
最后,给大家一些实战建议:
SearchQueryList
、Reflection
),可跳到 tools_and_schemas.py
查看定义。tools_and_schemas.py
或 utils.py
。本文章可作为写作、讲解、二次开发的基础材料,欢迎补充和完善!希望你能像带着学员实战一样,真正掌握 LangGraph 应用开发的精髓。
原先的 google-gemini 该项目的地址如下,大家可以跳转去看:https://github.com/google-gemini/gemini-fullstack-langgraph-quickstart
我在这个项目的基础上进行了部分修改,有完整的中文使用教程,以及本地 Windows 部署脚本调整,方便大家直接运行和二次开发。如果对这块感兴趣,可以关注我的微信公众号,直接发消息“langgraph”,即可获得该项目的完整代码和使用教程。
53AI,企业落地大模型首选服务商
产品:场景落地咨询+大模型应用平台+行业解决方案
承诺:免费场景POC验证,效果验证后签署服务协议。零风险落地应用大模型,已交付160+中大型企业
2025-07-09
Prompt Optimizer — AI 提示词优化神器
2025-07-09
昆仑万维发布 Skywork-R1V3.0:开源模型挑战多模态推理极限,逼近人类专家水平
2025-07-08
大模型开源许可解析:脉络与最新发展趋势
2025-07-08
探索 RAG-Anything:开启多模态 RAG 的新纪元,让文档“活”起来!
2025-07-08
国内 Agent 赛道最大融资诞生!阿里云为何重注这家“真能干活”的 AI 公司?
2025-07-08
Agent 记忆拆解 | Gemini CLI
2025-07-07
字节跳动开源Trae-Agent:让AI成为你的"全栈开发合伙人"
2025-07-07
16.2k星星!一个「想让所有人都用上」的开源AI多智能体系统框架。
2025-06-17
2025-06-17
2025-04-13
2025-04-29
2025-04-12
2025-04-29
2025-04-15
2025-04-29
2025-05-29
2025-05-12
2025-07-08
2025-07-04
2025-07-03
2025-06-28
2025-06-25
2025-06-25
2025-06-21
2025-06-16