iOS开发之MQTT探究

简介:

1、 什么是MQTT?

MQTT(MessageQueueing Telemetry Transport Protocol)的全称是消息队列遥感传输协议的缩写,是由IBM公司推出的一种基于轻量级代理的发布/订阅模式的消息传输协议,运行在TCP协议栈之上,为其提供有序、可靠、双向连接的网络连接保证。由于其开放、简单和易于实现所以能够应用在资源受限的环境中,对于M2M和物联网应用程序来说是一个相当不错的选择。

2、 为什么要用MQTT?

MQTT协议是针对如下情况设计的:

M2M(Machine to Machine) communication,机器端到端通信,比如传感器之间的数据通讯 因为是Machine to Machine,需要考虑: Machine,或者叫设备,比如温度传感器,硬件能力很弱,协议要考虑尽量小的资源消耗,比如计算能力和存储等 M2M可能是无线连接,网络不稳定,带宽也比较小

MQTT的特点:

1.发布/订阅消息模式,提供一对多的消息发布,解除应用程序耦合。这一点很类似于1. 这里是列表文本XMPP,但是MQTT的信息冗余远小于XMPP.

2.对负载内容屏蔽的消息传输。

3.使用TCP/IP提供网络连接。主流的MQTT是基于TCP连接进行数据推送的,但是同样有基于UDP的版本,叫做MQTT-SN。这两种版本由于基于不同的连接方式,优缺点自然也就各有不同了。

4.三种消息传输方式QoS:

  • 0代表“至多一次”,消息发布完全依赖底层 TCP/IP 网络。会发生消息丢失或重复。这一级别可用于如下情况,环境传感器数据,丢失一次读记录无所谓,因为不久后还会有第二次发送。
  • 1代表“至少一次”,确保消息到达,但消息重复可能会发生。
  • 2代表“只有一次”,确保消息到达一次。这一级别可用于如下情况,在计费系统中,消息重复或丢失会导致不正确的结果。 备注:由于服务端采用Mosca实现,Mosca目前只支持到QoS 1
  • 如果发送的是临时的消息,例如给某topic所有在线的设备发送一条消息,丢失的话也无所谓,0就可以了(客户端登录的时候要指明支持的QoS级别,同时发送消息的时候也要指明这条消息支持的QoS级别),如果需要客户端保证能接收消息,需要指定QoS为1,如果同时需要加入客户端不在线也要能接收到消息,那么客户端登录的时候要指定session的有效性,接收离线消息需要指定服务端要保留客户端的session状态。
  • mqtt基于订阅者模型架构,客户端如果互相通信,必须在同一订阅主题下,即都订阅了同一个topic,客户端之间是没办法直接通讯的。订阅模型显而易见的好处是群发消息的话只需要发布到topic,所有订阅了这个topic的客户端就可以接收到消息了。
  • 发送消息必须发送到某个topic,重点说明的是不管客户端是否订阅了该topic都可以向topic发送了消息,还有如果客户端订阅了该主题,那么自己发送的消息也会接收到。

5.小型传输,开销很小(固定长度的头部是2字节),协议交换最小化,以降低网络流量。这就是为什么在介绍里说它非常适合“在物联网领域,传感器与服务器的通信,信息的收集”,要知道嵌入式设备的运算能力和带宽都相对薄弱,使用这种协议来传递消息再适合不过了。

6.使用Last Will和Testament特性通知有关各方客户端异常中断的机制。Last Will:即遗言机制,用于通知同一主题下的其他设备发送遗言的设备已经断开了连接。Testament:遗嘱机制,功能类似于Last Will 。

3、 怎么使用MQTT

在mac上搭建MQTT服务器

 
 
  1. $ brew install mosquitto 

等待下载完成,服务会自动运行起来

 
 
  1. mosquitto has been installed with a default configuration file. 
  2. You can make changes to the configuration by editing:  
  3. /usr/local/etc/mosquitto/mosquitto.conf  
  4. To have launchd start mosquitto now and restart at login:  
  5. brew services start mosquitto  
  6. Or, if you don't want/need a background service you can just run:  
  7. mosquitto -c /usr/local/etc/mosquitto/mosquitto.conf 

