【SpringAi最新版入门(二)】

搭建可视化页面

在resource中新建一个文件夹static,创建index.html,要注意的是文件夹必须是static,否则springboot无法加载index.html

html 复制代码
<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Spring AI 聊天助手</title>
    <style>
        * {
            margin: 0;
            padding: 0;
            box-sizing: border-box;
            font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
        }

        body {
            background-color: #f5f7fa;
            display: flex;
            justify-content: center;
            align-items: center;
            min-height: 100vh;
            padding: 20px;
        }

        .chat-container {
            width: 100%;
            max-width: 800px;
            background: white;
            border-radius: 16px;
            box-shadow: 0 8px 30px rgba(0, 0, 0, 0.08);
            overflow: hidden;
            display: flex;
            flex-direction: column;
        }

        .chat-header {
            background-color: #6366f1;
            color: white;
            padding: 20px;
            text-align: center;
            font-size: 18px;
            font-weight: 600;
        }

        .chat-body {
            flex: 1;
            padding: 24px;
            min-height: 500px;
            max-height: 700px;
            overflow-y: auto;
            background: #fafbfc;
        }

        .message-bubble {
            background: white;
            padding: 16px 20px;
            border-radius: 14px;
            box-shadow: 0 2px 6px rgba(0, 0, 0, 0.04);
            line-height: 1.7;
            font-size: 15px;
            color: #333;
            white-space: pre-wrap;
            word-break: break-word;
            border: 1px solid #eee;
        }

        .loading {
            color: #6366f1;
            font-style: italic;
            margin-top: 10px;
            animation: fade 1.5s infinite;
        }

        @keyframes fade {
            0%, 100% { opacity: 0.5; }
            50% { opacity: 1; }
        }

        .chat-footer {
            padding: 16px 20px;
            border-top: 1px solid #eee;
            display: flex;
            gap: 12px;
        }

        #promptInput {
            flex: 1;
            padding: 14px 18px;
            border: 1px solid #ddd;
            border-radius: 12px;
            outline: none;
            font-size: 15px;
            transition: border 0.2s;
        }

        #promptInput:focus {
            border-color: #6366f1;
        }

        #sendBtn {
            padding: 14px 24px;
            background: #6366f1;
            color: white;
            border: none;
            border-radius: 12px;
            font-weight: 500;
            cursor: pointer;
            transition: background 0.2s;
        }

        #sendBtn:hover {
            background: #4f46e5;
        }

        #sendBtn:disabled {
            background: #a5b4fc;
            cursor: not-allowed;
        }
    </style>
</head>
<body>

<div class="chat-container">
    <div class="chat-header">Spring AI 智能助手</div>
    <div class="chat-body" id="response"></div>
    <div class="chat-footer">
        <input type="text" id="promptInput" placeholder="请输入你的问题......" autocomplete="off">
        <button id="sendBtn">发送</button>
    </div>
</div>

<script>
    const responseEl = document.getElementById('response');
    const sendBtn = document.getElementById('sendBtn');
    const promptInput = document.getElementById('promptInput');

    let eventSource = null;

    // 发送问题
    function sendMessage() {
        const prompt = promptInput.value.trim();
        if (!prompt) return;

        // 清空界面 & 禁用按钮
        responseEl.innerHTML = '<div class="loading">AI 思考中,请稍候......</div>';
        sendBtn.disabled = true;
        promptInput.disabled = true;

        // 关闭旧连接
        if (eventSource) eventSource.close();

        // 创建新连接
        eventSource = new EventSource(`http://localhost:8080/ai/chat?prompt=${encodeURIComponent(prompt)}`);

        let fullText = '';
        eventSource.onmessage = function (event) {
            fullText += event.data;
            responseEl.innerHTML = `<div class="message-bubble">${fullText}</div>`;
        };

        eventSource.onerror = function () {
            responseEl.innerHTML += '<div style="color:red;">连接断开。。。</div>';
            closeConnection();
        };

        eventSource.onopen = function () {
            responseEl.innerHTML = '<div class="loading">正在接收回复......</div>';
        };
    }

    function closeConnection() {
        if (eventSource) eventSource.close();
        sendBtn.disabled = false;
        promptInput.disabled = false;
        promptInput.value = '';
        eventSource = null;
    }

    // 绑定事件
    sendBtn.addEventListener('click', sendMessage);
    promptInput.addEventListener('keydown', e => e.key === 'Enter' && sendMessage());
</script>

</body>
</html>

启动springboot后访问浏览器,效果展示

流式输出

修改之前的chat函数:

java 复制代码
 @GetMapping("/chat")
    public Flux<String> chat(@RequestParam String prompt) {
        return chatClient.prompt(prompt).stream().content();
    }

去浏览器测试,

那么ai在返回结果时就可以一个词一个词的输出了,极大的增强了用户的体验,而不是像之前一样输出所有文字后在返回给用户。

记忆化

到目前为止,ai其实只能记录我们的一次对话,无法获取之前对话的内容。我们可以配置chatclient来解决这个问题:

java 复制代码
@Bean
    public ChatClient chatClient(OllamaChatModel model) {
        return ChatClient.builder(model)
                .defaultAdvisors(MessageChatMemoryAdvisor.builder(MessageWindowChatMemory.builder()
                        .maxMessages(10) //最大记忆的对话次数
                        .build())
                        .conversationId("use") //会话id
                        .build())
                .build();
    }

可以看到我们为chatclint配置了defaultAdvisors,在advisor中我们可以配置MessageWindowChatMemory从而实现记忆功能。

相关推荐
Lenyiin2 小时前
Python数据类型与运算符:深入理解Python世界的基石
java·开发语言·python
补三补四2 小时前
模型评估方法论:从基础指标到AI裁判的全面指南
人工智能
小江的记录本2 小时前
【大语言模型】大语言模型——核心概念(预训练、SFT监督微调、RLHF/RLAIF对齐、Token、Embedding、上下文窗口)
java·人工智能·后端·python·算法·语言模型·自然语言处理
西西学代码2 小时前
查找设备页面(amap_map)
开发语言·前端·javascript
我叫张土豆2 小时前
我把 Spring Boot 升级到 4.0.2 后,顺手重构了整个 AI 脚手架:删模块、加 Skills Agent、补 Resume RAG
人工智能·spring boot·重构
念越2 小时前
算法每日一题 Day01|双指针解决移动零问题
java·算法·力扣
敖正炀2 小时前
StampedLock 详解
java·后端
AllData公司负责人2 小时前
AllData数据中台集成开源项目Apache Doris建设实时数仓平台
java·大数据·数据库·数据仓库·apache doris·实时数仓平台·doris集群
白宇横流学长2 小时前
助农产品在线交易平台设计与实现【源码+文档】
java