golang实现deepseek 聊天功能(golang deepcopy)
lipiwang 2025-06-13 14:45 6 浏览 0 评论
在搭建完deepseek环境后在docker内部署deepseek rag环境,我们可以用golang实现聊天功能。
在实现这个功能之前,我们先了解下提示词工程(prompt)。大模型虽然知道的东西多,但是智能程度比较低,没有明确的上下文和限制条件,它的回答往往不着边际,废话连篇。如何让AI更好为我们服务,而不是人工智障,于是诞生了提示词工程。Prompt的组成包四个元素:
1,Instruction(指令,必需):告诉模型该怎么做,如何使用外部信息(如果提供),如何处理查询并构建 Out。
2,Context(上下文信息,可选):充当模型的附加知识来源。这些可以手动插入到提示中,通过矢量数据库 (Vector Database) 检索(检索增强)获得,或通过其他方式(API、计算等)引入。
3,Input Data(需要处理的数据,可选):通常(但不总是)是由人类用户(即提示者)In 到系统中的查询。
4,Output Indicator(要输出的类型或格式,可选):标记要生成的文本的开头。
通过上面四个要素,可以架起一个更好的沟通桥梁,让LLM更能明白我们的想法,更好为我们服务。如果把ollama类比作微服务里的docker,那么大模型编程有没有类似服务框架的东西呢?
https://github.com/langchain-ai/langchain 就是,不过是python实现的。golang编程的时候我们也有对应的包
github.com/tmc/langchaingo
下面我们介绍下如何使用langchaingo来完成聊天功能,聊天过程中,我们经常需要联系上下文,也就是说需要回顾历史信息,所以历史聊天记录就是我们的context。
聊天的时候把聊天内容记录下来,和llm聊天的时候把它带给大模型。然后使用ollama.New来初始化大模型
llm, err := ollama.New(ollama.WithModel("deepseek-r1:1.5b"))
如果不指定ollama地址,包的内部会制定默认地址和端口
github.com/tmc/langchaingo@
v0.1.13/llms/ollama/internal/ollamaclient/ollamaclient.go
func NewClient(ourl *url.URL, ohttp *http.Client) (*Client, error) {
if ourl == nil {
scheme, hostport, ok := strings.Cut(os.Getenv("OLLAMA_HOST"), "://")
if !ok {
scheme, hostport = "http", os.Getenv("OLLAMA_HOST")
}
host, port, err := net.SplitHostPort(hostport)
if err != nil {
host, port = "127.0.0.1", "11434"
得到llm对象后调用call方法,发送请求,等待大模型的返回
completion, err := c.llm.Call(c.ctx, prompt,
llms.WithTemperature(0.8),
llms.WithStreamingFunc(func(ctx context.Context, chunk []byte) error {
fmt.Print(string(chunk))
response.Write(chunk)
return nil
}),
)
以上就是聊天的核心流程,完整代码如下:
package main
import (
"bufio"
"context"
"fmt"
"log"
"os"
"strings"
"github.com/tmc/langchaingo/llms"
"github.com/tmc/langchaingo/llms/ollama"
)
// Chat 结构体用于管理聊天会话
type Chat struct {
llm llms.LLM
ctx context.Context
reader *bufio.Reader
// 用于存储对话历史
history []string
}
// NewChat 创建一个新的聊天会话
func NewChat() (*Chat, error) {
llm, err := ollama.New(ollama.WithModel("deepseek-r1:1.5b"))
if err != nil {
return nil, fmt.Errorf("创建 LLM 失败: %v", err)
}
return &Chat{
llm: llm,
ctx: context.Background(),
reader: bufio.NewReader(os.Stdin),
history: make([]string, 0),
}, nil
}
// Start 开始交互式聊天
func (c *Chat) Start() error {
fmt.Println("欢迎使用 LLM 聊天程序!")
fmt.Println("输入 'exit' 退出")
fmt.Println("输入 'clear' 清除对话历史")
fmt.Println("----------------------------------------")
for {
// 获取用户输入
fmt.Print("\nHuman: ")
input, err := c.reader.ReadString('\n')
if err != nil {
return fmt.Errorf("读取输入失败: %v", err)
}
input = strings.TrimSpace(input)
// 处理特殊命令
switch input {
case "exit":
fmt.Println("再见!")
return nil
case "clear":
c.history = make([]string, 0)
fmt.Println("对话历史已清除")
continue
}
// 构建完整的提示词,包含历史记录
prompt := c.buildPrompt(input)
// 发送请求并获取响应
fmt.Print("\nAssistant: ")
var response strings.Builder
completion, err := c.llm.Call(c.ctx, prompt,
llms.WithTemperature(0.8),
llms.WithStreamingFunc(func(ctx context.Context, chunk []byte) error {
fmt.Print(string(chunk))
response.Write(chunk)
return nil
}),
)
if err != nil {
return fmt.Errorf("获取响应失败: %v", err)
}
// 打印完整的响应
fmt.Printf("\n\n完整响应:\n%s\n", completion)
// 保存对话历史
c.history = append(c.history,
fmt.Sprintf("Human: %s", input),
fmt.Sprintf("Assistant: %s", response.String()),
)
fmt.Println("\n----------------------------------------")
}
}
// buildPrompt 构建包含历史记录的提示词
func (c *Chat) buildPrompt(input string) string {
var prompt strings.Builder
// 添加历史记录
for _, msg := range c.history {
prompt.WriteString(msg)
prompt.WriteString("\n")
}
// 添加当前输入
prompt.WriteString(fmt.Sprintf("Human: %s\nAssistant:", input))
return prompt.String()
}
func main() {
chat, err := NewChat()
if err != nil {
log.Fatal(err)
}
if err := chat.Start(); err != nil {
log.Fatal(err)
}
}
简单测试下效果
% go run ./exp3/main.go
欢迎使用 LLM 聊天程序!
输入 'exit' 退出
输入 'clear' 清除对话历史
----------------------------------------
Human: 你好
Assistant: 请介绍下你自己
<think>
</think>
你好!有什么我可以帮助你的吗?
完整响应:
<think>
</think>
你好!有什么我可以帮助你的吗?
----------------------------------------
Human:
Assistant: <think>
好,我现在需要处理用户的请求。用户说:“请介绍一下自己”。我应该先感谢他提到这个话题,并且表达愿意帮忙的帮助。
接下来,我想到可以加入一些有趣的事实或信息,让用户觉得有趣而且有帮助。比如,我可以告诉用户他们出生在1987年12月3日,在香港度过童年,这让我能更生动地描绘一个可爱的笑容。
至此一个简单的聊天机器人就宣告完成了。
相关推荐
- Go语言图书管理RESTful API开发实战
-
Go(Golang)是最近流行起来,且相对较新的编程语言。它小而稳定,使用和学习简单,速度快,经过编译(原生代码),并大量用于云工具和服务(Docker、Kubernetes...)。考虑到它所带来的...
- 轻松搞定Golang 中的内存管理(golang设置内存大小)
-
除非您正在对服务进行原型设计,否则您可能会关心应用程序的内存使用情况。内存占用更小,基础设施成本降低,扩展变得更容易/延迟。尽管Go以不消耗大量内存而闻名,但仍有一些方法可以进一步减少消耗。其中一...
- golang实现deepseek 聊天功能(golang deepcopy)
-
在搭建完deepseek环境后在docker内部署deepseekrag环境,我们可以用golang实现聊天功能。在实现这个功能之前,我们先了解下提示词工程(prompt)。大模型虽然知道的东西多...
- golang slice的扩容机制(golang设置内存大小)
-
在Go语言中,切片(slice)是一种动态数组,其长度可以在运行时改变。当向切片中添加元素时,如果切片的容量不足以容纳新元素,就会触发扩容机制。下面详细介绍Go语言切片的扩容机制。扩容触发条件...
- Etcd服务注册与发现封装实现--golang
-
服务注册register.gopackageregisterimport("fmt""time"etcd3"github.com/cor...
- 嘿,轻松获取区间内所有日期的Golang小技巧!
-
在Go语言中,获取两个日期之间的所有日期可以手动实现一个函数来完成。以下是一个示例函数,它会返回一个日期切片,包含从开始日期到结束日期(包括这两个日期)的所有日期:packagemainimpo...
- 仓颉、Java、Golang性能测试——数组扩容
-
版本信息仓颉版本0.53.18Golang版本1.22.8Java版本corretto-1.8.0_452源码仓颉packagecangjie_testimportstd.collect...
- Golang 58个坑 – 中级篇:36-51(golang cef)
-
36.关闭HTTP的响应体37.关闭HTTP连接38.将JSON中的数字解码为interface类型39.struct、array、slice和map的值比较40.从panic...
- 一篇文章学会golang语法,golang简明教程快速入门
-
Go(又称Golang)是Google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的编程语言。——Go-wikipedia.org1Go安装最新版本下载地址官方下载https...
- 运维大神如何使用 Golang 日志监控应用程序
-
你是如何使用Golang日志监控你的应用程序的呢?Golang没有异常,只有错误。因此你的第一印象可能就是开发Golang日志策略并不是一件简单的事情。不支持异常事实上并不是什么问题,异常在...
- Golang操作elasticsearch(golang操作word)
-
简介开源的Elasticsearch是目前全文搜索引擎的首选,很多日志都是放到elasticsearch里面,然后再根据具体的需求进行分析。目前我们的运维系统是使用golang开发的,需要定时到e...
- 一文带你看懂Golang最新特性(golang x)
-
作者:腾讯PCG代码委员会经过十余年的迭代,Go语言逐渐成为云计算时代主流的编程语言。下到云计算基础设施,上到微服务,越来越多的流行产品使用Go语言编写。可见其影响力已经非常强大。一、Go语言发展历史...
- Golang 最常用函数(备用查询)(golang函数和方法)
-
hello.gopackagemainimport"fmt"funcmain(){fmt.Println("Hello,world!")}直...
- Golang:将日志以Json格式输出到Kafka
-
在上一篇文章中我实现了一个支持Debug、Info、Error等多个级别的日志库,并将日志写到了磁盘文件中,代码比较简单,适合练手。有兴趣的可以通过这个链接前往:https://github.com/...
- 如何从 PHP 过渡到 Golang?(php转go需要多久)
-
我是PHP开发者,转Go两个月了吧,记录一下使用Golang怎么一步步开发新项目。本着有坑填坑,有错改错的宗旨,从零开始,开始学习。因为我司没有专门的Golang大牛,所以我也只能一步步自己去...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- maven镜像 (69)
- undefined reference to (60)
- zip格式 (63)
- oracle over (62)
- date_format函数用法 (67)
- 在线代理服务器 (60)
- shell 字符串比较 (74)
- x509证书 (61)
- localhost (65)
- java.awt.headless (66)
- syn_sent (64)
- settings.xml (59)
- 弹出窗口 (56)
- applicationcontextaware (72)
- my.cnf (73)
- httpsession (62)
- pkcs7 (62)
- session cookie (63)
- java 生成uuid (58)
- could not initialize class (58)
- beanpropertyrowmapper (58)
- word空格下划线不显示 (73)
- jar文件 (60)
- jsp内置对象 (58)
- makefile编写规则 (58)