当前位置: 首页 > news >正文

搭建IM服务 so easy

为什么80%的码农都做不了架构师?>>>   hot3.png

现在很多网站、APP都通过IM服务来实现消息推送及数据即时同步功能,即时通讯组件逐渐成为产品的标配。目前国内有很多成熟稳定的第三方即时通讯服务厂家,比如:融云。使用这些专业的服务可以提高开发效率而且服务稳定有保障。

如果自己DIY或者需要在封闭的局域网内使用IM服务,该怎么办呢?下文就简单介绍一下EasyPM曾经实践过的自行搭建IM服务过程。

数据同步方式

实现数据同步,有"推"、"拉" 两种思路,具体有以下几种方式:

  1. 使用HTTP轮循方式
    • 说明:定时向HTTP服务端接口(Web Service API)获取最新消息,可结合ajax技术实现页面无刷新效果,这是主动拉取消息的机制。
    • 优点:实现简单、可控性强、部署成本低
    • 缺点:实时性差,增加服务端负载
  2. 使用XMPP协议
    • 说明:XMPP是基于可扩展标记语言(XML)的协议,它用于即时消息(IM)以及在线现场探测。它促进在服务器之间的准即时操作,其前身是Jabber,是一个开源形式组织产生的网络即时通信协议。XMPP目前被IETF国际标准组织完成了标准化工作。
    • 优点:协议成熟、强大、可扩展性强、目前主要应用于众多IM系统
    • 缺点:协议比较复杂、冗余(基于XML)、费流量
  3. 使用MQTT协议
    • 说明:MQTT(Message Queuing Telemetry Transport,消息队列遥测传输)是IBM开发的一个即时通讯协议,该协议支持所有平台,是轻量级的、基于代理的“发布/订阅”模式的消息传输协议
    • 优点:MQTT协议简洁、可扩展性强、流量开销很小、网络传输时间短
    • 缺点:还不够成熟、实现较复杂

EasyPM作为一个团队协作应用,消息/数据推送功能是不可或缺的。因为MQTT比较轻量且网络开销小等特点,我们选择了支持MQTT协议的Apollo。

Apollo是什么?

Apollo是apache旗下的基金项目,它是以Apache ActiveMQ5.x为基础,采用全新的线程和消息调度架构重新实现的消息中间件,针对多核处理器进行了优化处理,它的速度更快、更可靠、更易于维护。apollo与ActiveQQ一样支持多协议:STOMP、AMQP、MQTT、Openwire、 SSL、WebSockets,本文只介绍MQTT协议的使用。

mqtt

下载Apollo

进入Apollo下载页面 ,选择下载合适的版本

如果操作是系统是Windows Vista或更高版本,则需要安装Microsoft Visual C++ 2010 Redistributable: 64位JVM 32位JVM

创建Apollo实例及服务

  1. 创建实例 进入E:\apache-apollo-1.7之下的bin目录,打开cmd窗口,执行命令:apollo create D:\apollo_broker,命令执行成功后,在D盘下会有apollo_broker目录,这便是apollo的服务实例,apollo之旅便从这里开始。 在D:\apollo_broker下有个bin目录,其中有两个文件: apollo-broker.cmd是通过cmd命令启动apollo服务的 apollo-broker-service.exe,是用于创建window服务的
  2. 命令行启动服务 在D:\apollo_broker\bin目录下打开cmd窗口,执行apollo-broker run命令来启动apollo服务, 启动成功可以在浏览器中查看运行情况,访问地址为 http://127.0.0.1:61680 , 默认用户名/密码:admin/password
  3. 创建windows服务 找到cmd.exe文件,点击鼠标右键,以管理员身份运行,输入创建windows服务命令,如下图:

创建成功后,在windows服务中会有一个apollo_broker服务,设置随系统自动启动

MQTT协议的应用

MQTT协议有众多客户端实现,相关客户端请参考apollo官方文档 本文采用eclipse的paho客户端实现

####web端接收消息介绍 将 javascript客户端项目下载下来,并在其项目根目录下执行mvn命令,进行编译,生成target目录,其下生成mqttws31.js、mqttws31-min.js两个js文件,将其拷贝到自己项目相关目录下,并在页面中引用,即可实现javascript客户端的消息订阅和发布,demo代码如下:

var client = new Paho.MQTT.Client(location.hostname, 61623,"/", "clientId"); 
/* 61623是ws连接的默认端口,可以在apollo中间件中进行配置
(关于apollo的配置请参考:
 http://activemq.apache.org/apollo/documentation/user-manual.html
) */
// set callback handlers 
client.onConnectionLost = onConnectionLost; 
client.onMessageArrived = onMessageArrived; 
// connect the client 
client.connect({userName:'admin',password:'password',onSuccess:onConnect}); 
// called when the client connects 
function onConnect() { // 连接成功后的处理 
		// Once a connection has been made, make a subscription and send a message. 
		console.log("onConnect"); 
		client.subscribe("/topic/event"); // 订阅消息的主题 
		var message = new Paho.MQTT.Message("Hello,this is a test"); 
		message.destinationName = "/topic/event"; 
		client.send(message); // 发送消息 
} 
// called when the client loses its connection 
function onConnectionLost(responseObject) { // 连接丢失后的处理 
		if (responseObject.errorCode !== 0) { 
				console.log("onConnectionLost:"+responseObject.errorMessage); 
		} 
} 
// called when a message arrives 
function onMessageArrived(message) { // 消息接收成功后的处理 
		console.log("onMessageArrived:"+message.payloadString); 
}

