Langchain-实战篇-搭建本地问答机器人-01

f146ed0fe16c4b51a17db0a7e247ed3b.png

项目背景

在这个快节奏的职场世界里,我们每个人都可能在某个时刻,化身为一头辛勤的牛或一匹奔腾的马,面对入职签合同时的迷茫,或是离职时的纠纷,心中难免会涌现出各种疑问。比如:

  • "这份合同里的条款,是保护我还是给我挖坑?"
  • "加班费怎么算,难道是按草料的重量来计价的吗?"
  • "离职时的赔偿金,是不是该按马的奔跑速度来计算?"

别担心,就算我们不是法律专家,也能在《中华人民共和国劳动法》的庇护下,找到解决问题的钥匙。现在,让我们用一点幽默来轻松一下,同时开发一款劳动法问答机器人,让它成为我们职场路上的得力助手。

实施方案

简单来讲,我们的计划是将《中华人民共和国劳动法》的内容向量化,并存储到向量化数据库中。在用户提出与劳动法相关的问题时,将问题检索向量数据库,通过提示工程组装交给大模型进行回复,最终得到最终答案。

模型选择:GPT-3.5 Turbo

下面为开发这个项目的流程说明以及详解:

e840f26dcab54438b5d9a965f9fe1ef8.png

1. 数据加载 (Loading)

  • 描述: 将《中华人民共和国劳动法》的法律文档加载到本地系统中。
  • 目的: 使文档变成LangChain能够识别和处理的格式。

2. 文本切割 (Split)

  • 描述: 将大型法律文档切割成指定大小的文档块。
  • 目的: 便于后续的存储和高效检索。

3. 向量化存储 (Storage)

  • 描述: 对切割后的文档块进行向量化处理。
  • 目的: 将文档块转换为LangChain可以索引的向量形式,并存储到向量数据库中。

4. 数据检索 (Retrieval)

  • 描述: 用户提出问题后,在向量数据库中检索相关的文档块。
  • 目的: 快速找到与用户问题最相关的法律条文。

5. 提示工程(Prompt)

  • 描述: 组装用户输入的问题,以及检索后的文档快。
  • 目的: 为LLM提供数据。

在整个项目开发过程中,LangChain平台提供了一系列工具和组件,极大地简化了开发流程,使我们能够轻松实现各项功能。

环境配置

安装python 3.11版本

网上已经有很多教程了,这里就不累赘了

安装langchain以及其它会用到的包

pip install langchain==0.2.6pip install bs4pip install langchain_communitypip install langchain_openai

step1 数据处理

1、数据加载  Loading

准备数据,《中华人民共和国劳动法》内容链接:中华人民共和国劳动法

Langchain提供了很多文档加载器,以面对各种类型的文档。本次我们使用的WebBaseLoader

详细使用方法请看我文章中第二小节

03|LangChain | 从入门到实战 -六大组件之Retrival_longcontext-CSDN博客

当然你也可以看官方文档:WebBaseLoader | 🦜️🔗 LangChain

我们来观察一下数据,可以看到所有条目都是位于div class="TRS_PreAppend"

91c762c7d4c045669ace43957524ca60.gif

8ac867390c5c4eb1829be2744dbe6e90.png

加载代码

import bs4
from langchain_community.document_loaders import WebBaseLoader
import os
os.environ["OPENAI_API_KEY"] = 'openAi key'
os.environ["OPENAI_API_BASE"] = "代理"loader = WebBaseLoader(web_path="https://www.mohrss.gov.cn/SYrlzyhshbzb/zcfg/flfg/201601/t20160119_232110.html",bs_kwargs=dict(parse_only=bs4.SoupStrainer(class_=("TRS_PreAppend")))
)
docs = loader.load()
print()

打上断点可以发现,此时得到了一个 big documents 

Langchain工具中的,WebBaseLoader 将 url网页的代码配合bs4 提取 html数据中 div 标签属性为TRS_PreAppend的数据。

d496e29926374703b016ac5f2ee9fcc6.png

2、切割数据 Split

