MQTT-Client-FrameWork使用整理

摘要:
自己manager){MQTSSLSecurityPolicyTransport*transport=[[MQTSSLSecurity PolicyTransportalloc]init];transport.host=@“xxxx.com”;transport.port=8883;transport.tls=YES;NSString*ca=[[NSBundlebandleForClass:[MQTTSessionclass]]pathForResource:@“ca”of Type:@“der”];//TODO:双向身份验证需要客户端证书//NSString*client=[[NSBondlebandleforClass:[MQTTSessionclass]]pathForResource:@“certificate”类型:@“p12”];//运输certificates=[MMQTSSLSecurityPolicyTransportclientCertsFromP12:clientpassphrase:@“password”];MQTTSSL安全策略*安全策略=[MMQTSSLSecurityPolicyPolicyWithPinningMode:MMQTSSLPinningModeCertificate];安全政策。allowInvalidCertificates=是;安全政策。validatesDomainName=NO;安全政策。validateCertificateChain=否;安全政策。pinnedCertificates=@[[NSDatadataWithContentsOfFile:ca]];运输securityPolicy=安全策略;自己manager=[[MQTTSessionManageralloc]init];自我管理者。委托=自己;////不要使用证书//[self.managerconnectTo:@“xxxxx.com”//port:1883//tls:NO//keepalive:60//心跳间隔不应大于120s//clean:true//auth:true//user:@“username”//pass:@“password”//will:false//willTopic:nil//willMsg:nil//willQos:0/willRetainFlag:false//withClientId:@“xxxxxxxxxxxxx”]////使用证书[self.anagerconnectto:@“xxx.com”端口:8883tls:YEKeepalive:60clean:trueauth:YEUser:@“用户名”pass:@“密码”will:falsewillTopic:nilwilMsg:nilwillQos:0willRetainFlag:FALSwithClientId:@“xxxxxxxx”securityPolicy:securityPolicy证书:nil];}否则{[self.managerconnectToLast];}/*MQTTLient:观察MQTTSessionManager的状态以显示连接状态*/[self.manageraddObserver:self-forKeyPath:@“state”选项:NSKeyValueObservingOptionInitial|NSKeyValueObservingOptionNewcontext:nil];参数说明:主机IP端口tls是否使用ssl/tls来验证保活心跳是一个时间间隔。客户端定期向代理服务器发送PINGRequest消息。AuthMQTT允许您发送用户名和密码,以验证客户机的身份并对其进行授权。topicillMsg将消息nsdate类型的willTopic将。

作者: wbl

MQTT

MQTT(Message Queuing Telemetry Transport,消息队列遥测传输)是IBM开发的一个即时通讯协议,有可能成为物联网的重要组成部分。该协议支持所有平台,几乎可以把所有联网物品和外部连接起来,被用来当做传感器和制动器(比如通过Twitter让房屋联网)的通信协议

MQTT特点

MQTT协议是为大量计算能力有限,且工作在低带宽、不可靠的网络的远程传感器和控制设备通讯而设计的协议,它具有以下主要的几项特性:

1. 使用发布/订阅消息模式,提供一对多的消息发布,解除应用程序耦合;

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

3. 使用 TCP/IP 提供网络连接;

4. 有三种消息发布服务质量:

  • “至多一次”,消息发布完全依赖底层 TCP/IP 网络。会发生消息丢失或重复。这一级别可用于如下情况,环境传感器数据,丢失一次读记录无所谓,因为不久后还会有第二次发送。
  • “至少一次”,确保消息到达,但消息重复可能会发生。
  • “只有一次”,确保消息到达一次。这一级别可用于如下情况,在计费系统中,消息重复或丢失会导致不正确的结果。

5. 小型传输,开销很小(固定长度的头部是 2 字节),协议交换最小化,以降低网络流量;

6. 使用 Last Will 和 Testament 特性通知有关各方客户端异常中断的机制;

好了废话不多说,直接上干货

双向认证方法:让后台生成 ca.crt 和 client.p12文件(client.p12文件由client.crt和client.key合成)
单项认证方法:让后台生成 ca.crt

1.使用命令行把crt转化为der格式

openssl x509 -in ca.crt -out ca.der -outform der

2.建立连接

if (!self.manager) {
        MQTTSSLSecurityPolicyTransport *transport = [[MQTTSSLSecurityPolicyTransport alloc]init];
        transport.host = @"xxxx.com";
        transport.port = 8883;
        transport.tls = YES;
        NSString*  ca = [[NSBundle bundleForClass:[MQTTSession class]] pathForResource:@"ca" ofType:@"der"];
        ////TODO:双向认证需加入client证书
    //        NSString*  client = [[NSBundle bundleForClass:[MQTTSession class]] pathForResource:@"certificate" ofType:@"p12"];
    //        transport.certificates = [MQTTSSLSecurityPolicyTransport clientCertsFromP12:client passphrase:@"password"];
        MQTTSSLSecurityPolicy *securityPolicy = [MQTTSSLSecurityPolicy policyWithPinningMode:MQTTSSLPinningModeCertificate];
        securityPolicy.allowInvalidCertificates = YES;
        securityPolicy.validatesDomainName = NO;
        securityPolicy.validatesCertificateChain = NO;
        securityPolicy.pinnedCertificates = @[[NSData dataWithContentsOfFile:ca]];
        transport.securityPolicy = securityPolicy;
        
        self.manager = [[MQTTSessionManager alloc] init];
        self.manager.delegate = self;
        ////不使用证书
    //        [self.manager connectTo:@"xxxxx.com"
    //                           port:1883
    //                            tls:NO
    //                      keepalive:60  //心跳间隔不得大于120s
    //                          clean:true
    //                           auth:true
    //                           user:@"username"
    //                           pass:@"password"
    //                           will:false
    //                      willTopic:nil
    //                        willMsg:nil
    //                        willQos:0
    //                 willRetainFlag:FALSE
    //                   withClientId:@"xxxxxxxxxxxxx"];
        ////使用证书(这里采用单项认证,双向认证只需把certificates:参数设置为transport.certificates即可)
        [self.manager connectTo:@"xxxxx.com"
                           port:8883
                            tls:YES
                      keepalive:60
                          clean:true
                           auth:YES
                           user:@"username"
                           pass:@"password"
                           will:false
                      willTopic:nil
                        willMsg:nil
                        willQos:0
                 willRetainFlag:FALSE
                   withClientId:@"xxxxxxxxx"
                 securityPolicy:securityPolicy
                   certificates:nil];
    } else {
        [self.manager connectToLast];
    }
    /*
      MQTTCLient: observe the MQTTSessionManager's state to display the connection status
     */
    
    [self.manager addObserver:self
                   forKeyPath:@"state"
                      options:NSKeyValueObservingOptionInitial | NSKeyValueObservingOptionNew
                      context:nil];

参数解释:

  • host 主机
  • ip 端口
  • tls 是否使用ssl/tls认证
  • keepalive 心跳是一个时间间隔,客户端有规律地向代理服务器发送PING Request消息。服务器用PING Response消息进行响应,这种机制可以使双方据此判定对方是否还活着以及能否到达。
  • clean clean标志位向代理服务器表明客户端是否要建立持续的会话。持续的会话(CleanSession设置为false)意思是代理将存储所有服务质量 (QoS) 为1或2的客户端的订阅信息以及所有错过的消息。如果clean设置为true,则代理将不会存储客户端的任何信息,并将清除之前持续会话的所有信息。
  • auth MQTT允许发送用户名和密码以便验证客户端的身份和进行授权。
  • user 如果auth为yes 需要填用户名
  • pass 如果auth为yes 需要填密码
  • will 是否设遗嘱 是的话需要填写willTopic和willMsg 遗嘱消息是MQTT遗嘱特征的组成部分。当某个客户端恶意断开连接时,遗嘱消息将通知其他客户端。一个连接着的客户端将在CONNECT消息中以MQTT消息和主题的形式提供其遗嘱。如果客户端恶意断开了连接,代理将发送此消息来代表客户端。
  • willTopic 遗嘱的topic
  • willMsg 遗嘱消息 nsdate类型。
  • willQos 服务质量(QoS) 等级。
  • willRetainFlag 会话表示标志表示,代理和客户端之间自从前面的交互以来是否是持久会话。值为0,服务器必须在客户端断开之后继续存储/保持客户端的订阅状态。这些状态包括: 存储订阅的消息QoS1和QoS2消息 正在发送消息期间连接丢失导致发送失败的消息 以便当客户端重新连接时以上消息可以被重新传递。 值为1,服务器需要立刻清理连接状态数据。 (tips:断线重连时 如果为yes,会自动订阅回消息,如果为no,则要手动订阅topic,不然会收不到消息)。
  • clientId 客户端标识符(不能固定,每个用户需要不同的clientId,相同的会导致掉线),是每个连接到MQTT代理服务器的MQTT客户端的编号。单词identifier本身已经表明,ClientID应当是唯一的。代理服务器使用它来识别客户端以及当前的客户端状态。如果你不想让代理获得当前状态,那么在MQTT3.1.1(当前的标准)中也可以发送一个空的ClientID,这样的话,连接就不会附带任何状态。前提条件是clean为true,否则,连接会被拒绝。

3.服务器返回数据

状态信息:

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
    switch (self.manager.state) {
        case MQTTSessionManagerStateClosed:
            NSLog(@"MQTTSessionManagerStateClosed");
            break;
        case MQTTSessionManagerStateClosing:
            NSLog(@"MQTTSessionManagerStateClosing");
            break;
        case MQTTSessionManagerStateConnected:
            NSLog(@"MQTTSessionManagerStateConnected");
            //连接成功订阅
            for (int i=0; i<3; i++) {
                [self.manager subscribeToTopic:[NSString stringWithFormat:@"%d%d%d%d%d%d%d%d",i,i,i,i,i,i,i,i] atLevel:1];
            }
            break;
        case MQTTSessionManagerStateConnecting:
            NSLog(@"MQTTSessionManagerStateConnecting");
            break;
        case MQTTSessionManagerStateError:
            NSLog(@"MQTTSessionManagerStateError");
            break;
        case MQTTSessionManagerStateStarting:
            NSLog(@"MQTTSessionManagerStateStarting");
        default:
            NSLog(@"default");
            break;
    }
}

