Day 92:【99天精通Python】终极项目 - AI 聊天机器人 (中) - 知识库与 RAG

Day 92:【99天精通Python】终极项目 - AI 聊天机器人 (中) - 知识库与 RAG

前言

欢迎来到第92天!

在昨天的课程中,我们搭建了一个带记忆的流式聊天 API。但是,这个 AI 只能基于它自身的通用知识来回答问题。如果我们想让它成为一个特定领域的专家(比如,回答关于我们公司内部文档的问题),就需要给它挂载"外脑"。

这就是RAG (Retrieval-Augmented Generation)的用武之地。
我们将实现一个文件上传功能,让用户可以上传 PDF,然后 AI 就能基于这个 PDF 的内容来回答问题。

本节内容:

  • RAG 流程与 LangChain 实现
  • 文件上传 API
  • 创建并持久化向量数据库
  • 改造聊天 API,支持 RAG
  • 实战测试:基于 PDF 的问答

一、文件处理与向量化

我们需要一个 API 来接收用户上传的文件,并将其处理成向量存起来。

1.1 文件上传接口 (app.py)

importosfromflaskimportrequest,jsonifyfromwerkzeug.utilsimportsecure_filenamefromlangchain_community.document_loadersimportPyPDFLoaderfromlangchain_text_splittersimportRecursiveCharacterTextSplitterfromlangchain_openaiimportOpenAIEmbeddingsfromlangchain_community.vectorstoresimportChroma UPLOAD_FOLDER='./uploads'DB_FOLDER='./vector_db'app.config['UPLOAD_FOLDER']=UPLOAD_FOLDERifnotos.path.exists(UPLOAD_FOLDER):os.makedirs(UPLOAD_FOLDER)ifnotos.path.exists(DB_FOLDER):os.makedirs(DB_FOLDER)# 初始化 Embedding 模型 (只需一次)embeddings=OpenAIEmbeddings(api_key="...",base_url="...")@app.route("/api/upload",methods=["POST"])defupload_file():if'file'notinrequest.files:returnjsonify({"error":"缺少文件"}),400file=request.files['file']session_id=request.form.get("session_id","default_session")iffile.filename=='':returnjsonify({"error":"未选择文件"}),400# 1. 保存文件filename=secure_filename(file.filename)filepath=os.path.join(app.config['UPLOAD_FOLDER'],filename)file.save(filepath)# 2. 加载和切分loader=PyPDFLoader(filepath)docs=loader.load()text_splitter=RecursiveCharacterTextSplitter(chunk_size=1000,chunk_overlap=200)splits=text_splitter.split_documents(docs)# 3. 向量化并存入 ChromaDB# 每个 session 有自己独立的数据库db_path=os.path.join(DB_FOLDER,session_id)vectorstore=Chroma.from_documents(documents=splits,embedding=embeddings,persist_directory=db_path)returnjsonify({"message":f"文件 '{filename}' 上传并处理成功"}),200

二、改造聊天逻辑:双模式切换

现在,我们的聊天 API 需要能够判断:当前会话是否有关联的知识库?

  • :走 RAG 流程。
  • 没有:走昨天的普通对话流程。

2.1 构建 RAG 链

我们需要一个更复杂的链,它能结合历史记录和知识库。

# app.py 中fromlangchain.chainsimportcreate_history_aware_retriever,create_retrieval_chainfromlangchain.chains.combine_documentsimportcreate_stuff_documents_chainfromlangchain_core.promptsimportChatPromptTemplate,MessagesPlaceholderdefget_rag_chain(session_id):"""根据 session_id 获取 RAG 链"""db_path=os.path.join(DB_FOLDER,session_id)ifnotos.path.exists(db_path):returnNone# 没有知识库# 加载向量数据库vectorstore=Chroma(persist_directory=db_path,embedding_function=embeddings)retriever=vectorstore.as_retriever()llm=ChatOpenAI(api_key="...",base_url="...",streaming=True)# 1. 重构问题 Prompt (让AI把历史和新问题整合成一个独立问题)contextualize_q_prompt=ChatPromptTemplate.from_messages([("system","根据聊天历史和最新问题,生成一个无需历史即可理解的独立问题。"),MessagesPlaceholder("chat_history"),("user","{input}"),])history_aware_retriever=create_history_aware_retriever(llm,retriever,contextualize_q_prompt)# 2. 问答 Promptqa_prompt=ChatPromptTemplate.from_messages([("system","你是一个问答助手。请根据下面的上下文回答问题:\n\n{context}"),MessagesPlaceholder("chat_history"),("user","{input}"),])question_answer_chain=create_stuff_documents_chain(llm,qa_prompt)# 3. 组装最终的 RAG 链rag_chain=create_retrieval_chain(history_aware_retriever,question_answer_chain)returnrag_chain