现在我们有了数据,但是是一个非常大的文本内容。

或许你有疑问,为啥要切割呢?

在我以往的文章中也讲过,LLM对于一次对话的token是有限制的。当然现在也有许多非常优秀的大模型甚至可以做到一次对话128k,但是本次我们使用的模型为 GPT-3.5 Turbo ,所以还是需要切割的。

所以需要需要切割成小块的 chunk documents,以提交给LLM,才能进行问答。

同样的Langchain也提供了非常多的切割方法。

03|LangChain | 从入门到实战 -六大组件之Retrival_longcontext-CSDN博客我在这个小节有介绍常用的

0868f345dc0e4892b170976380e0fa53.png

Langchain官方介绍地址:How-to guides | 🦜️🔗 LangChain

17f284df1c8a4c6fbc40e56ddb9cdff2.png

在这里我们使用一个最基本的切割器:RecursiveCharacterTextSplitter

此文本分割器是推荐用于一般文本的分割器。它由字符列表参数化。它会尝试按顺序分割这些字符,直到块足够小。默认列表是["\n\n", "\n", " ", ""]。这样做的目的是尽量将所有段落(然后是句子,然后是单词)放在一起,因为这些段落在一般情况下似乎是语义上最相关的文本片段。

编写切割代码

from langchain.text_splitter import RecursiveCharacterTextSplitter
text_splitter = RecursiveCharacterTextSplitter(chunk_size=1000, chunk_overlap=200
)
splits = text_splitter.split_documents(docs)
  • chunk_size:指定切割的每个块的大小(以字符为单位)
  • chunk_overlap:指定块之间的重叠部分的大小(以字符为单位)

这个是指切割后的每个 document 里包含几个上一个 document 结尾的内容, 主要作用是为了增加每个 document 的上下文关联。

比如,chunk_overlap=0时, 第一个 document 为 aaaaaa,第二个为 bbbbbb;

当 chunk_overlap=2 时, 第一个 document 为 aaaaaa,第二个为 aabbbbbb。

不过,这个也不是绝对的,要看所使用的那个文本分割模型内部的具体算法。

我们来看看切割后的内容,得到11个块

f03b4d8cecc8487299bf682ec58b098b.png

3、向量化存储 Vector

在当前项目中,我们已经成功解析并拆分获得了11个数据块。为了提高检索效率,避免在每次检索时重新进行解析操作,我们将这些数据块存储到数据库中,以便在需要时能够直接从数据库中快速查询。

我们选择使用向量数据库来存储和查询这些数据块。向量数据库的工作原理是将数据项转换成向量形式,这使得它在处理相似性搜索和聚类等任务时表现出色,具有高效率。

设想一下,如果我们有一系列复杂的机械零件,每个零件都有其独特的属性,如耐用性、重量、尺寸和材料类型。向量数据库将这些属性转化为数值向量,例如[(7, 2, 5, 3), (6, 3, 4, 2)],每个元素代表该零件在某个特定属性上的评分或量度。当我们需要找到与给定零件相似的其他零件时,向量数据库会先将该零件的属性转换为向量,然后在数据库中搜索并返回与之最相似的其它零件的向量。

为了将数据块存入向量数据库,我们首先需要对这些数据进行向量化处理,也就是常说的“数据嵌入”。在这个过程中,LangChain 提供了多种嵌入技术,我们选择了 OpenAIEmbeddings。这种嵌入方式利用 OpenAI 的 text-embedding-ada-002 模型来实现数据的向量化。在使用此嵌入方式之前,我们需要设置 OPENAI_API_KEY,这是调用 OpenAI 服务所需的密钥。

通过这种方式,我们不仅优化了数据的存储和检索过程,还为系统增添了强大的语义理解能力,确保了在处理复杂查询时的准确性和效率。

向量数据库有很多,需要了解可以跳转到这个地方查看:Vector stores | 🦜️🔗 LangChain

这里我们选择使用Chroma,更多用法:Chroma | 🦜️🔗 LangChain

安装

pip install langchain-chroma

