点对点通信就像是私聊。websocket 实现点对点通信可以采用三种方式来实现。
<br/>
下面分别用三种方法来实现 zhangsan 与 lisi 的私聊。
[TOC]
# 1. convertAndSend方法实现
步骤如下:
**1. 调用方法convertAndSend**
```java
@RestController
public class TestController {
@Autowired
private SimpMessagingTemplate messagingTemplate;
@MessageMapping("/v5/point")
public void pointChat01(Map<String, Object> params) {
//关键点就是订阅地址需要是动态的
messagingTemplate.convertAndSend("/topic/v5/chat/" + params.get("to"), params.get("message"));
}
}
```
**2. 前端代码**
```js
//订阅地址是:/topic/v5/chat/lisi,下面发送时传入了 'to':'lisi' 到后端组成了该地址
client.subscribe('/topic/v5/chat/lisi', (res) => {})
client.send('/app/v5/point', {}, JSON.stringify({'to':'lisi', 'message':'你好,李四.'}))
```
<br/>
# 2. @SendToUser实现
步骤如下:
**1. 配置类中设置setUserDestinationPrefix**
```java
@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {
@Override
public void registerStompEndpoints(StompEndpointRegistry registry) {
registry.addEndpoint("/endpoint-websocket")
.setAllowedOrigins("*")
.withSockJS();
}
@Override
public void configureMessageBroker(MessageBrokerRegistry registry) {
registry.enableSimpleBroker("/topic");
registry.setApplicationDestinationPrefixes("/app");
//点对点通信时客户端需要添加的前缀,不设置默认为/user
registry.setUserDestinationPrefix("/users/");
}
}
```
**2. 方法上标注注解`@SendToUser`**
```java
@RestController
public class TestController {
@Autowired
private SimpMessagingTemplate messagingTemplate;
/**
* 缺点就是订阅地址是死的,zhangsan发送消息给lisi,则必须有一个@SendToUser("/topic/v6/lisi")方法
* 反过来又必须有一个 @SendToUser("/topic/v6/zhangsan") 的方法。
*/
@MessageMapping("/v6/point")
@SendToUser("/topic/v6/lisi")
public String pointChat02(String message) {
return message;
}
}
```
**3. 前端代码**
```js
//添加了前缀 /users
client.subscribe('/users/topic/v6/lisi', (res) => {})
client.send('/app/v6/point', {}, '你好,李四.')
```
<br/>
# 3. convertAndSendToUser方法实现
步骤如下:
**1. 配置类中设置setUserDestinationPrefix**
```java
@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {
@Override
public void registerStompEndpoints(StompEndpointRegistry registry) {
registry.addEndpoint("/endpoint-websocket")
.setAllowedOrigins("*")
.withSockJS();
}
@Override
public void configureMessageBroker(MessageBrokerRegistry registry) {
registry.enableSimpleBroker("/topic");
registry.setApplicationDestinationPrefixes("/app");
//点对点通信时客户端需要添加的前缀,不设置默认为/user
registry.setUserDestinationPrefix("/users/");
}
}
```
**2. 存储用户连接的sessionId**
```java
@Component
public class ConnectListener implements ApplicationListener<SessionConnectEvent> {
@Override
public void onApplicationEvent(SessionConnectEvent event) {
StompHeaderAccessor header = StompHeaderAccessor.wrap(event.getMessage());
//下面将通过前端传递一个key为userId的请求头
String userId = header.getFirstNativeHeader("userId");
String sessionId = header.getSessionId();
//UserConstants.usersMap是我自己创建的一个public static Map对象,用来存储每个用户的sessionId
UserConstants.usersMap.put(userId, sessionId);
}
}
```
**3. 调用方法convertAndSendToUser**
```java
@RestController
public class TestController {
@Autowired
private SimpMessagingTemplate messagingTemplate;
/**
* 方法 convertAndSendToUser(String user, String destination, Object payload, @Nullable Map<String, Object> headers)
* user:用户的sessionId
* destination:订阅地址
* payload:订阅的数据
* headers:请求头
*/
@MessageMapping("/v7/point")
public void pointChat03(Map<String, Object> params) {
String sessionId = UserConstants.usersMap.get((String) params.get("userId"));
SimpMessageHeaderAccessor header = SimpMessageHeaderAccessor.create(SimpMessageType.MESSAGE);
//把sessionId放到请求头中
header.setSessionId(sessionId);
header.setLeaveMutable(true);
//必须以配置类中的 registry.enableSimpleBroker("/topic") 注册的 /topic 为前缀
messagingTemplate.convertAndSendToUser(sessionId, "/topic/message", "你好,李四!", header.getMessageHeaders());
}
}
```
**4. 前端代码**
```js
//前端连接的时候传入key为userId的请求头
client.connect({'userId': 'lisi'}, () => {
//订阅地址添加 /users 为前缀
client.subscribe('/users/topic/message', (res) => {})
})
//这里我也传了 userId 去与后端匹配
client.send('/app/v7/point', {}, JSON.stringify({'userId': 'lisi', 'message': '你好,李四.'}))
```
- 跨域问题
- 跨域是什么
- 跨域解决方案
- 从后端解决
- nginx反向代理
- WebSocket
- websocket是什么
- websocket协议
- 使用场景
- 实现方式
- 注解与html5原生方式
- websocketAPI
- 实现步骤
- 文件上传
- 文件下载
- 广播通信
- 定时推送
- 编程与socketjs方式
- socketjs与stompjs框架
- 实现步骤
- 重载目的地
- SimpMessagingTemplate
- 定时向前端推送数据
- 5种监听事件
- 点对点通信
- 拦截器
- HandshakeInterceptor
- ChannelInterceptor
- poi之excel表格
- 表格版本
- POI常用类
- POI依赖
- 写表格
- 编写表格过程
- 单元格边框样式
- 单元格背景色
- 冻结行或列
- 单元格合并
- 单元格内换行
- 文档内跳转
- 读表格
- Web中的Excel操作
- 导出表格
- 读取表格
- poi之word文档
- word版本
- 写word
- 基本使用
- 标题样式
- 添加图片
- EasyExcel表格
- EasyExcel是什么
- 与其他Excel工具对比
- EasyExcel依赖
- 读Excel
- 简单读取
- 指定列位置
- 读取多个sheet
- 格式转换
- 多行表头
- 同步读
- 写Excel
- 简单写入
- 单元格样式
- 拦截器
- 列宽
- 冻结行或列
- 合并单元格
- 填充Excel
- SpringSecurity
- SpringSecurity是什么
- 同类型产品对比
- 环境搭建
- 相关概念
- 密码加密
- Web权限控制
- UserDetailsService接口
- 登录认证
- 自定义登录页
- 未授权跳转登录页
- 权限控制
- 自定义403页面
- 权限注解
- 记住我功能
- 注销功能
- CSRF
- CSRF是什么
- CSRF保护演示
- 前后端分离权限控制
- 环境搭建
- 认证实现
- 会话管理
- 动态权限管理
- 微服务权限控制
- 权限控制方案
- SpringBoot整合RabbitMQ
- 整合步骤
- Fanout交换机演示
- Direct交换机演示
- Topic交换机演示
- @RabbitListener方法
- JWT认证与授权
- 环境搭建
- 密码加密
- 认证与授权