iOS client注册

 
 
  1. #import "ViewController.h" 
  2.  
  3. #define kMQTTServerHost @"iot.eclipse.org" 
  4. #define kTopic @"MQTTExample/Message" 
  5.  
  6. @interface ViewController () 
  7. @property (weak, nonatomic) IBOutlet UILabel *showMessage; 
  8. @property (nonatomic, strong) MQTTClient *client; 
  9. @end 
  10.  
  11. @implementation ViewController 
  12.  
  13. - (void)viewDidLoad 
  14.     [super viewDidLoad]; 
  15.  
  16.     //1.在app登录后,后台返回 namepassword+topic 
  17.  
  18.     //2.namepassword用于连接主机 
  19.  
  20.     //3.topic 用于订阅主题 
  21.  
  22.  
  23.     UILabel *tempShowMessage = self.showMessage; 
  24.  
  25.     NSString *clientID = [UIDevice currentDevice].identifierForVendor.UUIDString; 
  26.  
  27.     self.client = [[MQTTClient alloc] initWithClientId:clientID]; 
  28.  
  29.     //连接服务器  连接后,会通过block将连接结果code返回,然后执行此段代码块 
  30.  
  31.     //这个接口是修改过后的接口,修改后抛出了namepassword 
  32.     [self.client connectToHost:kMQTTServerHost andName:@"cbt" andPassword:@"1223" completionHandler:^(MQTTConnectionReturnCode code) { 
  33.         if (code == ConnectionAccepted)//连接成功 
  34.         { 
  35.             // 订阅 
  36.             [self.client subscribe:kTopic withCompletionHandler:^(NSArray *grantedQos) { 
  37.                 // The client is effectively subscribed to the topic when this completion handler is called 
  38.                 NSLog(@"subscribed to topic %@", kTopic); 
  39.                 NSLog(@"return:%@",grantedQos); 
  40.             }]; 
  41.         } 
  42.     }]; 
  43.  
  44.  
  45.     //MQTTMessage  里面的数据接收到的是二进制,这里框架将其封装成了字符串 
  46.     [self.client setMessageHandler:^(MQTTMessage* message) 
  47.     { 
  48.         dispatch_async(dispatch_get_main_queue(), ^{ 
  49.             //接收到消息,更新界面时需要切换回主线程 
  50.             tempShowMessage.text= message.payloadString; 
  51.         }); 
  52.     }];   
  53.  
  54.  
  55.  
  56. - (void)dealloc8 
  57.     // disconnect the MQTT client 
  58.     [self.client disconnectWithCompletionHandler:^(NSUInteger code) 
  59.     { 
  60.         // The client is disconnected when this completion handler is called 
  61.         NSLog(@"MQTT is disconnected"); 
  62.     }]; 
  63. @end 