现在,我们编写一个代码,将前面切割好的文档,存储到向量数据库中

from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings
db = Chroma.from_documents(documents=splits,embedding=OpenAIEmbeddings(),persist_directory="./chroma_db")
  • documents:上一步切割后的文档块列表。
  • embedding:嵌入函数,这里我们使用 OpenAI 的OpenAIEmbeddings。
  • persist_directory:数据存储的目录。

执行代码后,会在项目目录下自动生成了chroma_db目录,存放着向量化后的数据

7d86cefb61d5448798a7d770a263031e.png

4、数据处理完整代码

到现在,我们完成了从数据获取,到切割,存储到向量数据库的步骤,此刻贴上 etl_datas.py完整代码

import bs4
from langchain_community.document_loaders import WebBaseLoader# 配置OpenAi密匙
import os
os.environ["OPENAI_API_KEY"] = 'openai key'
os.environ["OPENAI_API_BASE"] = "代理地址"# 构建加载器,加载URL数据
loader = WebBaseLoader(web_path="https://www.mohrss.gov.cn/SYrlzyhshbzb/zcfg/flfg/201601/t20160119_232110.html",bs_kwargs=dict(parse_only=bs4.SoupStrainer(class_=("TRS_PreAppend")))
)
docs = loader.load()# 切割文档
from langchain.text_splitter import RecursiveCharacterTextSplitter
text_splitter = RecursiveCharacterTextSplitter(chunk_size=1000, chunk_overlap=200)
splits = text_splitter.split_documents(docs)# 向量化数据
from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings
db = Chroma.from_documents(documents=splits,embedding=OpenAIEmbeddings(),persist_directory="./chroma_db")

step2 创建问答机器人

a48009aefc9347ccba19aeb43d09cb3c.png

现在我们要开始构建第二步,创建问答机器人。

这里运用Langchain当中的模块清单:

  1. Retrieval 检索器,Langchain集成了与向量数据库 增删改查的封装

  2. Chain  我在文章当中有详细介绍 04|LangChain | 从入门到实战 -六大组件之chain-CSDN博客b41a090847ae4657b160f57b78b34420.png

  3. Prompt  提示模版工程,让LLM获悉外部的数据,以解决某些任务,得到具体结果行为
  4.  LangServe  LangServe帮助开发人员将可运行LangChain 对象和链部署 为 REST API。更多内容参考:GitHub - langchain-ai/langserve: LangServe 🦜️🏓

总的来说,Chain 帮助我们串联整个流程,并且利用 LangServe,快速构建交互API,配合web网页进行操作。

1、创建提示 Prompt

你可以理解 Prompt是与LLM沟通的方式,Prompt编写的好坏决定了最后的效果,关于Prompt也有

提示词是一门大学问,可以阅读一下这个地址:🟢 AI 简介

这里我们运用RAG工程

RAG(Retrieval Augmented Generation)检索增强生成,通过检索增强大模型的生成能力。

079ea8b26a2646f794365165d7b6c100.png

参考论文:https://arxiv.org/abs/2005.11401

假设一位你需要从繁杂的《中华人民共和国劳动法》中找到相关的法律条文来协助处理劳动争议或合同问题,那么他有三种方式可以使用:

  • 最原始的方法:他可以直接翻阅纸质版的《中华人民共和国劳动法》,或者查询其电子版,然后仔细阅读以掌握具体的法律条文和操作方法。如果他遇到的劳动问题较为复杂,就需要综合多个章节的内容,自行分析和理解法律条文的适用。

  • 借助问答机器人:他也可以向问答机器人(chatbot)咨询,机器人能够提供相关的法律知识。但这种方法可能存在两个问题:一是机器人可能只能提供类似常见问题解答(FAQ)式的服务,需要用户自己整合答案;二是机器人需要大量的预训练和知识库的构建,这通常需要专业法律人员的大量工作,逐条输入法律条目及其相似问法,这在大规模推广时可能不太实用。

  • 使用RAG技术:第三种方式是将《中华人民共和国劳动法》的电子版上传到RAG(Retrieval-Augmented Generation)系统中,系统能在几分钟内创建索引,供专员进行咨询。RAG系统不仅能提供综合了整个法律文本的多个知识点的答案,还能以专家的口吻给出建议:“要解决这个问题,你需要先确认两个前提条件,并有三种可能的解决途径。下面我们逐步分析具体的操作步骤…”。

