赞
踩
物联网(Internet of Things,IoT)最近曝光率越来越高。虽然HTTP是网页的事实标准,不过机器之间(Machine-to-Machine,M2M)的大规模沟通需要不同的模式:之前的请求/回答(Request/Response)模式不再合适,取而代之的是发布/订阅(Publish/Subscribe)模式。这就是轻量级、可扩展的MQTT(Message Queuing Telemetry Transport)可以施展拳脚的舞台。
MQTT是基于二进制消息的发布/订阅编程模式的消息协议,最早由IBM提出的,如今已经成为OASIS规范。由于规范很简单,非常适合需要低功耗和网络带宽有限的IoT场景,比如:
由于物联网的环境是非常特别的,所以MQTT遵循以下设计原则:
运用MQTT协议,设备可以很方便地连接到物联网云服务,管理设备并处理数据,最后应用到各种业务场景,如下图所示:
与请求/回答这种同步模式不同,发布/定义模式解耦了发布消息的客户(发布者)与订阅消息的客户(订阅者)之间的关系,这意味着发布者和订阅者之间并不需要直接建立联系。打个比方,你打电话给朋友,一直要等到朋友接电话了才能够开始交流,是一个典型的同步请求/回答的场景;而给一个好友邮件列表发电子邮件就不一样,你发好电子邮件该干嘛干嘛,好友们到有空了去查看邮件就是了,是一个典型的异步发布/订阅的场景。
熟悉编程的同学一定非常熟悉这种设计模式了,因为它带来了这些好处:MQTT是通过主题对消息进行分类的,本质上就是一个UTF-8的字符串,不过可以通过反斜杠表示多个层级关系。主题并不需要创建,直接使用就是了。
主题还可以通过通配符进行过滤。其中,+可以过滤一个层级,而*只能出现在主题最后表示过滤任意级别的层级。举个例子:注意,MQTT允许使用通配符订阅主题,但是并不允许使用通配符广播。
为了满足不同的场景,MQTT支持三种不同级别的服务质量(Quality of Service,QoS)为不同场景提供消息可靠性:
服务质量是个老话题了。级别2所提供的不重不丢很多情况下是最理想的,不过往返多次的确认一定对并发和延迟带来影响。级别1提供的至少一次语义在日志处理这种场景下是完全OK的,所以像Kafka这类的系统利用这一特点减少确认从而大大提高了并发。级别0适合鸡肋数据场景,食之无味弃之可惜,就这么着吧。
MQTT拥有14种不同的消息类型:
- //
- // ViewController.m
- // mqttkit_demo
- //
- // Created by 张闯 on 17/5/15.
- // Copyright © 2017年 张闯. All rights reserved.
- //
-
- #import "ViewController.h"
- #import <MQTTKit.h>
-
- static NSString *const kClientId = @"189XXXX3425";
- static NSString *const kTopicS = @"/MQTTKit/example/send";
- static NSString *const kTopicR = @"/MQTTKit/example/receive";
-
- @interface ViewController ()
-
- @property (weak, nonatomic ) IBOutlet UITextField *messageText;
- @property (weak, nonatomic ) IBOutlet UILabel *sendMessageLabel;
- @property (weak, nonatomic ) IBOutlet UILabel *receivedMessageLabel;
- @property (nonatomic, strong) MQTTClient *client;
-
- @end
-
- @implementation ViewController
-
- - (void)viewDidLoad {
- [super viewDidLoad];
- // Do any additional setup after loading the view, typically from a nib.
-
- self.client = [[MQTTClient alloc] initWithClientId:kClientId];
-
- __weak typeof(self) WeakSelf = self;
- dispatch_async(dispatch_get_global_queue(0, 0), ^{
-
- //链接到服务器
- [WeakSelf.client connectToHost:@"mq.xxxxxxxx.top"
- completionHandler:^(MQTTConnectionReturnCode code) {
-
- NSLog(@"connectionReturnCode: ____%ld", code);
- __strong typeof(self) StrongSelf = self;
- if (code == ConnectionAccepted) {
- //订阅主题
- [StrongSelf.client subscribe:kTopicS
- withCompletionHandler:nil];
- [StrongSelf.client subscribe:kTopicR
- withCompletionHandler:nil];
-
- }
- }];
-
- //监听接收数据
- [WeakSelf.client setMessageHandler:^(MQTTMessage *message) {
- NSString *text = message.payloadString;
- __strong typeof(self) StrongSelf = self;
-
- if ([message.topic isEqualToString:kTopicR]) {
- dispatch_async(dispatch_get_main_queue(), ^{
- //接收到消息,更新界面时需要切换回主线程
- StrongSelf.receivedMessageLabel.text = text;
- NSLog(@"received message %@", text);
- });
- }
- }];
- });
-
- //断开重连
- [self.client disconnectWithCompletionHandler:^(NSUInteger code) {
- NSLog(@"MQTT client is disconnected");
- [self.client reconnect];
- }];
- }
-
-
- - (IBAction)send:(id)sender {
-
- //发送消息
- [self.client publishString:self.messageText.text
- toTopic:kTopicS
- withQos:AtMostOnce
- retain:NO
- completionHandler:^(int mid) {
- NSLog(@"message has been delivered");
- self.sendMessageLabel.text = self.messageText.text;
- }];
-
- }
-
- - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
- [self.view endEditing:YES];
- }
-
-
- - (void)didReceiveMemoryWarning {
- [super didReceiveMemoryWarning];
- // Dispose of any resources that can be recreated.
- }
-
-
- @end
![](https://csdnimg.cn/release/blogv2/dist/pc/img/newCodeMoreWhite.png)
- target 'mqttkit_demo' do
-
- pod 'MQTTKit'
-
- end
其他参考资料:
http://blog.shiqichan.com/introducing-mqtt/
http://www.jianshu.com/p/19765f425259
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。