2.2 升级流式生成器

# app.pyfromlangchain_core.messagesimportHumanMessage,AIMessagedefstream_generator(session_id,user_input):# 获取或创建记忆ifsession_idnotinconversations:conversations[session_id]=ConversationBufferMemory(return_messages=True)memory=conversations[session_id]# 尝试获取 RAG 链rag_chain=get_rag_chain(session_id)ifrag_chain:print(f"[{session_id}] 使用 RAG 模式")# RAG 链需要 chat_history 对象chat_history=memory.load_memory_variables({})['history']asyncdefrag_stream():# RAG 链的流式调用asyncforchunkinrag_chain.astream({"input":user_input,"chat_history":chat_history}):if"answer"inchunk:yieldf"data:{json.dumps({'token':chunk['answer']})}\\n\n"# 手动更新记忆memory.save_context({"input":user_input},{"output":"..."})# 这里可以后续优化returnrag_stream()else:print(f"[{session_id}] 使用普通对话模式")# 普通对话链 (同 Day 91)# ... (省略)

三、测试 RAG 流程

  1. 启动 Flask:python app.py

  2. 上传文件:

    curl-X POST -F"file=@/path/to/your.pdf"-F"session_id=pdf_session"http://127.0.0.1:5001/api/upload
  3. 基于文档提问:

    curl-X POST -H"Content-Type: application/json"\-d'{"session_id": "pdf_session", "input": "总结一下这份文档的主要内容"}'\http://127.0.0.1:5001/api/chat

    你会看到 AI 的回答是基于你上传的 PDF 内容的。

  4. 普通聊天:

    curl-X POST -H"Content-Type: application/json"\-d'{"session_id": "normal_session", "input": "你好"}'\http://127.0.0.1:5001/api/chat

    这次 AI 会进行普通回复。


四、常见问题

Q1:create_history_aware_retriever是干嘛的?

如果用户问"它怎么样?“,这个链会结合历史记录(比如上一句是"我正在看《三体》”),把问题重构成"《三体》怎么样?",然后再去知识库里检索。

Q2:Chroma.from_documents很慢怎么办?

Embedding 计算是耗时的。对于大文件,不应该在 HTTP 请求中同步处理。
生产环境方案:文件上传后,把"处理任务"丢给Celery (Day 64)后台 Worker 去慢慢执行。

Q3:如何支持 Word/Txt 文件?

LangChain 提供了各种DocumentLoader

  • TextLoader
  • UnstructuredWordDocumentLoader(需pip install unstructured)

五、小结

聊天 /chat

用户问题

有知识库?

RAG 链

普通对话链

LLM

流式回复

上传 /upload

上传 PDF

切分

Embedding

存入 VectorDB

关键要点

  1. RAG是给 AI 扩展私有知识的"外挂"。
  2. 向量数据库是 RAG 的核心,负责存储和检索知识片段。
  3. 通过session_id可以为每个用户或每个文档创建独立的知识库。

六、课后作业

  1. 前端进度条from_documents过程是耗时的,思考如何在前端展示文件处理的进度?(提示:后端可以返回一个task_id,前端轮询查询任务状态)。
  2. 多文档支持:允许一个session_id关联多个上传的文档。
  3. 元数据过滤 (进阶):在切分文档时,给每个split添加元数据(如metadata={'source': 'doc1.pdf'}),在检索时可以进行过滤(只在doc1.pdf中搜索)。

下节预告