通过RAG技术,你可以能够快速获得全面、综合的法律建议,极大地提高了工作效率和解决问题的准确性。这种方法不仅减少了查找和分析法律条文的时间,还为用户提供了一种更为智能和高效的法律咨询服务。

下面我们使用Langchain提供的一个最基本的RAG 提示

"""
You are an assistant for question-answering tasks. Use the following pieces of retrieved context to answer the question. If you don't know the answer, just say that you don't know. Use three sentences maximum and keep the answer concise.

Question: {question} 

Context: {context} 

Answer:
"""
 

翻译成中文

你是问答任务的助手。使用以下检索到的上下文来回答问题。如果你不知道答案,就说你不知道。最多使用三个句子并保持答案简洁。
问题: {question}
上下文: {context}
答案:

这里我们使用Langhcain最基础的模版 PromptTemplate ,可以看到下面有2个参数

  1. {question}  将用户传入的问题进行填充
  2. {context}  向量数据库检索出来的数据
from langchain.prompts.prompt import PromptTemplateprompt_template_str = """
You are an assistant for question-answering tasks. Use the following pieces of retrieved context to answer the question. If you don't know the answer, just say that you don't know. Use three sentences maximum and keep the answer concise.Question: {question} Context: {context} Answer:
"""
prompt_template = PromptTemplate.from_template(prompt_template_str)

我在我的文章中有详细介绍:02|LangChain | 从入门到实战 -六大组件之Models IO_langchain编程:从入门到实践-CSDN博客

c3ddfa6b171644e8a57fd7496f7f8838.png

2、检索向量数据库

Langchain集成了与向量数据库 增删改查的封装,开箱即用。

前面我们将数据存储在本地db中,现在,我们利用Langchain查询一下DB,看看效果。

# 配置OpenAi密匙
import osos.environ["OPENAI_API_KEY"] = ''
os.environ["OPENAI_API_BASE"] = ""from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings# 链接本地向量数据库
vectorstore = Chroma(persist_directory="./chroma_db",embedding_function=OpenAIEmbeddings()
)# 构建检索器
retriever = vectorstore.as_retriever(search_type="similarity",search_kwargs={"k": 4}
)# 查询
docs = retriever.invoke("劳动者解除劳动合同需要提前多久告诉人事?")print()

在上面代码中使用的是 OpenAIEmbeddings() 向量模型

并且利用检索器的相似度算法,去查询检索出4条可能相关的知识,并且我们看第一个检索出来的数据,可以看到答案就在这里

72706f3488a5405f9f98af321ae9cee2.png

3、创建Chain

现在我们已经得到了 检索出来的知识,现在我们要将前面流程通过Langchain提供的Chain串联起来。

在此之前,我们先安装一个包,用于后面生成Chain的图

pip install grandalf

其中这个包会比较麻烦

pip install pygraphviz

如果在你安装 pygraphviz 这个包报错了,请参考我另外一个方法

GitHub - Pomurnik/pygraphviz-windows-whl: Wheels for pygraphviz Python libary with Graphviz 3.0 version.

3e26089a56df45b287f3e50247733984.png

因为我们是python3.11 环境,所以选择这个,点击下载

82f39f95ffb240b59df3db30b864c6f9.png

下载好的文件复制到工程下,进入到这个目录执行安装即可

pip install .\pygraphviz-1.12-cp311-cp311-win_amd64.whl

a55cbd4ced93438e975f30273cceb4b5.png

现在运行我这个代码

