1.RAG概念
1.什么是RAG
首先我们先理清楚什么是RAG(检索增强) 是一种信息检索技术结合AI进行内容生成的混合架构,可以有效解决大模型产生幻觉
简单说,当我们向大模型进行提问时,大模型可以从数据库中拿到相关性比较强的数据来进行结合回答,比如我们想要知道某个商品的某些详细数据,大模型不知道这些参数,就可以结合数据库中已经存在的数据进行结合回答。
2.RAG的作用
RAG技术的出现能够使得大模型问题会更加精确
1. 知识实时性增强
通过RAG改造后的大模型可以通过更新RAG的知识库来使得大模型能学习新的知识,例如回答"2023年世界杯冠军是谁?"时,RAG可直接检索最新结果,而非依赖训练数据(可能截止到更早时间)。
2. 减少幻觉(Hallucination)
- 基于事实生成:传统生成模型可能编造不存在的信息(如虚假数据、事件)。RAG通过检索真实来源提供依据,生成内容更可信
3. 处理长尾/专业领域问题
- 弥补训练数据不足:大模型在冷门或专业领域(如医学、法律)可能表现不佳。RAG通过检索领域特定知识库(如论文、法规),生成更专业的回答。
2.Spring AI RAG应用开发
2.1 基于阿里百联的向量数据库
2.1.1创建知识库
在百练平台创建一个知识库
知识库名称 是我们后续所需要写到代码中的

创建知识库前如果我们并没有添加任何文件内容,则需要先进入下一步时
点击数据中心添加数据

先添加类目


然后往该目录中倒入数据

上传完成后可以对对应的文档进行打标签
这样能更精准的召回所需要的内容

此时回到创建知识库
选择对应的泪目

metaData抽取一定要现在开启,不然创建后就不能开启了
选择智能切分,会调用ai大模型进行文档的切割,最大长度是指切分后的文章大小,重叠程度就可以强化上下文的联系

创建完成后我们就可以测试命中率

可以测试我们的输入,所能召回的被切割文档有哪些

2.1.2 Spring AI 结合 云知识库
java
package com.xiaog.aiapp.RAG;
import com.alibaba.cloud.ai.dashscope.api.DashScopeApi;
import com.alibaba.cloud.ai.dashscope.rag.DashScopeDocumentCloudReader;
import com.alibaba.cloud.ai.dashscope.rag.DashScopeDocumentRetriever;
import com.alibaba.cloud.ai.dashscope.rag.DashScopeDocumentRetrieverOptions;
import org.springframework.ai.chat.client.advisor.RetrievalAugmentationAdvisor;
import org.springframework.ai.chat.client.advisor.api.Advisor;
import org.springframework.ai.rag.retrieval.search.DocumentRetriever;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* 基于阿里百联平台自己搭建的知识库
*/
@Configuration
public class LoveAppRagCloudAdvisorConfig{
@Value("${spring.ai.dashscope.api-key}")
private String apiKey;
@Bean
public Advisor loveAppRagCloudAdvisor() {
DashScopeApi dashScopeApi = new DashScopeApi(apiKey);
// 初始化基于阿里百联的向量数据库
DocumentRetriever documentRetriever = new DashScopeDocumentRetriever(dashScopeApi
, DashScopeDocumentRetrieverOptions.builder()
.withIndexName("小G的知识库") //知识库的名称
.build());
//构建检索增强
return RetrievalAugmentationAdvisor.builder()
.documentRetriever(documentRetriever)
.build();
}
}
java
//注入Advisor
@Resource
private Advisor loveAppRagCloudAdvisor;
/**
* 基于云知识库 的对话
*/
public void dochatRag(String message , String sessionId){
ChatResponse chatResponse = chatClient.prompt()
.user(message)
.user(message)
.advisors(advisorSpec -> advisorSpec.param(CHAT_MEMORY_CONVERSATION_ID_KEY, sessionId) //根据会话id获取对话历史
.param(CHAT_MEMORY_RETRIEVE_SIZE_KEY, 10))//获取历史消息的条数
.advisors(loveAppRagCloudAdvisor)
.call()
.chatResponse();
System.out.println(chatResponse.getResult().getOutput().getText());
}
测试
java
@Test
void dochatRag() {
app.dochatRag("小明现在是单身,想在网上交友谈恋爱,请给他一点建议,精简总结一下", "1");
}
输出:

被召回的分片,通过自定义Advisor捕捉 到被一同传入
内容:

