Kafka在Vue和Spring Boot中的使用实例
Kafka在Vue和Spring Boot中的使用实例
一、项目概述
本项目演示了如何在Vue前端和Spring Boot后端中集成Kafka,实现实时消息的发送和接收,以及数据的实时展示。
后端实现:springboot配置、kafka配置、消息模型和仓库、消息服务和消费者、websocket配置、REST api控制器
前端实现:vue项目创建、websocket客户端配置、api服务、消息聊天组件、统计图表组件、主页面和路由配置
1.1 功能特点
- 前端实时发送消息到Kafka
- 后端接收Kafka消息并处理
- 后端发送消息到Kafka
- 前端实时接收并展示Kafka消息
- 消息历史记录展示
- 消息统计图表展示
1.2 技术栈
- 前端:Vue 3 + Element Plus + ECharts
- 后端:Spring Boot 2.7.x + Spring Kafka
- 消息中间件:Apache Kafka
- 数据库:MySQL (存储消息历史)
二、环境准备
2.1 安装Kafka
# 下载Kafka
wget https://downloads.apache.org/kafka/3.5.1/kafka_2.13-3.5.1.tgz# 解压
tar -xzf kafka_2.13-3.5.1.tgz# 启动Zookeeper
bin/zookeeper-server-start.sh config/zookeeper.properties# 启动Kafka
bin/kafka-server-start.sh config/server.properties
2.2 创建Topic
# 创建消息主题
bin/kafka-topics.sh --create --bootstrap-server localhost:9092 \--replication-factor 1 --partitions 3 --topic chat-messages# 创建通知主题
bin/kafka-topics.sh --create --bootstrap-server localhost:9092 \--replication-factor 1 --partitions 3 --topic notifications
三、后端实现
3.1 添加依赖
<!-- pom.xml -->
<dependencies><!-- Spring Boot Starter --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!-- Spring Kafka --><dependency><groupId>org.springframework.kafka</groupId><artifactId>spring-kafka</artifactId></dependency><!-- MySQL --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><scope>runtime</scope></dependency><!-- Lombok --><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional></dependency>
</dependencies>
3.2 配置Kafka
# application.yml
spring:kafka:bootstrap-servers: localhost:9092producer:key-serializer: org.apache.kafka.common.serialization.StringSerializervalue-serializer: org.springframework.kafka.support.serializer.JsonSerializerconsumer:group-id: chat-groupauto-offset-reset: earliestkey-deserializer: org.apache.kafka.common.serialization.StringDeserializervalue-deserializer: org.springframework.kafka.support.serializer.JsonDeserializerproperties:spring.json.trusted.packages: "com.example.kafkademo.model"datasource:url: jdbc:mysql://localhost:3306/kafka_demo?useSSL=false&serverTimezone=UTCusername: rootpassword: passworddriver-class-name: com.mysql.cj.jdbc.Driverjpa:hibernate:ddl-auto: updateshow-sql: true
3.3 创建消息模型
// Message.java
package com.example.kafkademo.model;import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;import javax.persistence.*;
import java.time.LocalDateTime;@Data
@NoArgsConstructor
@AllArgsConstructor
@Entity
@Table(name = "messages")
public class Message {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String sender;private String content;private LocalDateTime timestamp;@PrePersistpublic void prePersist() {if (timestamp == null) {timestamp = LocalDateTime.now();}}
}
3.4 创建消息仓库
// MessageRepository.java
package com.example.kafkademo.repository;import com.example.kafkademo.model.Message;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;import java.util.List;public interface MessageRepository extends JpaRepository<Message, Long> {List<Message> findTop50ByOrderByTimestampDesc();@Query("SELECT m.sender, COUNT(m) FROM Message m GROUP BY m.sender")List<Object[]> countMessagesBySender();
}
3.5 创建Kafka配置类
// KafkaConfig.java
package com.example.kafkademo.config;import com.example.kafkademo.model.Message;
import org.apache.kafka.clients.admin.AdminClientConfig;
import org.apache.kafka.clients.admin.NewTopic;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.kafka.config.ConcurrentKafkaListenerContainerFactory;
import org.springframework.kafka.core.*;
import org.springframework.kafka.support.serializer.JsonDeserializer;
import org.springframework.kafka.support.serializer.JsonSerializer;import java.util.HashMap;
import java.util.Map;@Configuration
public class KafkaConfig {@Value("${spring.kafka.bootstrap-servers}")private String bootstrapServers;@Beanpublic KafkaAdmin kafkaAdmin() {Map<String, Object> configs = new HashMap<>();configs.put(AdminClientConfig.BOOTSTRAP_SERVERS_CONFIG, bootstrapServers);return new KafkaAdmin(configs);}@Beanpublic NewTopic chatTopic() {return new NewTopic("chat-messages", 3, (short) 1);}@Beanpublic NewTopic notificationTopic() {return new NewTopic("notifications", 3, (short) 1);}@Beanpublic ProducerFactory<String, Message> producerFactory() {Map<String, Object> configProps = new HashMap<>();configProps.put(org.apache.kafka.clients.producer.ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, bootstrapServers);configProps.put(org.apache.kafka.clients.producer.ProducerConfig.KEY_SERIALIZER_CLASS_CONFIG, org.apache.kafka.common.serialization.StringSerializer.class);configProps.put(org.apache.kafka.clients.producer.ProducerConfig.VALUE_SERIALIZER_CLASS_CONFIG, JsonSerializer.class);return new DefaultKafkaProducerFactory<>(configProps);}@Beanpublic KafkaTemplate<String, Message> kafkaTemplate() {return new KafkaTemplate<>(producerFactory());}@Beanpublic ConsumerFactory<String, Message> consumerFactory() {Map<String, Object> props = new HashMap<>();props.put(org.apache.kafka.clients.consumer.ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, bootstrapServers);props.put(org.apache.kafka.clients.consumer.ConsumerConfig.GROUP_ID_CONFIG, "chat-group");props.put(org.apache.kafka.clients.consumer.ConsumerConfig.KEY_DESERIALIZER_CLASS_CONFIG, org.apache.kafka.common.serialization.StringDeserializer.class);props.put(org.apache.kafka.clients.consumer.ConsumerConfig.VALUE_DESERIALIZER_CLASS_CONFIG, JsonDeserializer.class);props.put(JsonDeserializer.TRUSTED_PACKAGES, "com.example.kafkademo.model");return new DefaultKafkaConsumerFactory<>(props, new org.apache.kafka.common.serialization.StringDeserializer(),new JsonDeserializer<>(Message.class));}@Beanpublic ConcurrentKafkaListenerContainerFactory<String, Message> kafkaListenerContainerFactory() {ConcurrentKafkaListenerContainerFactory<String, Message> factory = new ConcurrentKafkaListenerContainerFactory<>();factory.setConsumerFactory(consumerFactory());return factory;}
}
3.6 创建消息服务
// MessageService.java
package com.example.kafkademo.service;import com.example.kafkademo.model.Message;
import com.example.kafkademo.repository.MessageRepository;
import lombok.RequiredArgsConstructor;
import org.springframework.kafka.core.KafkaTemplate;
import org.springframework.stereotype.Service;import java.util.List;@Service
@RequiredArgsConstructor
public class MessageService {private final KafkaTemplate<String, Message> kafkaTemplate;private final MessageRepository messageRepository;public void sendMessage(Message message) {// 发送消息到KafkakafkaTemplate.send("chat-messages", message);// 保存消息到数据库messageRepository.save(message);}public void sendNotification(String content) {Message notification = new Message();notification.setSender("系统");notification.setContent(content);kafkaTemplate.send("notifications", notification);}public List<Message> getRecentMessages() {return messageRepository.findTop50ByOrderByTimestampDesc();}public List<Object[]> getMessageStats() {return messageRepository.countMessagesBySender();}
}
3.7 创建Kafka消费者
// KafkaConsumer.java
package com.example.kafkademo.kafka;import com.example.kafkademo.model.Message;
import lombok.extern.slf4j.Slf4j;
import org.springframework.kafka.annotation.KafkaListener;
import org.springframework.messaging.simp.SimpMessagingTemplate;
import org.springframework.stereotype.Component;@Component
@Slf4j
public class KafkaConsumer {private final SimpMessagingTemplate messagingTemplate;public KafkaConsumer(SimpMessagingTemplate messagingTemplate) {this.messagingTemplate = messagingTemplate;}@KafkaListener(topics = "chat-messages", groupId = "chat-group")public void listenChatMessages(Message message) {log.info("收到聊天消息: {}", message);// 通过WebSocket发送消息到前端messagingTemplate.convertAndSend("/topic/messages", message);}@KafkaListener(topics = "notifications", groupId = "chat-group")public void listenNotifications(Message message) {log.info("收到通知: {}", message);// 通过WebSocket发送通知到前端messagingTemplate.convertAndSend("/topic/notifications", message);}
}
3.8 创建WebSocket配置
// WebSocketConfig.java
package com.example.kafkademo.config;import org.springframework.context.annotation.Configuration;
import org.springframework.messaging.simp.config.MessageBrokerRegistry;
import org.springframework.web.socket.config.annotation.EnableWebSocketMessageBroker;
import org.springframework.web.socket.config.annotation.StompEndpointRegistry;
import org.springframework.web.socket.config.annotation.WebSocketMessageBrokerConfigurer;@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {@Overridepublic void configureMessageBroker(MessageBrokerRegistry config) {config.enableSimpleBroker("/topic");config.setApplicationDestinationPrefixes("/app");}@Overridepublic void registerStompEndpoints(StompEndpointRegistry registry) {registry.addEndpoint("/ws").setAllowedOrigins("http://localhost:8080").withSockJS();}
}
3.9 创建控制器
// MessageController.java
package com.example.kafkademo.controller;import com.example.kafkademo.model.Message;
import com.example.kafkademo.service.MessageService;
import lombok.RequiredArgsConstructor;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;import java.util.List;@RestController
@RequestMapping("/api/messages")
@RequiredArgsConstructor
@CrossOrigin(origins = "http://localhost:8080")
public class MessageController {private final MessageService messageService;@PostMappingpublic ResponseEntity<Message> sendMessage(@RequestBody Message message) {messageService.sendMessage(message);return ResponseEntity.ok(message);}@GetMappingpublic ResponseEntity<List<Message>> getRecentMessages() {return ResponseEntity.ok(messageService.getRecentMessages());}@GetMapping("/stats")public ResponseEntity<List<Object[]>> getMessageStats() {return ResponseEntity.ok(messageService.getMessageStats());}
}
四、前端实现
4.1 创建Vue项目
# 创建Vue项目
npm create vue@latest kafka-vue-demo# 进入项目目录
cd kafka-vue-demo# 安装依赖
npm install# 安装额外依赖
npm install element-plus axios sockjs-client @stomp/stompjs echarts
4.2 配置WebSocket客户端
// src/utils/websocket.js
import SockJS from 'sockjs-client';
import { Stomp } from '@stomp/stompjs';class WebSocketClient {constructor() {this.stompClient = null;this.connected = false;}connect() {const socket = new SockJS('http://localhost:8081/ws');this.stompClient = Stomp.over(socket);this.stompClient.connect({}, this.onConnected, this.onError);}onConnected = () => {this.connected = true;console.log('WebSocket连接成功');// 订阅消息主题this.stompClient.subscribe('/topic/messages', this.onMessageReceived);this.stompClient.subscribe('/topic/notifications', this.onNotificationReceived);}onError = (error) => {console.error('WebSocket连接错误:', error);this.connected = false;}onMessageReceived = (payload) => {const message = JSON.parse(payload.body);console.log('收到消息:', message);// 触发消息接收事件window.dispatchEvent(new CustomEvent('messageReceived', { detail: message }));}onNotificationReceived = (payload) => {const notification = JSON.parse(payload.body);console.log('收到通知:', notification);// 触发通知接收事件window.dispatchEvent(new CustomEvent('notificationReceived', { detail: notification }));}disconnect() {if (this.stompClient) {this.stompClient.disconnect();this.connected = false;}}
}export default new WebSocketClient();
4.3 创建API服务
// src/api/messageApi.js
import axios from 'axios';const API_URL = 'http://localhost:8081/api';export default {sendMessage(message) {return axios.post(`${API_URL}/messages`, message);},getRecentMessages() {return axios.get(`${API_URL}/messages`);},getMessageStats() {return axios.get(`${API_URL}/messages/stats`);}
};
4.4 创建消息组件
<!-- src/components/MessageChat.vue -->
<template><div class="message-chat"><el-card class="chat-container"><template #header><div class="card-header"><h2>实时聊天</h2></div></template><div class="message-list" ref="messageList"><div v-for="message in messages" :key="message.id" class="message-item" :class="{ 'message-self': message.sender === username }"><div class="message-sender">{{ message.sender }}</div><div class="message-content">{{ message.content }}</div><div class="message-time">{{ formatTime(message.timestamp) }}</div></div></div><div class="message-input"><el-inputv-model="newMessage"placeholder="输入消息..."@keyup.enter="sendMessage"><template #append><el-button type="primary" @click="sendMessage">发送</el-button></template></el-input></div></el-card></div>
</template><script>
import { ref, onMounted, onUnmounted, nextTick } from 'vue';
import { ElMessage } from 'element-plus';
import messageApi from '../api/messageApi';
import websocket from '../utils/websocket';export default {name: 'MessageChat',props: {username: {type: String,required: true}},setup(props) {const messages = ref([]);const newMessage = ref('');const messageList = ref(null);// 加载历史消息const loadMessages = async () => {try {const response = await messageApi.getRecentMessages();messages.value = response.data;scrollToBottom();} catch (error) {console.error('加载消息失败:', error);ElMessage.error('加载消息失败');}};// 发送消息const sendMessage = async () => {if (!newMessage.value.trim()) return;const message = {sender: props.username,content: newMessage.value,timestamp: new Date()};try {await messageApi.sendMessage(message);newMessage.value = '';} catch (error) {console.error('发送消息失败:', error);ElMessage.error('发送消息失败');}};// 处理接收到的消息const handleMessageReceived = (event) => {const message = event.detail;messages.value.push(message);scrollToBottom();};// 处理接收到的通知const handleNotificationReceived = (event) => {const notification = event.detail;ElMessage.info(notification.content);};// 滚动到底部const scrollToBottom = async () => {await nextTick();if (messageList.value) {messageList.value.scrollTop = messageList.value.scrollHeight;}};// 格式化时间const formatTime = (timestamp) => {if (!timestamp) return '';const date = new Date(timestamp);return date.toLocaleTimeString();};onMounted(() => {loadMessages();websocket.connect();window.addEventListener('messageReceived', handleMessageReceived);window.addEventListener('notificationReceived', handleNotificationReceived);});onUnmounted(() => {websocket.disconnect();window.removeEventListener('messageReceived', handleMessageReceived);window.removeEventListener('notificationReceived', handleNotificationReceived);});return {messages,newMessage,messageList,sendMessage,formatTime};}
};
</script><style scoped>
.message-chat {height: 100%;display: flex;flex-direction: column;
}.chat-container {height: 100%;display: flex;flex-direction: column;
}.card-header {display: flex;justify-content: space-between;align-items: center;
}.message-list {flex: 1;overflow-y: auto;padding: 10px;margin-bottom: 10px;height: 400px;
}.message-item {margin-bottom: 10px;padding: 10px;border-radius: 5px;background-color: #f5f7fa;max-width: 80%;
}.message-self {margin-left: auto;background-color: #ecf5ff;
}.message-sender {font-weight: bold;margin-bottom: 5px;
}.message-time {font-size: 12px;color: #909399;text-align: right;margin-top: 5px;
}.message-input {margin-top: 10px;
}
</style>
4.5 创建统计图表组件
<!-- src/components/MessageStats.vue -->
<template><div class="message-stats"><el-card><template #header><div class="card-header"><h2>消息统计</h2></div></template><div class="chart-container" ref="chartContainer"></div></el-card></div>
</template><script>
import { ref, onMounted, onUnmounted } from 'vue';
import { ElMessage } from 'element-plus';
import * as echarts from 'echarts';
import messageApi from '../api/messageApi';export default {name: 'MessageStats',setup() {const chartContainer = ref(null);let chart = null;// 加载统计数据const loadStats = async () => {try {const response = await messageApi.getMessageStats();const stats = response.data;const senders = stats.map(item => item[0]);const counts = stats.map(item => item[1]);updateChart(senders, counts);} catch (error) {console.error('加载统计数据失败:', error);ElMessage.error('加载统计数据失败');}};// 更新图表const updateChart = (senders, counts) => {if (!chart) return;const option = {title: {text: '用户消息数量统计'},tooltip: {trigger: 'axis',axisPointer: {type: 'shadow'}},xAxis: {type: 'category',data: senders},yAxis: {type: 'value'},series: [{name: '消息数量',type: 'bar',data: counts}]};chart.setOption(option);};// 初始化图表const initChart = () => {if (chartContainer.value) {chart = echarts.init(chartContainer.value);loadStats();}};// 处理窗口大小变化const handleResize = () => {if (chart) {chart.resize();}};onMounted(() => {initChart();window.addEventListener('resize', handleResize);});onUnmounted(() => {if (chart) {chart.dispose();}window.removeEventListener('resize', handleResize);});return {chartContainer};}
};
</script><style scoped>
.message-stats {height: 100%;
}.chart-container {height: 400px;
}
</style>
4.6 创建主页面
<!-- src/App.vue -->
<template><div class="app-container"><el-container><el-header><h1>Kafka实时聊天演示</h1><div class="user-info" v-if="!username"><el-input v-model="inputUsername" placeholder="请输入用户名" /><el-button type="primary" @click="login">登录</el-button></div><div class="user-info" v-else><span>欢迎, {{ username }}</span><el-button type="danger" @click="logout">退出</el-button></div></el-header><el-main v-if="username"><el-row :gutter="20"><el-col :span="16"><MessageChat :username="username" /></el-col><el-col :span="8"><MessageStats /></el-col></el-row></el-main><el-main v-else class="login-container"><el-card class="login-card"><h2>欢迎使用Kafka实时聊天</h2><p>请输入用户名开始聊天</p></el-card></el-main></el-container></div>
</template><script>
import { ref } from 'vue';
import MessageChat from './components/MessageChat.vue';
import MessageStats from './components/MessageStats.vue';export default {name: 'App',components: {MessageChat,MessageStats},setup() {const username = ref('');const inputUsername = ref('');const login = () => {if (inputUsername.value.trim()) {username.value = inputUsername.value;inputUsername.value = '';}};const logout = () => {username.value = '';};return {username,inputUsername,login,logout};}
};
</script><style>
.app-container {height: 100vh;
}.el-header {background-color: #409eff;color: white;display: flex;justify-content: space-between;align-items: center;padding: 0 20px;
}.user-info {display: flex;align-items: center;gap: 10px;
}.login-container {display: flex;justify-content: center;align-items: center;
}.login-card {width: 400px;text-align: center;padding: 20px;
}.el-main {padding: 20px;
}
</style>
4.7 配置路由
// src/router/index.js
import { createRouter, createWebHistory } from 'vue-router';
import App from '../App.vue';const routes = [{path: '/',name: 'Home',component: App}
];const router = createRouter({history: createWebHistory(import.meta.env.BASE_URL),routes
});export default router;
4.8 配置主入口
// src/main.js
import { createApp } from 'vue';
import ElementPlus from 'element-plus';
import 'element-plus/dist/index.css';
import App from './App.vue';
import router from './router';const app = createApp(App);app.use(ElementPlus);
app.use(router);app.mount('#app');
五、运行项目
5.1 启动后端
# 进入后端项目目录
cd kafka-spring-demo# 启动Spring Boot应用
./mvnw spring-boot:run
5.2 启动前端
# 进入前端项目目录
cd kafka-vue-demo# 启动开发服务器
npm run dev
5.3 访问应用
打开浏览器,访问 http://localhost:8080 即可使用应用。
六、功能演示
6.1 发送和接收消息
- 打开多个浏览器窗口,分别以不同用户名登录
- 在任意窗口发送消息,所有窗口都能实时接收到消息
- 消息会显示发送者、内容和时间
6.2 消息统计
- 发送多条消息后,统计图表会显示每个用户发送的消息数量
- 图表会自动更新,反映最新的统计数据
6.3 系统通知
- 后端可以发送系统通知,所有用户都能收到
- 通知会以Element Plus的消息提示形式显示
七、扩展功能
7.1 添加消息类型
// 在Message类中添加消息类型字段
private String type; // 消息类型:text, image, file等
7.2 添加消息已读状态
// 在Message类中添加已读状态字段
private boolean read;
7.3 添加私聊功能
// 在Message类中添加接收者字段
private String receiver; // 私聊接收者,为空表示群聊
7.4 添加消息搜索功能
// 在MessageRepository中添加搜索方法
List<Message> findByContentContainingOrSenderContaining(String content, String sender);
八、总结
本项目演示了如何在Vue前端和Spring Boot后端中集成Kafka,实现实时消息的发送和接收。通过WebSocket和Kafka的结合,我们实现了一个功能完善的实时聊天应用。
8.1 技术要点
- Kafka消息的发送和接收
- WebSocket实时通信
- Vue组件化开发
- Spring Boot后端服务
- 数据可视化
8.2 项目亮点
- 实时消息推送
- 消息历史记录
- 消息统计图表
- 系统通知功能
- 响应式设计
8.3 后续优化方向
- 添加消息加密
- 实现消息撤回功能
- 添加文件上传功能
- 优化移动端适配
- 添加用户认证和授权
相关文章:
Kafka在Vue和Spring Boot中的使用实例
Kafka在Vue和Spring Boot中的使用实例 一、项目概述 本项目演示了如何在Vue前端和Spring Boot后端中集成Kafka,实现实时消息的发送和接收,以及数据的实时展示。 后端实现:springboot配置、kafka配置、消息模型和仓库、消息服务和消费者、we…...
层归一化详解及在 Stable Diffusion 中的应用分析
在深度学习中,归一化(Normalization)技术被广泛用于提升模型训练的稳定性和收敛速度。本文将详细介绍几种常见的归一化方式,并重点分析它们在 Stable Diffusion 模型中的实际使用场景。 一、常见的归一化技术 名称归一化维度应用…...
【C++DFS 马拉车】3327. 判断 DFS 字符串是否是回文串|2454
本文涉及知识点 CDFS 马拉车 LeetCode3327. 判断 DFS 字符串是否是回文串 给你一棵 n 个节点的树,树的根节点为 0 ,n 个节点的编号为 0 到 n - 1 。这棵树用一个长度为 n 的数组 parent 表示,其中 parent[i] 是节点 i 的父节点。由于节点 …...
前端开发vue项目(node-modules 可视化神器 Node Modules Inspector)
node-modules 可视化神器 Node Modules Inspector 简介功能特点使用场景实现原理 使用Node Modules Inspector提供 简介 Node Modules Inspector 是一个用于检查和分析 Node.js 项目中模块依赖关系的工具 功能特点 依赖分析:它能够深入剖析 Node.js项目中的模块依…...
25统计建模半自动化辅助排版模板及论文排版格式要求
1.除封面页外,不得在其他页出现学校、参赛队及指导教师的信息。 2.目录应由论文的篇、章、节、条、款以及附录题录等的序号、题名和页码组成。正文页码单独编列,其页码从正文第一页开始编写。 3.标题和正文:论文正文总标题(题目…...
武汉迅狐科技:AI赋能企业营销,打造智能获客新范式
在数字化营销竞争日益激烈的今天,武汉迅狐科技有限公司凭借其创新的AI技术和智能营销解决方案,正在帮助企业突破传统获客瓶颈,实现营销效率的指数级提升。作为一家专注于AI获客软件研发的高新技术企业,迅狐科技推出的矩阵系统、数…...
Tomcat:Java Web 应用开发的核心容器
在Java Web开发领域,Apache Tomcat凭借其开源特性、轻量级架构和强大的功能支持,成为开发者部署和运行Servlet、JSP应用的首选容器。作为Apache软件基金会旗下的Jakarta项目成果,Tomcat不仅实现了Java EE(现Jakarta EE)…...
Tomcat 安装与配置:超详细指南
目录 一、安装前的准备工作(一)配置 JAVA_HOME 环境变量 二、下载 Tomcat(一)Windows 系统(二)macOS/Linux 系统 三、安装 Tomcat(一)Windows 系统(二)Linux …...
科技快讯 | DeepSeek 公布模型新学习方式;Meta发布开源大模型Llama 4;谷歌推出 Android Auto 14.0 正式版
Meta发布开源大模型Llama 4,首次采用“混合专家架构“ 4月6日,Meta推出开源AI模型Llama 4,包括Scout和Maverick两个版本,具备多模态处理能力。Scout和Maverick参数量分别为170亿和4000亿,采用混合专家架构。Meta同时训…...
skynet.netpack四个核心函数详解
目录 1. netpack.filter(queue, msg, sz)2. netpack.pop(queue)3. netpack.tostring(msg, sz)4. netpack.clear(queue)完整使用场景示例总结 在 Skynet 中,netpack 模块提供了四个核心函数,用于处理网络数据包的接收、粘包解析和队列管理。以下是这四个函…...
Zephyr与Linux核心区别及适用领域分析
一、核心定位与目标场景 特性Zephyr RTOSLinux目标领域物联网终端、实时控制系统(资源受限设备)服务器、桌面系统、复杂嵌入式设备(如路由器)典型硬件MCU(ARM Cortex-M, RISC-V),内存<1MBMP…...
Linux网络编程(十五)——优于select的epoll
文章目录 15 优于select的epoll 15.1 epoll理解及应用 15.1.1 基于select的I/O复用技术速度慢的原因 15.1.2 select的优点 15.1.3 实现epoll时必要的函数和结构体 15.1.4 epoll_creat1 15.1.5 epoll_ctl 15.1.6 epoll_wait 15.1.7 基于epoll的回声服务器端 15.2 条件…...
PhotoShop学习07
1.为图像添加纹理 图层混合模式是混合 2 张图片的一种快捷方式,一般情况下为图片添加纹理外观可以用到混合模式。 这里有一副图片,我可以为其添加纹理,使之呈现出不同的效果。首先需要为当前图层添加一个纹理图片,可以使用置入嵌…...
【缓存击穿】Java的“SingleFlight”解决方案
在Java中实现类似Golang的SingleFlight机制,可以通过以下步骤解决缓存击穿问题。该方案使用ConcurrentHashMap管理并发请求,并通过CompletableFuture实现异步结果合并。 实现代码 import java.util.concurrent.Callable; import java.util.concurrent.…...
createContext+useContext+useReducer组合管理React复杂状态
createContext、useContext 和 useReducer 的组合是 React 中管理全局状态的一种常见模式。这种模式非常适合在不引入第三方状态管理库(如 Redux)的情况下,管理复杂的全局状态。 以下是一个经典的例子,展示如何使用 createContex…...
海外直播平台交互设计师简历模板
营销团队管理技巧培训PPT啊,其实是一个非常有用的工具呢!它不仅能帮助管理者梳理思路,还能让团队成员快速掌握关键技能。说实话,一个好的PPT就像一位优秀的导师,在会议室里就能让人眼前一亮!比如有一次我参…...
基于springboot微信小程序课堂签到及提问系统(源码+lw+部署文档+讲解),源码可白嫖!
摘要 随着信息时代的来临,过去的课堂签到及提问管理方式的缺点逐渐暴露,本次对过去的课堂签到及提问管理方式的缺点进行分析,采取计算机方式构建基于微信小程序的课堂签到及提问系统。本文通过阅读相关文献,研究国内外相关技术&a…...
MCU软件开发使用指针有哪些坑?
目录 1、空指针访问 2、野指针(未初始化的指针) 3、指针越界 4、内存泄漏 5、悬空指针 6、指针类型不匹配 7、多任务环境中的指针访问 8、对齐问题 在MCU软件开发中,使用指针虽然可以提高程序的灵活性和性能,但也存在许多…...
ubuntu 20.04 编译和运行SC-LeGo-LOAM
1.搭建文件目录和clone代码 mkdir -p SC-LeGo-LOAM/src cd SC-LeGo-LOAM/src git clone https://github.com/AbangLZU/SC-LeGO-LOAM.git cd .. 2.修改代码 需要注意的是原作者使用的是Ouster OS-64雷达,需要更改utility.h文件中适配自己的雷达类型,而…...
FPGA_DDR(一) 仿真
对ddr进行读写实验,用了vivado的ddr的模型进行仿真 1 创建AXI_mig的ip核 选择axi 选择自己的型号,这里是ddr的位宽32,但是axi的话是256位宽 选择nobuffer,没有缓冲器 选择自己匹配引脚 默认 后面默认即可 生成ip,时间…...
【Spec2MP:项目管理之项目人力管理】
芯片设计项目中如何打造战斗力强悍的团队? 引言:芯片设计项目的核心是人 芯片设计是一项高度复杂、跨学科协作的工程,团队的专业性、协作效率和凝聚力直接影响项目成败。本文结合某芯片项目人力管理文档,从目标、职责、价值观、架…...
windows10下PointNet官方代码Pytorch实现
PointNet模型运行 1.下载源码并安装环境 GitCode - 全球开发者的开源社区,开源代码托管平台GitCode是面向全球开发者的开源社区,包括原创博客,开源代码托管,代码协作,项目管理等。与开发者社区互动,提升您的研发效率和质量。https://gitcode.com/gh_mirrors/po/pointnet.pyto…...
阿里云大模型训练与推理开发
本文主要描述阿里云大模型开发环境的搭建、训练数据集的制作流程、大模型如何训练数据集以及如何利用已训练完成的模型执行推理。 开发环境搭建 ModelScope社区是阿里云通义千问开源的大模型开发者社区。 如上所示,安装ModelScope社区大模型基础库开发框架的命令行…...
图灵逆向——题一-动态数据采集
目录列表 过程分析代码实现 过程分析 第一题比较简单,直接抓包即可,没有任何反爬(好像头都不用加。。。) 代码实现 答案代码如下: """ -*- coding: utf-8 -*- File : .py author : 鲨鱼爱兜兜 T…...
GS069W电动工具直流调速电路深度解析
产品概述 GS069W是我们推出的CMOS专用调速集成电路,采用SOP8封装,内置15V稳压结构,具有宽电压输入(4-24V)、低功耗、强抗干扰等特点,专为电动工具调速设计。 核心参数 工作电压:4-24Vÿ…...
PyQt6实例_A股日数据维护工具_下载某个股票未复权数据
目录 前置: 相关代码: 1 工作类 2 数据库交互 3 主界面调用 视频 前置: 1 本系列将以 “PyQt6实例_A股日数据维护工具” 开头放置在“PyQt6实例”专栏 2 日数据可在“数据库”专栏,“PostgreSQL_”开头系列博文中获取 3 权…...
【蓝桥杯】算法笔记6
1. 可行性剪枝应用 1.1. 题目 题目描述: 给定一个正整数n和一个正整数目标值target,以及一个由不同正整数组成的数组nums。要求从nums中选出若干个数,每个数可以被选多次,使得这些数的和恰好等于target。问有多少种不同的组合方式? 输入: 第一行:n和target,表示数组…...
C++ 中日期类的输入输出操作符重载实践
目录 引言 预备知识 输出流操作符 operator<< 重载 为什么要返回 ostream& 输入流操作符 operator>> 重载 实现思路 测试代码 总结 引言 在 C 编程中,当我们自定义数据类型时,为了让其能像内置类型一样方便地进行输入输出操…...
图论:最小生成树
最小生成树 (无向无环图) 概念 1.Prim算法 P3366 【模板】最小生成树 - 洛谷 邻接矩阵实现 #include<iostream> #include<cstring> using namespace std; const int INF 0x3f3f3f3f; const int N 5e3 10; int dis[N]; //记录每个结点到…...
linux中CosyVoice声音克隆安装教程——TTS文本转语音(数字人组件)
CosyVoice 作为一款先进的语音合成解决方案,其设计理念在于提供高效、稳定且灵活的语音生成工具。本教程将从环境配置、依赖安装、模型下载到服务部署全流程进行详细介绍,旨在为用户提供前瞻性的技术指导,同时兼顾细节解析和专业名词解释&…...
智能手表该存什么音频和文本?场景化存储指南
文章目录 为什么需要“场景化存储”?智能手表的定位手机替代不了的场景碎片化的场景存储 音频篇:智能手表该存什么音乐和音频?运动场景通勤场景健康场景 文本篇:哪些文字信息值得放进手表?(部分情况可使用图…...
怎么检查网站CDN缓存是否生效
为什么要使用CDN缓存? 网站使用缓存可显著提升加载速度,减少服务器负载和带宽消耗,优化用户体验,增强架构稳定性,助力SEO优化,实现资源高效利用与性能平衡。 通过合理配置 CDN 缓存策略,可降低…...
win10安装gitbash工具
问题描述:在Windows下没有预装bash命令处理工具 # WInR输入cmd回车进入命令行,执行以下命令出现乱码 bash 无法使用bash命令 解决方案:下载安装gitbash命令行工具 Git Bash 是一个在 Windows 上运行的终端仿真器,集成了 Git 和 Bash shell࿰…...
买不起了,iPhone 或涨价 40% ?
周知的原因,新关税对 iPhone 的打击,可以说非常严重。 根据 Rosenblatt Securities分析师的预测,若苹果完全把成本转移给消费者。 iPhone 16 标配版的价格,可能上涨43%。 iPhone 16 标配的价格是799美元,上涨43%&am…...
企业级 ClickHouse Docker 离线部署实践指南20250407
企业级 ClickHouse Docker 离线部署实践指南 引言 在数据分析与日志处理日益重要的今天,ClickHouse 凭借其高性能、列式存储架构,成为企业在大数据分析中的首选引擎之一。本文基于一位金融行业从业者在离线网络环境中部署 ClickHouse 的真实实践过程&a…...
多域名 SSL 证书能保护多少个域名?
一、基础保护数量范围 多域名 SSL 证书,顾名思义,可保护多个不同域名。通常情况下,不同证书颁发机构(CA)设定的基础保护数量有所差异。一般的多域名 SSL 证书能保护2 至 5 个域名,这些域名可以是完全独立…...
Linux系统学习Day04 阻塞特性,文件状态及文件夹查询
知识点4【文件的阻塞特性】 文件描述符 默认为 阻塞 的 比如:我们读取文件数据的时候,如果文件缓冲区没有数据,就需要等待数据的到来,这就是阻塞 当然写入的时候,如果发现缓冲区是满的,也需要等待刷新缓…...
【AI】高效地使用 AI 模型的 Prompt(提示词)
明确任务和目标 在使用 Prompt 之前,要清楚知道自己想要通过 AI 模型完成什么任务,例如生成文本、回答问题、进行翻译或创作故事等。明确的目标有助于构建更有针对性的 Prompt,引导模型生成符合期望的结果。 精准描述问题 提供具体细节&am…...
第二十:mysql——Undo Log、Redo Log和Binlog
二进制日志binlog(归档日志)、 事务日志redo log(重做日志) MySQL实例挂了或者宕机了,重启的时候InnoDB存储引擎会使用rede log日志恢复数据,保证事务的持久性和完整性 和undo log(回滚日志&a…...
LogicFlow-前端流程图开发
LogicFlow-前端流程图开发 一、安装使用 1、安装logicflow 通过npm安装logicflow npm install logicflow/core --save# 插件包(不使用插件时不需要引入) npm install logicflow/extension --save2、创建实例 import LogicFlow from "logicflow/…...
第四讲:类与对象(下)
目录 1、再谈构造函数 1.1、构造函数体赋值 1.2、初始化列表 1.3、explicit关键字 2、static成员 3、友元 3.1、友元函数 3.2、友元类 4、内部类 5、匿名对象 6、拷贝对象时的优化(了解) 7、重新理解类与对象 8、日期类的实现 9、练习题 9…...
ReAct 框架 | 提示词工程(1)
ReAct 框架 1、什么是 ReAct 框架?2、基于 ReAct 框架的提示词3、结合 LangChain 框架使用4、总结 1、什么是 ReAct 框架? ReAct : Reasoning Acting ,将推理与外部工具调用结合,通过交互式探索解决复杂问题。 优点…...
第一部分——Docker篇 第一章 Docker容器
关于系统的改造探索 开篇:系统改造的调研报告 第一部分——Docker篇 第一章 Docker容器 第二章 Docker安装 第三章 构建自定义镜像 第四章 搭建镜像仓库 第五章 容器编排 第六章 容器监控 文章目录 关于系统的改造探索第一部分——Docker篇 前言一、就是你了——…...
ubuntu,react的学习(1)
在此目录下,开启命令行 /home/kt/react 如下操作 tkt4028:~/react$ npm create vitelatest task-manager -- --template react Need to install the following packages: create-vite6.3.1 Ok to proceed? (y) y> npx > cva task-manager --template react…...
AR 赋能儿童娱乐:剧本杀与寻宝小程序搭建秘籍
在科技飞速发展的当下,儿童娱乐领域正经历着一场创新变革。AR(增强现实)技术的融入,为儿童剧本杀与寻宝游戏带来了前所未有的沉浸式体验。通过搭建专属小程序,孩子们能够在虚拟与现实交织的世界中开启奇幻冒险。接下来…...
2017年-全国大学生数学建模竞赛(CUMCM)试题速浏、分类及浅析
2017年-全国大学生数学建模竞赛(CUMCM)试题速浏、分类及浅析 全国大学生数学建模竞赛(China Undergraduate Mathematical Contest in Modeling)是国家教委高教司和中国工业与应用数学学会共同主办的面向全国大学生的群众性科技活动,目的在于激励学生学习数学的积极性,提高学…...
密码学基础——分组密码的运行模式
前面的文章中文我们已经知道了分组密码是一种对称密钥密码体制,其工作原理可以概括为将明文消息分割成固定长度的分组,然后对每个分组分别进行加密处理。 下面介绍分组密码的运行模式 1.电码本模式(ECB) 2.密码分组链接模式&…...
zk源码—2.通信协议和客户端原理一
大纲 1.ZooKeeper如何进行序列化 2.深入分析Jute的底层实现原理 3.ZooKeeper的网络通信协议详解 4.客户端的核心组件和初始化过程 5.客户端核心组件HostProvider 6.客户端核心组件ClientCnxn 7.客户端工作原理之会话创建过程 1.ZooKeeper如何进行序列化 (1)什么是序列化…...
【NLP】Transformer网络结构(2)
一、Transformer 整体架构 Transformer 由 Encoder 和 Decoder 堆叠组成,每个 Encoder/Decoder 层包含以下核心模块: Encoder 层:Multi-Head Self-Attention → Add & LayerNorm → Feed-Forward → Add & LayerNormDecoder 层&…...
【LeetCode77】组合
题目描述 给定区间 [1, n] 和一个整数 k,需要返回所有可能的 k 个数的组合。 思路 算法选择:回溯算法 回溯算法是一种试探性搜索方法,非常适合用来解决组合问题。基本思想是: 从数字 1 开始,逐步构建组合。当当前组…...