【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从而实现记忆功能。

相关推荐
zhangfeng11332 小时前
国家超算中心 scnet.cn 跨用户文件分享流程总结 多个用户之间 文件共享 不需要反复下载上传
人工智能·语言模型·大模型
ting94520005 小时前
Tornado 全栈技术深度指南:从原理到实战
人工智能·python·架构·tornado
果汁华5 小时前
Browserbase Skills:让 Claude Agent 真正“看见“网页世界
人工智能·python
ZhengEnCi5 小时前
04-缩放点积注意力代码实现 💻
人工智能·python
HackTwoHub6 小时前
AI大模型网关存在SQL注入、附 POC 复现、影响版本LiteLLM 1.81.16~1.83.7(CVE-2026-42208)
数据库·人工智能·sql·网络安全·系统安全·网络攻击模型·安全架构
wuminyu6 小时前
专家视角看Java字节码加载与存储指令机制
java·linux·c语言·jvm·c++
段一凡-华北理工大学6 小时前
【高炉炼铁领域炉温监测、预警、调控智能体设计与应用】~系列文章08:多模态数据融合:让数据更聪明
人工智能·python·高炉炼铁·ai赋能·工业智能体·高炉炉温
网络工程小王7 小时前
【LangChain 大模型6大调用指南】调用大模型篇
linux·运维·服务器·人工智能·学习
HIT_Weston7 小时前
63、【Agent】【OpenCode】用户对话提示词(示例)
人工智能·agent·opencode
CV-杨帆7 小时前
Phi-4-mini-flash-reasoning 部署安装与推理测试完整记录
人工智能