####服务端消息发送介绍 paho java客户端目前只支持J2SE和安卓,提供源码下载和maven库。 我们采用maven库,其地址如下: Official Releases Nightly Snapshots maven dependency配置:

<dependency> 
		<groupId>org.eclipse.paho</groupId> 
		<artifactId>org.eclipse.paho.client.mqttv3</artifactId> 
		<version>1.0.1</version> 
</dependency>

java实现代码:

		String topic        = "MQTT Examples"; 
        String content      = "Message from MqttPublishSample";
        int qos             = 2;
        String broker       = "tcp://127.0.0.1:61613";
        String clientId     = "JavaSample";
        MemoryPersistence persistence = new MemoryPersistence();
        try {
            MqttClient sampleClient = new MqttClient(broker, clientId, persistence);
            MqttConnectOptions connOpts = new MqttConnectOptions();
            connOpts.setCleanSession(true);
            System.out.println("Connecting to broker: "+broker);
            sampleClient.connect(connOpts);
            System.out.println("Connected");
            System.out.println("Publishing message: "+content);
            MqttMessage message = new MqttMessage(content.getBytes());
            message.setQos(qos);
            sampleClient.publish(topic, message);
            System.out.println("Message published");
            sampleClient.disconnect();
            System.out.println("Disconnected");
            System.exit(0);
        } catch(MqttException me) {
            System.out.println("reason "+me.getReasonCode());
            System.out.println("msg "+me.getMessage());
            System.out.println("loc "+me.getLocalizedMessage());
            System.out.println("cause "+me.getCause());
            System.out.println("excep "+me);
            me.printStackTrace();
        }

小结

至此,MQTT协议已部署完毕,java端可以发布消息,而javascript端则可以订阅并接收到java端发布的信息。在搭建的过程中,可以参考以下资源:

  1. MQTT
  2. ActiveMQ5
  3. Apollo官方文档
  4. eclipse paho

转载于:https://my.oschina.net/easypm/blog/644862

相关文章:

  • JavaIO详解--尽可能将BIO、NIO、AIO讲得通俗易懂
  • 用jedis获取redis连接(集群和非集群状态下)
  • Mysql的索引调优详解:如何去创建索引以及避免索引失效
  • (经验分享)作为一名普通本科计算机专业学生,我大学四年到底走了多少弯路
  • 经验分享:JAVA实习生刚进公司主要做些什么?以及进入职场后我的心理变化
  • oracle迁移mysql数据库注意(转)
  • 实习工作经历:代码在本地明明可以跑通,怎么放到服务器上就不行了呢?
  • 搭建一个包含多种Get请求和Post请求的工具类
  • 一致性hash原理与实现
  • 作为一个程序员需要了解多少网络方面的基础?网络基础总结(不断更新)
  • 四千字从源码分析ConcurrentHashMap的底层原理(JDK1.8)
  • redis学习笔记6--集合类型
  • 都2020年了,你还不知道count(1)和count(*)谁效率更高吗?
  • Linux下PF_PACKET的使用,RARP的server和client程序
  • 面试官:不会真有人不知道什么是线程池吧?
  • IE9 : DOM Exception: INVALID_CHARACTER_ERR (5)
  • SegmentFault for Android 3.0 发布
  • 【从零开始安装kubernetes-1.7.3】2.flannel、docker以及Harbor的配置以及作用
  • conda常用的命令
  • Linux后台研发超实用命令总结
  • Python 使用 Tornado 框架实现 WebHook 自动部署 Git 项目
  • webpack4 一点通
  • win10下安装mysql5.7
  • 编写高质量JavaScript代码之并发
  • 读懂package.json -- 依赖管理
  • 深入体验bash on windows,在windows上搭建原生的linux开发环境,酷!
  • 我看到的前端
  • Java总结 - String - 这篇请使劲喷我
  • python最赚钱的4个方向,你最心动的是哪个?
  • 测评:对于写作的人来说,Markdown是你最好的朋友 ...
  • ​力扣解法汇总1802. 有界数组中指定下标处的最大值
  • #LLM入门|Prompt#2.3_对查询任务进行分类|意图分析_Classification
  • $jQuery 重写Alert样式方法
  • (1)(1.19) TeraRanger One/EVO测距仪
  • (20050108)又读《平凡的世界》
  • (BFS)hdoj2377-Bus Pass
  • (附源码)spring boot北京冬奥会志愿者报名系统 毕业设计 150947
  • (附源码)springboot“微印象”在线打印预约系统 毕业设计 061642
  • (四) Graphivz 颜色选择
  • .net core 控制台应用程序读取配置文件app.config
  • .net core控制台应用程序初识
  • .net和php怎么连接,php和apache之间如何连接
  • .Net面试题4
  • .net知识和学习方法系列(二十一)CLR-枚举
  • @Repository 注解
  • [ Linux ] Linux信号概述 信号的产生
  • [ 数据结构 - C++] AVL树原理及实现
  • []FET-430SIM508 研究日志 11.3.31
  • [Android]Tool-Systrace
  • [ASP]青辰网络考试管理系统NES X3.5
  • [autojs]逍遥模拟器和vscode对接
  • [bzoj1006]: [HNOI2008]神奇的国度(最大势算法)
  • [C语言][C++][时间复杂度详解分析]二分查找——杨氏矩阵查找数字详解!!!
  • [Flutter]设置应用包名、名称、版本号、最低支持版本、Icon、启动页以及环境判断、平台判断和打包
  • [Head First设计模式]策略模式