前言:

两年前做过spring+activemq+stomp的ws推送,那个做起来很简单,但现在公司用的mq中间件是rabbitmq,因此需要通过rabbitmq去做ws通信。仔细搜了搜百度/谷歌,网上通过spring boot+rabbitmq+stomp的教程文章倒是一搜一大把,可惜目前的项目是非spring boot的,没法套用。只好自己去捣鼓。搞了几个小时,终于弄出来了,特此与大家分享下。

RabbitMQ:

怎么安装就不是本篇讨论的话题了,自己百度/谷歌之。rabbitmq默认自带了stomp插件,但是需要自己启用。命令为:

rabbitmq-plugins enable rabbitmq_stomp

来来来,给个文档地址参考参考,http://www.rabbitmq.com/stomp.html。默认用guest用户去连接,密码也是guest。

这里有个问题,看rabbitmq配置文件,stomp协议端口默认是61613,但是用ws协议连接却始终连接不上,所以只能用web stomp端口,端口号是15674,这个跟activemq有所区别。(P.S. 此处最好有大神来解惑,或者告知如何用61613来连

Javascript:

前端代码撸起来最方便,关键是调试也容易,因此先来。

var stompClient = null;

var headers = {
login: 'guest',
passcode: 'guest'
}; function wsConnect(url) {
var ws = new SockJS(url);
stompClient = Stomp.over(ws); //var ws = new WebSocket(url);
//stompClient = Stomp.over(ws); // SockJS does not support heart-beat: disable heart-beats
stompClient.heartbeat.outgoing = 0;
stompClient.heartbeat.incoming = 0; stompClient.connect(headers, function (frame) {
console.log('Connected: ' + frame); stompClient.subscribe('/topic/test', function (sms) {
var obj = JSON.parse(sms.body)
var count = obj.totalCount; console.log("count: " + count);
}); });
}

然后就连接呗。

$(function(){
var url = "http://host:15674/stomp";
wsConnect(url);
});

撸完准备测试,当然是选择chrome喽,页面加载后,打开console控制台,可以看到web socket连上了,前端大功告成。

  

Java:

定义一个StompService类专门用来发送stomp消息。注意:rabbitmq 3.7以后stomp插件不再支持sockjs,因此写法会有变化。

import org.springframework.beans.factory.annotation.Value;
import org.springframework.messaging.converter.StringMessageConverter;
import org.springframework.messaging.simp.stomp.StompHeaders;
import org.springframework.stereotype.Component;
import org.springframework.web.socket.WebSocketHttpHeaders;
import org.springframework.web.socket.client.WebSocketClient;
import org.springframework.web.socket.client.standard.StandardWebSocketClient;
import org.springframework.web.socket.messaging.WebSocketStompClient;
import org.springframework.web.socket.sockjs.client.SockJsClient;
import org.springframework.web.socket.sockjs.client.Transport;
import org.springframework.web.socket.sockjs.client.WebSocketTransport; import javax.annotation.PostConstruct;
import java.util.ArrayList;
import java.util.List; /**
* stomp服务 rabbitmq做中间件
* @author Selwyn
* @version $Id: WebSocketConfig.java, v 0.1 9/7/2018 9:59 AM Selwyn Exp $
*/
@Component
public class StompService { private static final String URL_TEMPLATE = "http://%s:%s/stomp"; @Value("${rabbit.host}")
private String host; //@Value("${rabbit.stomp.port}")
private Integer port = 15674; /**
* 连接用户名
*/
//@Value("${rabbit.stomp.login}")
private String login = "guest";
/**
* 连接密码
*/
//@Value("${rabbit.stomp.passCode}")
private String passCode = "guest"; private String url; @PostConstruct
public void init()
{
url = String.format(URL_TEMPLATE, host, port);
} /**
* 发送stomp消息
* @param dest 目的地 比如/topic/test
* @param toSend 要发送的信息
* @param <T>
*/
public <T> void connectAndSend(String dest, T toSend)
{
WebSocketClient client = new StandardWebSocketClient(); List<Transport> transports = new ArrayList<>(1);
transports.add(new WebSocketTransport( client) );
//rabbitmq 3.7以后就别这么写了。直接new WebSocketStompClient(client)就行
WebSocketClient transport = new SockJsClient(transports);
WebSocketStompClient stompClient = new WebSocketStompClient(transport);
//StompSessionHandlerAdapter默认的payload类型是String, 因此MessageConverter必须是StringMessageConverter
stompClient.setMessageConverter(new StringMessageConverter()); final CustomStompSessionHandler sessionHandler =
new CustomStompSessionHandler(dest, toSend); WebSocketHttpHeaders headers = new WebSocketHttpHeaders();
headers.setSecWebSocketProtocol("13"); //连接用户名/密码也是必须的,否则连不上
StompHeaders sHeaders = new StompHeaders();
sHeaders.add("login", this.login);
sHeaders.add("passcode", this.passCode); //开始连接,回调连接上后发送stomp消息
stompClient.connect(url, headers, sHeaders, sessionHandler); //要同步得到发送结果的话,用CountDownLatch来做或者connect结果的future对象做get
} }

然后编写回调handler类。

import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.springframework.messaging.simp.stomp.StompCommand;
import org.springframework.messaging.simp.stomp.StompHeaders;
import org.springframework.messaging.simp.stomp.StompSession;
import org.springframework.messaging.simp.stomp.StompSessionHandlerAdapter; /**
* 自定义stomp session 回调handler
* @author Selwyn
* @version $Id: CustomStompSessionHandler.java, v 0.1 9/7/2018 3:43 PM Selwyn Exp $
*/
@Slf4j
public class CustomStompSessionHandler extends StompSessionHandlerAdapter { /**
* 要发送的对象,将会json化传输出去
*/
private Object toSend; /**
* 目的地,一般是topic地址
*/
private String dest; public CustomStompSessionHandler(String dest, Object toSend) {
this.toSend = toSend;
this.dest = dest;
} @Override
public void handleFrame(StompHeaders headers, Object payload) {
super.handleFrame(headers, payload);
} @Override
public void afterConnected(StompSession session, StompHeaders connectedHeaders) {
super.afterConnected(session, connectedHeaders);
String msg = JSON.toJSONString(toSend);
try{
session.send(dest, msg);
}catch(Exception e)
{
log.error("failed to send stomp msg({}) to destination {}", msg, dest);
}finally {
//做完了关闭呗
session.disconnect();
}
} @Override
public void handleException(StompSession session, StompCommand command, StompHeaders headers, byte[] payload, Throwable exception) {
super.handleException(session, command, headers, payload, exception);
log.error("stomp error: {}", exception);
} @Override
public void handleTransportError(StompSession session, Throwable exception) {
super.handleTransportError(session, exception);
log.error("stomp transport error: {}", exception);
} public void setToSend(Object toSend) {
this.toSend = toSend;
} public void setDest(String dest) {
this.dest = dest;
}
}

再自己写个controller或者写个单元测试方法,这里就不给出代码了,撸完后启动服务,就可以测试消息推送了,实践证明,真香!

结尾:

其实整个过程还没完,需要考虑到连接中断等情况,客户端和服务后台都需要做好重连机制。通过sockjs这种方式连接是没有心跳机制的,这个比activemq带的stomp插件要low。个人建议,如果能用spring boot的话尽量用那种方式去实现stomp消息推送。

  

spring+rabbitmq+stomp搭建websocket消息推送(非spring boot方式)的更多相关文章

  1. 搭建websocket消息推送服务,必须要考虑的几个问题

    近年,不论是正在快速增长的直播,远程教育以及IM聊天场景,还是在常规企业级系统中用到的系统提醒,对websocket的需求越来越大,对websocket的要求也越来越高.从早期对websocket的应 ...

  2. node.js Websocket消息推送---GoEasy

    Goeasy, 它是一款第三方推送服务平台,使用它的API可以轻松搞定实时推送!个人感觉goeasy推送更稳定,推送 速度快,代码简单易懂上手快 浏览器兼容性:GoEasy推送 支持websocket ...

  3. C(++) Websocket消息推送---GoEasy

    Goeasy, 它是一款第三方推送服务平台,使用它的API可以轻松搞定实时推送!个人感觉goeasy推送更稳定,推送 速度快,代码简单易懂上手快 浏览器兼容性:GoEasy推送 支持websocket ...

  4. 【WebSocket】WebSocket消息推送

    准备使用WebSocket实现Java与Vue或者安卓间的实时通信,实现私密聊天.群聊.查询下资料备用. WebSocket客户端 websocket允许通过JavaScript建立与远程服务器的连接 ...

  5. spring boot下WebSocket消息推送(转)

    原文地址:https://www.cnblogs.com/betterboyz/p/8669879.html WebSocket协议 WebSocket是一种在单个TCP连接上进行全双工通讯的协议.W ...

  6. spring boot下WebSocket消息推送

    WebSocket协议 WebSocket是一种在单个TCP连接上进行全双工通讯的协议.WebSocket通信协议于2011年被IETF定为标准RFC 6455,并由RFC7936补充规范.WebSo ...

  7. websocket消息推送实现

    一.服务层 package com.demo.websocket; import java.io.IOException; import java.util.Iterator; import java ...

  8. WebSocket消息推送

    WebSocket协议是基于TCP的一种新的网络协议,应用层,是TCP/IP协议的子集. 它实现了浏览器与服务器全双工(full-duplex)通信,客户端和服务器都可以向对方主动发送和接收数据.在J ...

  9. swoole WebSocket 消息推送

    server.php <?php //连接本地的 Redis 服务 $redis = new Redis(); $redis->connect('127.0.0.1', 6379); $r ...

随机推荐

  1. Java入门程序

    JavaC.exe 编译器,编译.java文件 Java.exe 解释器,执行class文件 编译命令  javac HelloWorld.java 编译后 会产生同名的.class文件 javac编 ...

  2. 真的懂了:TCP协议中的三次握手和四次挥手(关闭连接时, 当收到对方的FIN报文时, 仅仅表示对方不在发送数据了, 但是还能接收数据, 己方也未必全部数据都发送对方了。相当于一开始还没接上话不要紧,后来接上话以后得让人把话讲完)

    一.TCP报文格式 下面是TCP报文格式图: (1) 序号, Seq(Sequence number), 占32位,用来标识从TCP源端向目的端发送的字节流,发起方发送数据时对此进行标记. (2) 确 ...

  3. OpenCV调试利器——Image Watch插件的安装和使用

    各大编译工具在调试的时候都可以实时查看变量的值,了解变量值的变动情况,在图像处理相关的程序调试中,是否也可以实时查看内存中图像变量的图形信息以及图像上指定区域或点位的数值变化情况呢? 在工业机器视觉领 ...

  4. 微信4.5 for Android安卓内测版体验【实时对讲】杀手级应用下载

    微信4.5 for Android 安卓 内测版 体验 程序启动画面,是一支在动的烛光 主要功能更新如下 支持语音提醒,到时间后自动弹出消息框 发起语音提醒请求 成功识别语音请求,并且保存在本地,应该 ...

  5. JavaScript:undefined And null差异

    班吃饭的时候,同事偶然问了一个问题:undefined和null究竟有什么差别?无法回答,回去查阅相关文档,算了有了一个了解,做相关的总结.在開始之前,请看例如以下代码,算是抛出这个问题: conso ...

  6. 潜移默化学会WPF(绚丽篇)--热烈欢迎RadioButton,改造成功,改造成ImageButton,新版导航

    原文:潜移默化学会WPF(绚丽篇)--热烈欢迎RadioButton,改造成功,改造成ImageButton,新版导航 本样式 含有  触发器 和 动画    模板  ,多条件触发器,还有布局 本人博 ...

  7. Easyui Tab刷新

    Easyui Tab刷新: function refreshTab(title){ var tab = $('#id').tab('getTab',title); $('#id').tab('upda ...

  8. Emgu-WPF学习使用-Rectangle识别

    原文:Emgu-WPF学习使用-Rectangle识别 环境:Win8 64位 Vs2015 Emgu 版本:emgucv-windesktop 3.2.0.2682 示例图上部流程:原图->灰 ...

  9. ASP.NET CORE系列【六】Entity Framework Core 之数据迁移

    原文:ASP.NET CORE系列[六]Entity Framework Core 之数据迁移 前言 最近打算用.NET Core写一份简单的后台系统,来练练手 然后又用到了Entity Framew ...

  10. 如何设置程序UAC控制

    在做项目的过程中,有很多情况会涉及到权限问题,要求必须以管理员的身份才能运行,如何强制我们的程序必须以管理员身份运行呢?在调查的过程中发现有很多方式,此处介绍一种简单的方式. 1.在VS中,右键点击工 ...