执行计划的最终装配
前面两篇讲了 documentQuestionRouter.route 的执行模式判定和章节解析策略,这一篇来看 ChatPreparationOrchestrator#prepare 方法的最后一段——把所有准备好的信息装配成最终的执行计划。
执行模式兜底
// 如果导航阶段没有给出明确 executionMode,则默认退回普通 RETRIEVAL 模式。
ExecutionMode executionMode = navigationDecision == null || navigationDecision.getExecutionMode() == null
? ExecutionMode.RETRIEVAL
: navigationDecision.getExecutionMode();
documentQuestionRouter.route 理论上总会返回一个 executionMode,但为了防止意外情况(比如路由器返回 null),这里做了一个兜底:如果没有明确的执行模式,就默认走 RETRIEVAL(普通混合检索)。这是最保守也最安全的选择。
检索问题的最终确定
// retrievalQuestion / retrievalSubQuestions 允许导航阶段进一步微调,若没有则沿用改写结果。
String retrievalQuestion = navigationDecision == null || navigationDecision.getRetrievalPlan() == null
? rewriteQuestion
: firstNonBlank(navigationDecision.getRetrievalPlan().getRetrievalQuestion(), rewriteQuestion);
List<String> retrievalSubQuestions = navigationDecision == null || navigationDecision.getRetrievalPlan() == null
|| navigationDecision.getRetrievalPlan().getSubQuestions() == null || navigationDecision.getRetrievalPlan().getSubQuestions().isEmpty()
? rewriteSubQuestions
: navigationDecision.getRetrievalPlan().getSubQuestions();
这里有个细节:retrievalQuestion 和 rewriteQuestion 是两个不同的字段。
rewriteQuestion:改写阶段产出的问题,用于记录"改写了什么"retrievalQuestion:最终用于检索的问题,可能被导航阶段进一步微调
导航阶段(documentQuestionRouter)在构建 retrievalPlan 时,可能会对检索问题做进一步调整,比如加入章节上下文、精简问题表达等。如果导航阶段没有调整,就直接沿用改写结果。
子问题同理:优先用导航阶段的子问题,没有就用改写阶段的。
执行计划装配
// 最终把记忆、改写、路由、时间敏感信息和无证据回复一起装配成执行计划。
return basePlan(question, chatMode, memoryContext, historyPlanningContext, historySummary, answerHistoryContext, currentDate, currentDateText,
requiresCurrentDateAnchoring, requiresFreshSearch)
// 这里开始在 basePlan 的公共骨架上,补上"本轮文档链路真正判定出来的执行模式"。
.mode(executionMode)
// 把文档路由器给出的结构导航决策挂进去,后续执行器会根据它决定走图查询还是混合检索。
.navigationDecision(navigationDecision)
// 用 rewrite 阶段产出的改写问题覆盖 basePlan 中"默认等于原问题"的占位值。
.rewriteQuestion(rewriteQuestion)
.rewriteSubQuestions(rewriteSubQuestions)
// retrievalQuestion / retrievalSubQuestions 允许被导航阶段进一步调整,因此这里写入的是最终检索版本。
.retrievalQuestion(retrievalQuestion)
.retrievalSubQuestions(retrievalSubQuestions)
// 下面这组 selectedDocument* 字段表示"当前真正路由到的主文档作用域"。
.selectedDocumentId(routedDocumentId)
.selectedDocumentName(routedDocumentName)
.selectedTaskId(routedTaskId)
// 而 retrievalDocumentIds / retrievalTaskIds 表示"候选检索范围",可以比 selectedDocument* 更宽。
.retrievalDocumentIds(routedDocumentIds)
.retrievalTaskIds(routedTaskIds)
// 文档模式下的无证据回复需要按问题类型做定制提示,因此在这里覆盖 basePlan 的默认兜底文案。
.noEvidenceReply(buildDocumentModeNoEvidenceReply(question, requiresFreshSearch))
.build();
付费内容提示
该文档的全部内容仅对「JavaUp项目实战&技术讲解」知识星球用户开放
加入星球后,你可以获得:
- 超级八股文:100万+字的全栈技术知识库,涵盖技术核心、数据库、中间件、分布式等深度剖析的讲解
- 讲解文档:超级AI智能体、黑马点评Plus、大麦、大麦pro、大麦AI、流量切换、数据中台的从0到1的详细文档
- 讲解视频:超级AI智能体、黑马点评Plus、大麦、大麦pro、大麦AI、流量切换、数据中台的核心业务详细讲解
- 1 对 1 解答:可以对我进行1对1的问题提问,而不仅仅只限于项目
- 针对性服务:有没理解的地方,文档或者视频还没有讲到可以提出,本人会补充
- 面试与简历指导:提供面试回答技巧,项目怎样写才能在简历中具有独特的亮点
- 中间件环境:对于项目中需要使用的中间件,可直接替换成我提供的云环境
- 面试后复盘:小伙伴去面试后,如果哪里被面试官问住了,可以再找我解答
- 远程的解决:如果在启动项目遇到问题,本人可以帮你远程解决
进入星球后,即可享受上述所有服务,保证不会再有其他隐藏费用。