# 配置OpenAi密匙
import osos.environ["OPENAI_API_KEY"] = ''
os.environ["OPENAI_API_BASE"] = ""from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings# 链接本地向量数据库
vectorstore = Chroma(persist_directory="./chroma_db",embedding_function=OpenAIEmbeddings()
)# 构建检索器
retriever = vectorstore.as_retriever(search_type="similarity",search_kwargs={"k": 4}
)# 查询
# docs = retriever.invoke("劳动者解除劳动合同需要提前多久告诉人事?")# 将前面查询出来的结果,提取成一个大的文本
def format_docs(docs):return "\n\n".join(doc.page_content for doc in docs)# 构建RAG链
from langchain.prompts.prompt import PromptTemplateprompt_template_str = """
You are an assistant for question-answering tasks. Use the following pieces of retrieved context to answer the question. If you don't know the answer, just say that you don't know. Use three sentences maximum and keep the answer concise.Question: {question} Context: {context} Answer:
"""
prompt_template = PromptTemplate.from_template(prompt_template_str)from langchain_core.runnables import RunnablePassthrough
from langchain_community.chat_models import ChatOpenAI
from langchain_core.output_parsers import StrOutputParser
llm = ChatOpenAI()
rag_chain = ({"context": retriever | format_docs, "question": RunnablePassthrough()} # 组装用户问题跟检索后的知识| prompt_template # 提示模版| llm # 大模型| StrOutputParser() # 结构化输出
)
# 生成Chain的调用图
rag_chain.get_graph().draw_png("rag-chain.png")

我们在目录下会得到一幅图,这个过程为链的调用过程,可能看起来有点抽象。

8bd1657be7d94043beae0cf7949b0d2f.png

现在我将图简化一下说明

518edab319234264844c298daf736b30.png

我们配合代码片段看

首先我们来说代码,这种写法为langchain的 LCEL 编码格式,LangChain Expression Language (LCEL) 翻译过来叫做langchain expression语言,详细内容参考

Conceptual guide | 🦜️🔗 LangChain

4eb541ede5974d0e9276a8eecf54fec8.png

       在构建劳动法问答机器人的过程中,Retriever组件扮演着至关重要的角色,它是整个信息检索流程的起点。用户输入的问题将首先被传递给Retriever,它负责在向量数据库中查询并筛选出相关的法律文档列表。以下是对您提供的段落进行优化后的描述:

在劳动法问答机器人的工作流程中,Retriever 是接收用户查询的第一站。用户提出的每一个问题,无论是关于劳动合同的细节还是劳动争议的解决,都会被自动送入Retriever进行处理。

Retriever的核心任务是从向量数据库中检索与用户问题最相关的法律文档。这一步骤完成后,检索到的文档列表将被传递给Format_docs组件。Format_docs的作用是将文档列表转换成适合LLM(大型语言模型)处理的字符串格式,确保信息的流畅性和可读性,并将转换后的字符串赋值给context变量,为下一步的LLM处理提供上下文。

在这一过程中,RunnablePassthrough作为一个直通组件,确保了从RetrieverFormat_docs的数据流转。由于我们处于整个链的第一环节,RunnablePassthrough()将捕获用户原始的问题,并将其作为问题变量question进行传递。

随着**contextquestion两个关键变量的准备就绪,我们便可以利用它们与预先定义的提示词模板结合,构建最终的prompt。这个prompt将引导LLM生成精准的答案。

LLM在接收到prompt后,将运用其语言理解与生成能力,提供一个经过深思熟虑的答案。生成的答案随后被送入**StrOutputParser**——一个专门的输出解析器,它负责对答案进行格式化处理,确保输出结果的准确性和可读性。

最终,经过StrOutputParser处理的答案将以一种易于理解且专业的形式呈现给用户,完成整个问答过程。

4、部署应用

我们现在都是代码级别,需要一个web页面及其后端接口才能更好的进行交互。

LangServe 帮我们解决了这个问题

LangServe GitHub - langchain-ai/langserve: LangServe 🦜️🏓

c94faf00f39d41b3a97db2e44b62f7c7.png

详细内容见:langchain/templates/README.md at master · langchain-ai/langchain · GitHub

安装包

pip install langservepip install sse_starlette