2.2 基于本地内存向量数据库
2.2.1 读取文档
创建一个读取文档的类,并且写一个读取文档的方法
我们此次所读取的文档是Markdown文档,所以编写一个读取markdown文档的方法
需要引入读取Markdown的依赖
<!--spring AI Markdown 文件读取--> <dependency> <groupId>org.springframework.ai</groupId> <artifactId>spring-ai-markdown-document-reader</artifactId> <version>1.0.0-M6</version> </dependency>
编写读取的类和方法
java
package com.example.demo.RAG;
import lombok.extern.slf4j.Slf4j;
import org.springframework.ai.document.Document;
import org.springframework.ai.reader.markdown.MarkdownDocumentReader;
import org.springframework.ai.reader.markdown.config.MarkdownDocumentReaderConfig;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.ResourcePatternResolver;
import org.springframework.stereotype.Component;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
@Component
@Slf4j
public class LoveAppDocumentLoader {
//文件解析
private final ResourcePatternResolver resourcePatternResolver;
LoveAppDocumentLoader(ResourcePatternResolver resourcePatternResolver) {
this.resourcePatternResolver = resourcePatternResolver;
}
public List<Document> loadMarkdownDocuments(){
List<Document> documentList=new ArrayList<>();
try {
// 这里可以修改为你要加载的多个 Markdown 文件的路径模式 读取资源目录下的所有 Markdown 文件
Resource[] resources = resourcePatternResolver.getResources("classpath:document/*.md");
//加载多篇Markdown文件
for (Resource resource : resources) {
String fileName = resource.getFilename(); // 获取文件名
//读取的配置设置
MarkdownDocumentReaderConfig config = MarkdownDocumentReaderConfig.builder()
.withHorizontalRuleCreateDocument(true)//添加水平线
.withIncludeCodeBlock(false)//添加代码块
.withIncludeBlockquote(false)//添加引用
.withAdditionalMetadata("filename", fileName) //添加(元信息)而外信息 //比如文件名 其他标签等
.build();
MarkdownDocumentReader reader = new MarkdownDocumentReader(resource, config);
//添加到集合
documentList.addAll(reader.get());
}
} catch (Exception e){
log.error("loadMarkdownDocuments error 文件读取失败",e);
}
return documentList;
}
}
2.2.2.加载文档到 基于内存的 向量数据库
将加载到的文档 进行 切割 、添加关键词 和摘要 再添加到向量数据库
所使用的 关键词 和文本摘要都需要涉及到大模型 ,所以也需要消耗对应的Token,在当前学习阶段用一次证明可以使用就能关掉了,否这token消耗大,且大模型回复较慢
java
package com.example.demo.RAG;
import jakarta.annotation.Resource;
import org.springframework.ai.chat.model.ChatModel;
import org.springframework.ai.document.Document;
import org.springframework.ai.embedding.EmbeddingModel;
import org.springframework.ai.transformer.KeywordMetadataEnricher;
import org.springframework.ai.transformer.SummaryMetadataEnricher;
import org.springframework.ai.transformer.splitter.TokenTextSplitter;
import org.springframework.ai.vectorstore.SimpleVectorStore;
import org.springframework.ai.vectorstore.VectorStore;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import java.util.List;
@Configuration
public class AppDocumentVectorStoreConfig {
@Resource
private LoveAppDocumentLoader loveAppDocumentLoader;
@Autowired
ChatModel chatModel;
@Bean
public VectorStore appVectorStore(EmbeddingModel dashScopeEmbeddingModel){
//初始化一个基于内存的向量数据库
SimpleVectorStore simpleVectorStore = SimpleVectorStore.builder(dashScopeEmbeddingModel).build();
//读取文挡
List<Document> documents = loveAppDocumentLoader.loadMarkdownDocuments();
//切分文档 文档按照Token进行切分 可以配置token数量 上下文token交集
TokenTextSplitter tokenTextSplitter = new TokenTextSplitter();
List<Document> documentsSplitter = tokenTextSplitter.apply(documents);
// 添加元数据 提取关键词生成器
KeywordMetadataEnricher keywordMetadataEnricher = new KeywordMetadataEnricher(chatModel,3);
List<Document> documentsMetaDataEnricher = keywordMetadataEnricher.apply(documentsSplitter);
//摘要生成器 生产 摘要
SummaryMetadataEnricher summaryMetadataEnricher = new SummaryMetadataEnricher(chatModel,
List.of(SummaryMetadataEnricher.SummaryType.PREVIOUS, //结合之前的文档
SummaryMetadataEnricher.SummaryType.CURRENT, //当前的文档
SummaryMetadataEnricher.SummaryType.NEXT)); //下一个文档
List<Document> documentList = summaryMetadataEnricher.apply(documentsMetaDataEnricher);
//添加到向量数据库
simpleVectorStore.add(documentList);
//返回向量数据库
return simpleVectorStore;
}
}
编写对话 接口
java
@Resource
private VectorStore appVectorStore;
public void dochatRag(String message , String sessionId){
ChatResponse chatResponse = chatClient.prompt()
.user(message)
.advisors(advisorSpec -> advisorSpec.param(CHAT_MEMORY_CONVERSATION_ID_KEY, sessionId) //根据会话id获取对话历史
.param(CHAT_MEMORY_RETRIEVE_SIZE_KEY, 10))//获取历史消息的条数
.advisors(new QuestionAnswerAdvisor(appVectorStore))
.call()
.chatResponse();
System.out.println(chatResponse.getResult().getOutput().getText());
}
编写测试
java
@Test
public void testLoveAppRag(){
app.dochatRag("我是小明 我是单身线上交友有哪些注意事项能提高脱单成功率,简略的回答一下","1");
app.dochatRag("有没有相应的课程推荐?","1");
}
结果:
