目前企业级 AI 需求最高频的项目究竟是什么?毫无疑问是:“让 AI 学习我几百页的 PDF 说明书或者行业研报,然后帮我回答里面的问题。”
其实打通这条 pipeline(流水线)的技术并不像想象的那么神秘。抛开各种复杂的中间件框架,本质上它只有一条直线。今天我们将这套链路中最核心的前半段——“文档的读取与向量入库”,用大白话和最纯粹的方式拆解出来。
RAG 的上半场:数据流水线
要让大模型在一堆 PDF 里找答案,前提是你要先把 PDF 揉碎喂给数据库。这称为数据摄入(Ingestion),也就是我们在示例中 pdf_to_vector_demo.py(见文末) 所做的事。分为三步:
第 1 步:解析 PDF (Extract)
传统的 PDF 是非结构化的。我们要用类似 PyMuPDF (fitz) 或者 pdfplumber 的工具,把它的一页页内容无情地榨成纯字符串。在这个阶段,如果有图表是很难提取的,纯文字相对容易。
第 2 步:文本切片 (Chunking)
PDF 转出来的字符串如果有 5 万字,千万不能直接一股脑扔进向量库!你要把它“切小块”。这里我们通常用滑动窗口法:把文章切成许多 500 字的小块,并且每一块和上一块保留约 50 个字的重叠 (Overlap),这样是为了防止刚好某个跨块的关联词被斩断。
第 3 步:向量化与落库 (Embedding & Store)
这步就是让 AI 走马上任。你每一块 500 字的文章,都要过一遍 Embedding 模型(比如阿里的 bge-m3、智谱的 glm-embedding 或者 Chroma 默认模型)。它们会吐出一个比如维度长达 1024 的浮点数数组。 紧接着,你把这个“带数字坐标”的豆腐块,加上它的出处:“第几页”、“来源哪个 PDF 的名字”等作为元数据(Metadata),一同插入到向量数据库(比如本地的 ChromaDB)中。
为什么要存元数据(Metadata)?
想象一下,某一天有人问:“2023年某公司的财报里提到了什么营收?” 通过元数据,你可以在底层直接加一成过滤(Filter):只在 年份=2023 且 类型=财报 的块空间里去做数学搜索,这将大幅度和根本性地杜绝大模型在别的破旧或者不相关的文件里产生幻觉!
当你做完了上面这 3 步,恭喜你,你的私有知识已经化作漫天繁星的坐标点,正静静地躺在向量空间里,随时准备响应你的问答了。
pdf_to_vector_demo.py(主要代码)
import osimport sysimport mathdef main():print("=" * 60)print("📚 [DEMO] RAG: PDF电子书切块与向量化提取演示")print("=" * 60)# 模拟准备一个 PDF 文件。实际运行时可替换为真实的电子书路径sample_pdf = os.path.join(os.path.dirname(__file__), "sample_ebook.pdf")if not os.path.exists(sample_pdf):print("\n⚠️ 当前目录没有找名为 sample_ebook.pdf 的文件。")print("由于这是演示脚本,请先将你的一本电子书(重命名为 sample_ebook.pdf)放置在: ")print(f"👉 {sample_pdf} \n然后再运行此脚本。")print("\n或者你也可以直接在下面代码中修改 sample_pdf 的路径!\n")# 为了演示继续运行,我们可以造两段假文本替代print("--- ⬇️ 降级为直接使用假文本进行演示 ---")full_text = "这是一部分用来测试的虚拟文本。在这个示例中,假设我们成功从PDF解析出了这部分文本:\n\n人工智能是一种研究... 我们希望让机器拥有智能的判断。"else:# 第一步:提取 PDF 文本full_text = extract_text_from_pdf(sample_pdf)if not full_text.strip():print("❌ 提取到的文本为空,可能是扫描版图片 PDF,需要 OCR 支持。")return# 第二步:将这本电子书切分成多个 Chunk(切块)chunks = chunk_text(full_text, chunk_size=300, overlap=50)# 第三步:利用大模型 API 把 Chunks 转成高维向量 (Embeddings)# 存入到内存版的“向量数据库” (List)vector_db = generate_embeddings_for_chunks(chunks)if not vector_db:print("❌ 向量库生成失败或为空,退出演示。")return# 第四步:模拟向量检索(针对已经向量化的内容提问测试)print("\n💡 提示: 演示中你可结合 Chunk内容自定义提问。")test_query = "什么是人工智能?" # 你可以修改为真实语境下的问题retrieve_similar_chunk(test_query, vector_db)if __name__ == "__main__":main()
本文是个人学习大模型与向量数据库流水线时的实战小记,希望对准备入门 AI RAG 开发的大家有所启发。
感谢关注,我会持续更新,欢迎查看完整的代码实现与学习记录:https://github.com/start007-smart/ai-learning
夜雨聆风