编写服务代码 app.py

cad7474d867e430395a1169ff878f2fb.png

# 配置OpenAi密匙
import osos.environ["OPENAI_API_KEY"] = ''
os.environ["OPENAI_API_BASE"] = ""from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings# 链接本地向量数据库
vectorstore = Chroma(persist_directory="./chroma_db",embedding_function=OpenAIEmbeddings()
)# 构建检索器
retriever = vectorstore.as_retriever(search_type="similarity",search_kwargs={"k": 4}
)# 查询
# docs = retriever.invoke("劳动者解除劳动合同需要提前多久告诉人事?")# 将前面查询出来的结果,提取成一个大的文本
def format_docs(docs):return "\n\n".join(doc.page_content for doc in docs)# 构建RAG链
from langchain.prompts.prompt import PromptTemplateprompt_template_str = """
You are an assistant for question-answering tasks. Use the following pieces of retrieved context to answer the question. If you don't know the answer, just say that you don't know. Use three sentences maximum and keep the answer concise.Question: {question} Context: {context} Answer:
"""
prompt_template = PromptTemplate.from_template(prompt_template_str)from langchain_core.runnables import RunnablePassthrough
from langchain_community.chat_models import ChatOpenAI
from langchain_core.output_parsers import StrOutputParser
llm = ChatOpenAI()
rag_chain = ({"context": retriever | format_docs, "question": RunnablePassthrough()} # 组装用户问题跟检索后的知识| prompt_template # 提示模版| llm # 大模型| StrOutputParser() # 结构化输出
)
# 生成Chain的调用图
# rag_chain.get_graph().draw_png("rag-chain.png")from fastapi import FastAPI
from langserve import add_routes
app = FastAPI(title="劳动法智能助手",version="1.0",
)
# 3. Adding chain route
add_routes(app,rag_chain,path="/low_ai",
)
if __name__ == "__main__":import uvicornuvicorn.run(app, host="localhost", port=7777)

现在咱们运行脚本

e84b960ce0a742e993f7d1737e9fc40d.png

访问 http://localhost:7777/low_ai/playground/

现在,输入你的问题!进行测试吧!

cd550283dac54c9d8559da25f62c13c6.gif

完整代码

数据处理:数据加载+数据切割+存储向量数据库

etl_datas.py

import bs4
from langchain_community.document_loaders import WebBaseLoader# 配置OpenAi密匙
import os
os.environ["OPENAI_API_KEY"] = 'openai key'
os.environ["OPENAI_API_BASE"] = "代理地址"# 构建加载器,加载URL数据
loader = WebBaseLoader(web_path="https://www.mohrss.gov.cn/SYrlzyhshbzb/zcfg/flfg/201601/t20160119_232110.html",bs_kwargs=dict(parse_only=bs4.SoupStrainer(class_=("TRS_PreAppend")))
)
docs = loader.load()# 切割文档
from langchain.text_splitter import RecursiveCharacterTextSplitter
text_splitter = RecursiveCharacterTextSplitter(chunk_size=1000, chunk_overlap=200)
splits = text_splitter.split_documents(docs)# 向量化数据
from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings
db = Chroma.from_documents(documents=splits,embedding=OpenAIEmbeddings(),persist_directory="./chroma_db")

应用构建:创建 llm ,创建prompt,检索数据库,构建 web 服务

app.py