数据信息:

- (void)handleMessage:(NSData *)data onTopic:(NSString *)topic retained:(BOOL)retained {
    NSLog(@"------------->>%@",topic);
    
    NSString *dataString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    NSLog(@"%@",dataString);
    
}

4.连接(断开后的再次连接):

- (IBAction)connect:(id)sender {
    [self.manager connectToLast];
}

5.订阅

- (IBAction)sub:(id)sender {
    [self.manager subscribeToTopic:@"55555555555555555" atLevel:1];
}

6.取消订阅

- (IBAction)unsub:(id)sender {
    [self.manager unsubscribeTopic:@"55555555555555555"];
}

7.断开连接

- (IBAction)disconnect:(id)sender {
[self.manager disconnect];

说明:MQTTSessionManager是没有subscribeToTopic:atLevel:和unsubscribeTopic:方法的,只有subscriptions。为了更灵活使用,所以需要自己添加

1. 在.h中添加:

//订阅
- (void)subscribeToTopic:(NSString *)topic
             atLevel:(MQTTQosLevel)qosLevel;
//取消订阅
- (void)unsubscribeTopic:(NSString *)topic;

2. 在.m中添加

//订阅
- (void)subscribeToTopic:(NSString *)topic
                 atLevel:(MQTTQosLevel)qosLevel
{
    if (self.state == MQTTSessionManagerStateConnected) {
        [self.session subscribeToTopic:topic atLevel:1 subscribeHandler:^(NSError *error, NSArray<NSNumber *> *gQoss) {
            if (!error) {
                
            }else{
                
            }
        }];
    }
}

//取消订阅
- (void)unsubscribeTopic:(NSString *)topic
{
    if (self.state == MQTTSessionManagerStateConnected) {
        [self.session unsubscribeTopic:topic unsubscribeHandler:^(NSError *error) {
            if (!error) {
                
            }else{
                
            }
        }];
    }
}

项目地址

https://github.com/wangbaolei/Mqtt-Client.git

免责声明:文章转载自《MQTT-Client-FrameWork使用整理》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇wpf datagrid笔记 part 1使用阿里云公网ip建立bind,监听客户端连接失败下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

iOS 测试 | iOS 自动化性能采集

​今天小编跟大家分享一篇来自学院内部学员的技术分享,本文主要介绍了作者在进行 iOS 自动化性能采集的一些经验,希望对大家在进行 iOS 自动化测试时有一些启发。 不要为小事遮住视线,我们还有更大的世界 前言 对于iOS总体生态是比较封闭的,相比Android没有像adb这种可以查看内存、cpu的命令.在日常做性能测试,需要借助xcode中instrum...

RocketMQ的消息发送及消费

RocketMQ消息支持的模式: 消息支持的模式分为三种:NormalProducer(普通同步),消息异步发送,OneWay。 消息同步发送: 普通消息的发送和接收在前面已经演示过了,在前面的案例中是基于同步消息发送模式。也就是说消息发送出去后,producer会等到broker回应后才能继续发送下一个消息. 消息异步发送: 异步发送是指发送方发出数据后...

Akka源码分析-Remote-收消息

上一遍博客中,我们分析了网络链接建立的过程,一旦建立就可以正常的收发消息了。发送消息的细节不再分析,因为对于本地的actor来说这个过程相对简单,它只是创立链接然后给指定的netty网路服务发送消息就好了。接收消息就比较麻烦了,因为这对于actor来说是透明的,netty收到消息后如何把消息分发给指定的actor呢?这个分发的过程值得研究研究。 之前分析过...

数据绑定流程分析

1.    数据绑定流程原理★ ①   Spring MVC 主框架将 ServletRequest  对象及目标方法的入参实例传递给 WebDataBinderFactory 实例,以创建 DataBinder 实例对象 ②   DataBinder 调用装配在 Spring MVC 上下文中的 ConversionService 组件进行数据类型转换、...

iOS中NSString转换成HEX(十六进制)-NSData转换成int

http://www.2cto.com/kf/201402/281501.html 1 2 3 4 5 6 NSString *str = @"0xff055008"; //先以16为参数告诉strtoul字符串参数表示16进制数字,然后使用0x%X转为数字类型 unsigned longred = strtoul([str UTF8Stri...

kafka 基础知识梳理-kafka是一种高吞吐量的分布式发布订阅消息系统

一、kafka 简介 今社会各种应用系统诸如商业、社交、搜索、浏览等像信息工厂一样不断的生产出各种信息,在大数据时代,我们面临如下几个挑战: 如何收集这些巨大的信息 如何分析它 如何及时做到如上两点 以上几个挑战形成了一个业务需求模型,即生产者生产(produce)各种信息,消费者消费(consume)(处理分析)这些信息,而在生产者与消费者之间,需要...