Day 93:AI 聊天机器人 (下) - 前端界面与部署- 万事俱备,只欠东风!明天我们将编写 HTML 和 JavaScript,打造一个真正的 Web 聊天界面,并完成项目的最终部署。


系列导航

  • 上一篇:Day 91 - AI聊天机器人(上)
  • 下一篇:Day 93 - AI聊天机器人(下)(待更新)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/1183503.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

零基础网络安全高效入门:核心就学这些,边练边学快速上手

目录 一、什么是网络安全 1.1 网络安全的定义:1.2 信息系统(Information System)1.3 信息系统安全三要素(CIA)1.4 网络空间安全1.5 国家网络空间安全战略1.6 网络空间关注点1.7 网络空间安全管理流程 二、网络安全术语…

通俗解释UDS诊断中31服务的三步控制流程

深入浅出:UDS 31服务的三步控制逻辑,如何精准操控ECU内部“隐藏功能”?你有没有遇到过这样的场景——一辆车在产线下线时需要自动完成电机校准,维修站里技师要手动触发某个传感器的自检程序,或者OTA升级前系统得先确认…

避免QTimer内存泄漏:入门阶段需要注意的问题

避免 QTimer 内存泄漏:新手最容易忽略的“小定时器”大问题你有没有遇到过这样的情况?一个看似简单的 Qt 应用,运行几个小时后内存越占越多,界面越来越卡,最后干脆崩溃退出。查了一圈代码,没发现哪里在疯狂…

新手教程:应对Keil5菜单及对话框中文乱码

手把手解决Keil5中文乱码:从界面乱码到文件路径全修复你有没有遇到过这种情况?打开Keil5,菜单栏突然变成一堆“锟斤拷”、“鑿鋮濠”,工程名字显示为方框,甚至连文件路径都看不清了——明明系统是中文的,为…

提升<|关键词|>效率:精准检索学术资源的实用技巧与工具推荐

做科研的第一道坎,往往不是做实验,也不是写论文,而是——找文献。 很多新手科研小白会陷入一个怪圈:在知网、Google Scholar 上不断换关键词,结果要么信息过载,要么完全抓不到重点。今天分享几个长期使用的…

2026年靠谱的,招聘背景调查,招才背调公司用户优选名录 - 品牌鉴赏师

引言在当今竞争激烈的职场环境中,企业为了降低用工风险、提升招聘质量,对员工背景调查的需求日益增长。招聘背景调查作为企业招聘流程中的关键环节,能够帮助企业全面了解候选人的真实情况,确保招聘到合适的人才。然…

2026国内最新组合螺丝生产厂家最新top5排行榜发布!广东等地优质组合螺丝/端子螺丝/螺丝定制/螺丝加工公司及供应商综合实力盘点,助力电子电器_新能源_汽车配件领域高效生产. - 品牌推荐2026

随着电子电器、新能源及汽车配件行业的快速发展,组合螺丝作为关键连接部件,其品质稳定性与供应效率直接影响终端产品的装配质量与生产节奏。据中国紧固件工业协会2025年度行业报告显示,国内组合螺丝市场规模突破300…

公司想辞退的你六种表现,你get到了么?

对于想辞退你的公司,碍于《劳动合同法》的威力,大多数单位都不会去走直接裁员的艰苦道路,而是利用一些手段辞退,甚至让员工自己辞职,以达到降低人员成本的目的。以下就是最常见的六种方式。1调换工作调岗:调…

DeepSeek-R1-Distill-Qwen-1.5B推荐部署方式:Ollama一键拉取实战体验

DeepSeek-R1-Distill-Qwen-1.5B推荐部署方式:Ollama一键拉取实战体验 1. 背景与技术定位 随着大模型轻量化趋势的加速,如何在有限算力条件下实现高性能推理成为边缘计算和本地化部署的核心挑战。DeepSeek-R1-Distill-Qwen-1.5B 正是在这一背景下诞生的…

阿里Qwen3-4B-Instruct-2507避坑指南:部署常见问题全解

阿里Qwen3-4B-Instruct-2507避坑指南:部署常见问题全解 1. 引言 1.1 背景与需求 随着端侧AI的快速发展,轻量级大模型在本地设备上的部署成为开发者关注的核心方向。阿里通义千问团队推出的 Qwen3-4B-Instruct-2507 凭借40亿参数实现了对部分百亿级闭源…