server向client推送消息

 
 
  1. #import "ViewController.h" 
  2. #import "MQTTKit.h" 
  3.  
  4.  
  5. #define kMQTTServerHost @"iot.eclipse.org" 
  6. #define kTopic @"MQTTExample/Message" 
  7. @interface ViewController () 
  8. @property (weak, nonatomic) IBOutlet UITextField *pushMessage; 
  9. @property (nonatomic, strong) MQTTClient *client; 
  10. @end 
  11.  
  12. @implementation ViewController 
  13.  
  14. - (void)viewDidLoad { 
  15.     [super viewDidLoad]; 
  16.  
  17.  
  18.     NSString *clientID = [UIDevice currentDevice].identifierForVendor.UUIDString; 
  19.  
  20.     self.client = [[MQTTClient alloc] initWithClientId:clientID]; 
  21.     [self.client connectToHost:kMQTTServerHost andName:@"cbt" andPassword:@"1223" completionHandler:^(MQTTConnectionReturnCode code) { 
  22.         if (code == ConnectionAccepted) 
  23.         { 
  24.             NSLog(@"服务器启动成功"); 
  25.         } 
  26.     }]; 
  27.  
  28.  
  29.  
  30. - (IBAction)push:(id)sender { 
  31.     NSString* payload = self.pushMessage.text; 
  32.     [self.client publishString:payload 
  33.                        toTopic:kTopic 
  34.                        withQos:AtMostOnce 
  35.                         retain:YES 
  36.              completionHandler:nil]; 
  37.     NSLog(@"推送内容:%@",payload); 





作者:Frida芥末
来源:51CTO
相关实践学习
RocketMQ一站式入门使用
从源码编译、部署broker、部署namesrv,使用java客户端首发消息等一站式入门RocketMQ。
消息队列 MNS 入门课程
1、消息队列MNS简介 本节课介绍消息队列的MNS的基础概念 2、消息队列MNS特性 本节课介绍消息队列的MNS的主要特性 3、MNS的最佳实践及场景应用 本节课介绍消息队列的MNS的最佳实践及场景应用案例 4、手把手系列:消息队列MNS实操讲 本节课介绍消息队列的MNS的实际操作演示 5、动手实验:基于MNS,0基础轻松构建 Web Client 本节课带您一起基于MNS,0基础轻松构建 Web Client
目录
相关文章
|
5天前
|
Unix 调度 Swift
苹果iOS新手开发之Swift 中获取时间戳有哪些方式?
在Swift中获取时间戳有四种常见方式:1) 使用`Date`对象获取秒级或毫秒级时间戳;2) 通过`CFAbsoluteTimeGetCurrent`获取Core Foundation的秒数,需转换为Unix时间戳;3) 使用`DispatchTime.now()`获取纳秒级精度的调度时间点;4) `ProcessInfo`提供设备启动后的秒数,不表示绝对时间。不同方法适用于不同的精度和场景需求。
17 3
|
4天前
|
Swift iOS开发 Kotlin
苹果iOS新手开发之Swift中实现类似Kotlin的作用域函数
Swift可通过扩展实现类似Kotlin作用域函数效果。如自定义`let`, `run`, `with`, `apply`, `also`,增强代码可读性和简洁性。虽无直接内置支持,但利用Swift特性可达成相似功能。
23 7
|
1天前
|
移动开发 前端开发 iOS开发
探索iOS开发的未来:SwiftUI与Combine的融合
随着苹果公司不断推进其操作系统的更新,iOS开发领域也迎来了诸多变革。在这篇文章中,我们将深入探讨SwiftUI和Combine这两个强大的框架,它们如何共同塑造着iOS应用开发的未来趋势。通过具体实例和数据支持,本文旨在揭示这些技术如何简化开发者的工作流,提升用户界面构建的效率,以及加强应用的响应性和性能表现。我们还将提出一个开放性问题,邀请读者思考并探索这些技术在未来可能带来的进一步影响。 【7月更文挑战第26天】
9 2
|
3天前
|
API 数据处理 开发工具
探索iOS开发的未来:SwiftUI和Combine的融合
【7月更文挑战第23天】随着Apple不断推动其软件开发工具的创新,SwiftUI和Combine框架的出现标志着iOS开发进入了一个新的时代。本文将深入探讨这两个框架如何简化界面设计和事件处理,以及它们如何共同为开发者提供一个更加高效、声明式的编程模型。我们将通过实际示例来展示如何利用SwiftUI构建用户界面,并使用Combine处理异步事件和状态管理。文章还将预测这些技术如何塑造iOS应用开发的未来趋势。
|
2天前
|
调度 Swift Android开发
苹果iOS新手开发之Swift中的并发任务和消息机制
Swift的消息机制类似Android的Handler,实现任务调度有三种方式: 1. **Grand Central Dispatch (GCD)**:使用`DispatchQueue`在主线程或后台线程执行任务。 2. **OperationQueue**:提供高级接口管理`Operation`对象。 3. **RunLoop**:处理事件如输入源、计时器,类似Android的`Looper`和`Handler`。 **示例**: - GCD:在不同线程执行代码块。 - OperationQueue:创建操作并执行。 - RunLoop:用Timer添加到RunLoop中。
12 2
|
7天前
|
Swift iOS开发 开发者
探索iOS开发中的SwiftUI框架
【7月更文挑战第19天】在移动应用开发的浪潮中,苹果公司的SwiftUI框架如同一股清新的海风,为iOS开发者带来了前所未有的编程体验。本文将深入探讨SwiftUI的核心特性,揭示其如何简化界面设计流程,提升开发效率,并展望SwiftUI在未来iOS开发领域的发展潜力。通过实例分析,我们将一同见证SwiftUI如何塑造更加直观、高效的编程模式。
|
3天前
|
前端开发 Android开发 iOS开发
探索安卓与iOS开发的差异性与互补性
在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统各据一方,引领着市场潮流。它们在技术架构、开发环境及用户群体等方面展现出独特的差异性,同时也存在着潜在的互补性。本文将深入剖析这两种平台的开发细节,从不同角度揭示其各自优势及相互之间的协同潜力,为开发者提供全面而深刻的视角。
10 2
|
3天前
|
IDE Android开发 Swift
探究iOS与安卓应用开发的核心差异
在数字时代,移动应用开发已成为技术革新的主战场。本文将通过对比分析iOS和安卓两大平台,深入探讨它们在开发环境、编程语言、用户界面设计、系统架构以及市场分布等方面的根本差异。我们将利用最新的行业报告和案例研究,结合统计数据,提供一个全面而深入的视角来理解这两个操作系统对开发者和技术选择的影响。
10 2
|
4天前
|
前端开发 Swift iOS开发
探索iOS开发的未来:SwiftUI和Combine的融合
在iOS开发领域,SwiftUI和Combine框架的出现标志着一个新时代的到来。本文深入探讨这两个框架如何共同推动iOS应用开发的现代化,通过具体案例分析它们的优势、挑战以及未来趋势。
|
4天前
|
数据处理 Swift iOS开发
探索iOS开发的未来之路:SwiftUI和Combine框架的融合
在本文中,我们将深入探讨iOS开发的新趋势——SwiftUI和Combine框架的结合使用。我们将从这两个框架的基本概念入手,逐步解析它们如何协同工作,以实现更加高效、响应式的用户界面构建。通过实例演示,我们将揭示这种组合如何简化代码结构,提高开发效率,并增强应用性能。最后,我们将展望这种技术栈在未来iOS开发中的潜在影响和应用前景。