# 配置OpenAi密匙
import osos.environ["OPENAI_API_KEY"] = 'openai key'
os.environ["OPENAI_API_BASE"] = "代理地址"from langchain_community.vectorstores import Chroma
from langchain_openai import OpenAIEmbeddings# 链接本地向量数据库
vectorstore = Chroma(persist_directory="./chroma_db",embedding_function=OpenAIEmbeddings()
)# 构建检索器
retriever = vectorstore.as_retriever(search_type="similarity",search_kwargs={"k": 4}
)# 查询
# docs = retriever.invoke("劳动者解除劳动合同需要提前多久告诉人事?")# 将前面查询出来的结果,提取成一个大的文本
def format_docs(docs):return "\n\n".join(doc.page_content for doc in docs)# 构建RAG链
from langchain.prompts.prompt import PromptTemplateprompt_template_str = """
You are an assistant for question-answering tasks. Use the following pieces of retrieved context to answer the question. If you don't know the answer, just say that you don't know. Use three sentences maximum and keep the answer concise.Question: {question} Context: {context} Answer:
"""
prompt_template = PromptTemplate.from_template(prompt_template_str)from langchain_core.runnables import RunnablePassthrough
from langchain_community.chat_models import ChatOpenAI
from langchain_core.output_parsers import StrOutputParser
llm = ChatOpenAI()
rag_chain = ({"context": retriever | format_docs, "question": RunnablePassthrough()} # 组装用户问题跟检索后的知识| prompt_template # 提示模版| llm # 大模型| StrOutputParser() # 结构化输出
)
# 生成Chain的调用图
# rag_chain.get_graph().draw_png("rag-chain.png")from fastapi import FastAPI
from langserve import add_routes
app = FastAPI(title="劳动法智能助手",version="1.0",
)
# 3. Adding chain route
add_routes(app,rag_chain,path="/low_ai",
)
if __name__ == "__main__":import uvicornuvicorn.run(app, host="localhost", port=7777)

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

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

相关文章

Python在Excel中设置数字格式和获取应用数字格式后的值

目录 安装Python Excel库 Python在Excel中设置数字格式 Python获取Excel中应用数字格式的单元格的显示值 总结 Excel 数字格式是用于控制单元格中数字显示方式的一组规则或代码。通过设置不同的数字格式,可以定义数字的显示方式,如小数位数、货币符号…

Java对应C++ STL的用法

sort: 1:java.util.Arrays中的静态方法Arrays.sort()方法,针对基本数据类型和引用对象类型的数组元素排序 2:java.util.Collections中的静态方法的Collections.sort()方法,针对集合框架中的动态数组,链表&…

iOS17系统适配

iOS17 新功能 文章目录 iOS17 新功能iOS17支持哪几款机型Xcode15新特性iOS17-开发适配指南 横屏待机 在iOS 17中,还带来了横屏待机功能,苹果将这个新功能命名为“Standby”模式,为 iPhone 带来了全新的玩法。iPhone启用之后,默认情…

1-Pandas是什么

Pandas是什么 Pandas 是一个开源的第三方 Python 库,从 Numpy 和 Matplotlib 的基础上构建而来,享有数据分析“三剑客之一”的盛名(NumPy、Matplotlib、Pandas)。Pandas 已经成为 Python 数据分析的必备高级工具,它的…

PyTorch Tensor进阶操作指南(二):深度学习中的关键技巧

本文主要讲tensor的裁剪、索引、降维和增维 Tensor与numpy互转、Tensor运算等,请看这篇文章 目录 9.1、首先看torch.squeeze()函数: 示例9.1:(基本的使用) 小技巧1:如何看维数 示例9.2:&a…

安全架构概述_1.信息安全面临的威胁

在当今以计算机、网络和软件为载体的数字化服务几乎成为人类赖以生存的手段。与之而来的计算机犯罪呈现指数上升趋势,因此,信息的可用性、完整性、机密性、可控性和不可抵赖性等安全保障显得尤为重要,而满足这些诉求,离不开好的安…

Vulnhub-AdmX

主机发现 靶机 : 192.168.145.131131 这台主机 存活 端口扫描 nmap -sV -O -p 1-65535 192.168.145.131 存在 80 端口 ,这里连ssh 端口都没了 80 端口存在 Apache httpd 2.4.1 存在 Apache 默认页面 像这种页面 ,没有什么具体的价值 扫描一…

嵌入式Linux系统编程 — 4.6 atoi、strtol等函数实现字符串与数字转换

目录 1 字符串转整形数据 1.1 函数功能 1.2 示例程序 2 字符串转浮点型数据 2.1 函数介绍 2.2 示例程序 3 数字转字符串 3.1 函数介绍 3.2 函数原型 在编程中,经常会需要将数字组成的字符串转换为相应的数字、或者将数字转换为字符串,在 C 函数…

