使用SpringBoot+Deepseek4j接入阿里云百炼DeepSeek R1满血版

Spring有Spring AI这个项目,不过这里找到了另一个项目Deepseek4j,也是个Spring starter,提供了一个基础框架用来承接各种ds大模型,可以对接包括本地部署的Ollama r1,腾讯、硅基流动、阿里、火山引擎这些版本的R1模型或V3模型,比较方便。以下是一个简单实践。

前端ai.html --> 后端SpringBoot + Deepseek4j --> 阿里云大模型服务平台百炼上的Deepseek R1满血版

使用Deepseek4j接入阿里云百炼DeepSeek R1满血版
  1. 先去阿里云大模型服务平台“百炼”上进行注册,获得API_KEY

  2. 本地搭建SpringBoot + Deepseek4j

    pom.xml 增加如下依赖,其他依赖略,SpringMVC项目或WebFlux都可以。

    <dependency>
      <groupId>io.github.pig-mesh.ai</groupId>
      <artifactId>deepseek-spring-boot-starter</artifactId>
      <version>1.4.2</version>
    </dependency>
    

    controller

    import io.github.pigmesh.ai.deepseek.core.DeepSeekClient;
    import io.github.pigmesh.ai.deepseek.core.chat.ChatCompletionResponse;
    import lombok.extern.slf4j.Slf4j;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.http.MediaType;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    import reactor.core.publisher.Flux;
    
    @Slf4j
    @RestController
    @RequestMapping("ai")
    public class AiController {
    
        @Autowired
        private DeepSeekClient deepSeekClient;
    
        @GetMapping(value = "/chat", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
        public Flux<ChatCompletionResponse> chat(String prompt) {
            return deepSeekClient.chatFluxCompletion(prompt);
        }
    }
    

    application.properties文件

    deepseek.base-url=https://dashscope.aliyuncs.com/compatible-mode/v1
    deepseek.model=deepseek-r1
    deepseek.api-key=百炼上申请的API_KEY
    deepseek.log-requests=true
    deepseek.log-responses=true
    deepseek.connect-timeout=10
    deepseek.read-timeout=30
    deepseek.call-timeout=60
    
  1. 写个测试问答api页面ai.html

    <!DOCTYPE html>
    <html lang="zh-CN">
    <head>
        <meta charset="UTF-8">
        <title>AI 问答测试页面(优化版)</title>
        <style>
        body {
          font-family: Arial, sans-serif;
          margin: 20px;
          max-width: 800px;
        }
        textarea {
          width: 100%;
          height: 100px;
          margin: 10px 0;
          padding: 8px;
          box-sizing: border-box;
        }
        .response-area {
          border: 1px solid #ddd;
          border-radius: 4px;
          margin: 10px 0;
          padding: 10px;
          background-color: #f8f9fa;
        }
        #reasoning {
          background-color: #fff;
          max-height: 200px;
          overflow-y: auto;
        }
        #answer {
          white-space: pre-wrap;
          min-height: 100px;
        }
        .toggle-header {
          cursor: pointer;
          color: #666;
          padding: 5px;
          background-color: #f0f0f0;
          border-radius: 3px;
          margin: 5px 0;
        }
        button {
          padding: 10px 20px;
          background-color: #007bff;
          color: white;
          border: none;
          border-radius: 4px;
          cursor: pointer;
        }
        button:hover {
          background-color: #0056b3;
        }
      </style>
    </head>
    <body>
    <h1>AI 问答 API 测试</h1>
    <textarea id="prompt" placeholder="请输入问题..."></textarea><br>
    <button id="send">发送</button>
    
    <h2>回答结果:</h2>
    <div class="response-area" id="answer"></div>
    
    <details>
        <summary class="toggle-header">推理过程(点击展开)</summary>
        <div class="response-area" id="reasoning"></div>
    </details>
    
    <script>
        let eventSource = null;
    
        document.getElementById('send').addEventListener('click', function() {
          const prompt = document.getElementById('prompt').value;
          if (!prompt) {
            alert("请输入问题");
            return;
          }
    
          // 关闭已有连接
          if (eventSource) {
            eventSource.close();
          }
    
          // 清空显示区域
          document.getElementById('answer').textContent = "";
          document.getElementById('reasoning').textContent = "";
    
          const url = `/ai/chat?prompt=${encodeURIComponent(prompt)}`;
          eventSource = new EventSource(url);
    
          eventSource.onmessage = (e) => {
            try {
              const data = JSON.parse(e.data);
              const delta = data.choices[0].delta;
    
              // 处理推理内容
              if (delta.reasoning_content !== undefined) {
                const reasoningElement = document.getElementById('reasoning');
                reasoningElement.textContent += delta.reasoning_content;
                reasoningElement.scrollTop = reasoningElement.scrollHeight;
              }
    
              // 处理最终回答
              if (delta.content !== undefined) {
                const answerElement = document.getElementById('answer');
                answerElement.textContent += delta.content;
                answerElement.scrollTop = answerElement.scrollHeight;
              }
            } catch (err) {
              console.error('数据解析错误:', err);
            }
          };
    
          eventSource.onerror = (e) => {
            console.error("SSE连接错误:", e);
            if (eventSource) {
              eventSource.close();
              eventSource = null;
            }
          };
        });
      </script>
    </body>
    </html>
    

效果:


测试效果
参考和进一步阅读

前端如何与后端响应式类型API交互
https://blog.csdn.net/waiter456/article/details/141114447
https://www.ruanyifeng.com/blog/2017/05/server-sent_events.html

Spring AI整合
https://javaai.pig4cloud.com/deepseek/bailian
https://www.cnblogs.com/guoxiaoyu/p/18700826

通过阿里云百炼平台调用DeepSeek-R1 满血版 API
https://www.aliyun.com/solution/tech-solution/deepseek-r1-for-platforms

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,589评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,615评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,933评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,976评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,999评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,775评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,474评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,359评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,854评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,007评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,146评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,826评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,484评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,029评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,153评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,420评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,107评论 2 356

推荐阅读更多精彩内容