实时降噪技术落地利器|FRCRN-16k大模型镜像详解

实时降噪技术落地利器|FRCRN-16k大模型镜像详解 1. 引言:语音降噪的工程化挑战与突破 在智能语音设备、远程会议系统和移动通信场景中,环境噪声始终是影响语音质量的核心障碍。传统降噪算法受限于固定滤波参数和有限的非线性建模能力&#…

学术搜索入口:快速查找学术资源的便捷通道

做科研的第一道坎,往往不是做实验,也不是写论文,而是——找文献。 很多新手科研小白会陷入一个怪圈:在知网、Google Scholar 上不断换关键词,结果要么信息过载,要么完全抓不到重点。今天分享几个长期使用的…

从零到一:通向CISP安全工程师的网络安全入门完全指南(附学习清单)

目录 一、什么是网络安全 1.1 网络安全的定义:1.2 信息系统(Information System)1.3 信息系统安全三要素(CIA)1.4 网络空间安全1.5 国家网络空间安全战略1.6 网络空间关注点1.7 网络空间安全管理流程 二、网络安全术语…

DeepSeek-R1-Distill-Qwen-1.5B参数压缩:结构化剪枝技术

DeepSeek-R1-Distill-Qwen-1.5B参数压缩:结构化剪枝技术 1. DeepSeek-R1-Distill-Qwen-1.5B模型介绍 DeepSeek-R1-Distill-Qwen-1.5B是DeepSeek团队基于Qwen2.5-Math-1.5B基础模型,通过知识蒸馏技术融合R1架构优势打造的轻量化版本。其核心设计目标在于…

Qwen-Image-2512部署费用高?Spot实例降本实战指南

Qwen-Image-2512部署费用高?Spot实例降本实战指南 1. 背景与痛点:大模型推理成本的现实挑战 随着多模态生成模型的快速发展,Qwen-Image-2512作为阿里云开源的最新图像生成模型,在分辨率、细节表现和语义理解能力上实现了显著提升…

2026管束抽芯机厂家权威推荐榜单:液压抽芯机/换热器抽芯机/液压遥控抽芯机/新型抽芯机/换热器管束抽芯机源头厂家精选。

在石化、电力、冶金等流程工业中,换热器是保障生产连续性的核心设备。据统计,2025年国内换热器市场规模已突破1200亿元,其配套的维护与检修设备需求随之显著增长。作为检修作业中的关键装备,抽芯机的性能直接决定着…

面试官问:生成订单30分钟未支付,则自动取消,该怎么实现?

今天给大家上一盘硬菜,并且是支付中非常重要的一个技术解决方案,有这块业务的同学注意自己试一把了哈!在开发中,往往会遇到一些关于延时任务的需求。例如生成订单30分钟未支付,则自动取消生成订单60秒后,给…

Java面试题目收集整理归纳(2026年持续更新)

开始的碎碎念 本文大多是各大企业的topN题目,针对java中高级开发,本文会持续收集更新内容,如果大家有优质的Java面试题,也欢迎大家来投稿。 特意整理出这个分类的目录,方便大家平时复习和收藏哈。希望正在准备面试的…

本科生必备的毕业论文选题攻略,附热门平台Top10详细排名

10大论文选题工具核心对比 排名 工具名称 核心功能 效率评分 适用场景 1 aicheck 智能选题大纲生成 ★★★★★ 完全无头绪时的选题生成 2 aibiye 选题优化可行性分析 ★★★★☆ 已有初步方向的优化调整 3 知网 学术资源库选题参考 ★★★★☆ 专业领域深度…

基于DeepSeek-OCR-WEBUI的多语言文本识别技术实践

基于DeepSeek-OCR-WEBUI的多语言文本识别技术实践 1. 引言:复杂场景下的OCR新范式 随着企业数字化进程加速,传统光学字符识别(OCR)技术在面对扫描件模糊、版面复杂、多语言混排等现实问题时逐渐暴露出准确率低、结构化能力弱的短…