连环计 | 第6集 | 百姓有倒悬之危,君臣有累卵之急 | 貂蝉 | 三国演义 | 逐鹿群雄

🙋大家好!我是毛毛张! 🌈个人首页: 神马都会亿点点的毛毛张 📌这篇博客分享的是《三国演义》文学剧本第Ⅰ部分《群雄逐鹿》的第6️⃣集《连环计》的经典语句和文学剧本全集台词 文章目录 1.经典语句2.文学剧本台词 …

LabVIEW材料样本结构缺陷检测

本文介绍了一种基于LabVIEW的实验室振动特性分析测试装置,通过分析振动特性来检测结构缺陷。文章详细描述了具体案例、硬件型号、工作原理、软件功能以及注意事项。 硬件型号 振动传感器:PCB Piezotronics 352C33加速度计 数据采集卡:NI PXI…

python(基础语法,pandas,numpy,正则表达式,数据预处理)

python学习推荐网址: 白月黑羽 一、语法基础 目标: • list、tuple、set、dict的基本用法 • 内置函数 len(), eval(),range(),sort(…

Linux自动化交互脚本expect开发

在日常开发任务中,运行shell脚本有时候会提示输入密码的操作,如何让脚本自动输入密码呢?这时使用expect帮我们输入,Expect是基于Tcl发展而来的,它不仅可以进行交互,还可以根据程序的提示模拟标准输入&#…

西安高校大学智能制造实验室数字孪生可视化系统平台建设项目验收

随着工业4.0时代的到来,智能制造成为推动制造业转型升级的关键。为了培养学生的创新能力和实践能力,西安高校大学决定建设智能制造实验室,并引入数字孪生技术,构建可视化系统平台。项目旨在通过数字孪生技术,实现对制造…

微信小程序毕业设计-线上教育商城系统项目开发实战(附源码+论文)

大家好!我是程序猿老A,感谢您阅读本文,欢迎一键三连哦。 💞当前专栏:微信小程序毕业设计 精彩专栏推荐👇🏻👇🏻👇🏻 🎀 Python毕业设计…

刷代码随想录有感(120):贪心算法——买卖股票的最佳时机

题干&#xff1a; 代码&#xff1a; class Solution { public:int maxProfit(vector<int>& prices) {int low INT_MAX;int res INT_MIN;for(int i 0; i < prices.size(); i){low min(low, prices[i]);res max(res, prices[i] - low);}return res;} }; 贪心…

ARCGIS添加在线地图

地图服务地址&#xff1a;http://map.geoq.cn/ArcGIS/rest/services 具体方法&#xff1a; 结果展示&#xff1a;

T4打卡 学习笔记

所用环境 ● 语言环境&#xff1a;Python3.11 ● 编译器&#xff1a;jupyter notebook ● 深度学习框架&#xff1a;TensorFlow2.16.1 ● 显卡&#xff08;GPU&#xff09;&#xff1a;NVIDIA GeForce RTX 2070 设置GPU from tensorflow import keras from tensorflow.keras…

【2024最新华为OD-C/D卷试题汇总】[支持在线评测] LYA的字符串拼接游戏(200分) - 三语言AC题解(Python/Java/Cpp)

&#x1f36d; 大家好这里是清隆学长 &#xff0c;一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 &#x1f4bb; ACM银牌&#x1f948;| 多次AK大厂笔试 &#xff5c; 编程一对一辅导 &#x1f44f; 感谢大家的订阅➕ 和 喜欢&#x1f497; &#x1f…

GO sync包——读写锁

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

多表查询实训

前提 本篇博客&#xff0c;我将通过讲解例题的方式&#xff0c;带大家进一步掌握多表查询的使用规则和使用技巧 正文 前提 先建好表 表1 salgrade (薪资等级表&#xff09; 表2 emp(员工信息表&#xff09; 表3 dept&#xff08;部门信息表&#xff09;&#xff0c;插入相…