ios 通过搜索设备MAC地址绑定

news2024/10/7 4:33:54

最近做了一个物联网项目,涉及到了设备绑定配网这块,需要了解一下iOS BLE与设备绑定的相关知识点,第一次接触蓝牙相关的项目,所以开始熟悉蓝牙的相关信息。没有去深入研究BabyTooth库,只是感觉CoreBluetooth已经让我更好的理解整个流程

这个物联网项目的设备绑定流程是需要APP端把WIFI的信息传给硬件设备,不过硬件存在一些瑕疵

  1. 硬件设备不会告诉APP端WIFI密码是否未填/错误

  1. 硬件设备不会告诉APP端设备是通过2.4G网络还是5G网络

  1. 设备属于同一个型号,但是设备支持的功能却不同,设备很混乱,例如一种配网功能有些需要2分钟内,有些不需要

  1. 子设备配网需要主设备在线,并且配网成功有WIFI信息才能正常使用

  1. 子设备的生产并没有合理化的展开,有些子设备的MAC是错误的,有些子设备有蓝牙名称有些没有

所以只能通过跑定时器60s来拟定设备绑定错误,且通过一个大量的文案提示来告诉用户错误的可能性,让用户自行判断是设备问题导致的无法配网还是因为WIFI频段的问题还是WIFI密码错误的问题等

参考了部分APP绑定硬件设备流程,标准流程都有,但是部分在配网页面是120s且不存在综合错误统计页面,最后的配网准备流程等待过长时间且没有过多的文案提示,对用户来说并不是很友好

项目中实用到的错误综合参考页面

故障拓展的因素种类过多,需要考虑的极限异常种类也很多,比如使用WIFI配网时网络突然中断,配网成功后网络出现波动等

总之,一个萝卜一个坑,为了保证用户在购买硬件产品后,不需要使用说明书就能实现APP操作硬件

用户体验,我辈义不容辞! ! !

目前的硬件设备想要配置网络使用的2.4G网络,也可以是2.4G和5G二合一的网络

关于WIFI 2.4G和5G状态的判断
尝试过使用WIFI信号的频道强度来获取当前连接的WIFI是2.4G和5G,并不适用
简单的通过WIFI的名称,通过获取到的字符串来识别是否包含2.4G和5G,需要判断逻辑考虑不全且没什么意义
  • 如果有好的方式,请私信联系我

一些注意事项

  1. 开启蓝牙模块,需要系统和APP的蓝牙全部开启

  1. 需要获取WIFI信息

  1. 地址定位权限需要开启,为什么呢?

在 iOS 13 当中,苹果增加了无线网络和蓝牙位置隐私保护,API 方面有所变化,并新增了控制选项,有助于在用户使用无线网络和蓝牙时,防止应用未经你同意而获取你的位置信息。
  • APP权限检测流程

#import <CoreBluetooth/CoreBluetooth.h>

外围设备和中央设备在CoreBluetooth中使用CBPeripheralManager和CBCentralManager表示。

CBPeripheralManager:外围设备通常用于发布服务、生成数据、保存数据。外围设备发布并广播服务,告诉周围的中央设备它的可用服务和特征。

CBCentralManager:中央设备使用外围设备的数据.中央设备扫描到外围设备后会就会试图建立连接,一旦连接成功就可以使用这些服务和特征。

外围设备和中央设备之间交互的桥梁是服务(CBService)和特征(CBCharacteristic)
获取当前手机的WIFI信息
-(void)makeWifiIsCanUes
{

    id info = nil;
    
    NSString *str ;
    NSArray *ifs = (__bridge_transfer id)CNCopySupportedInterfaces();
    for (NSString *ifnam in ifs) {
        info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam);
        str = info[@"SSID"];
        _BBIDdStr = info[@"BSSID"];
        NSLog(@"%@----",info);
    }

    if (str.length == 0) {
        NSLog(@" 您的手机还未连接WiFi网络");
    }else{
        self.ssidStr = [BZAirKissShareTools fetchSSIDInfo][@"SSID"];
        self.wifiName.text = [BZAirKissShareTools fetchSSIDInfo][@"SSID"];
    }
}

主设备是根据有指定的SN号可以配对的,所以在扫描周围设备方法的回调中对返回的peripheral.name

wifiNamePwd WIFI密码可以为空,但是WIFI名称是一定需要的
[self.centerManager scanDeviceWithTimeInterval:NSIntegerMax services:nil options:@{ CBCentralManagerScanOptionAllowDuplicatesKey: @YES }  callBack:^(EasyPeripheral *peripheral, searchFlagType searchType) {
        NSLog(@"peripheral : %@ ",peripheral);
        NSLog(@"peripheral.name : %@ ",peripheral.name);
        NSLog(@"peripheral.identifier.UUIDString : %@ ",peripheral.identifier.UUIDString);
        NSData *data = [peripheral.advertisementData objectForKey:@"kCBAdvDataManufacturerData"];
        NSString *aStr= [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];

        const char *valueString = [[data description] cStringUsingEncoding: NSUTF8StringEncoding];
        NSString *mac = [self convertToNSStringWithNSData:data];
        aStr = [aStr stringByReplacingOccurrencesOfString:@" " withString:@""];
        NSLog(@"aStr : %@",aStr);
        NSLog(@"advertisementData : %@",peripheral.advertisementData);
        
        NSLog(@"======================================================");
        NSLog(@"MAC : %@",mac);
        NSLog(@"======================================================");
        if ([peripheral.name containsString:self.sn.uppercaseString]) {
            
            [self.centerManager stopScanDevice];
            
            NSData *data =[self.wifiNamePwd dataUsingEncoding:NSUTF8StringEncoding];
            
            WEAKSELF;
            [self.characteristic writeValueWithData:data callback:^(EasyCharacteristic *characteristic, NSData *data, NSError *error) {
                NSLog(@"成功");
                if (error) {
                    [weakSelf connectionBackWithType:1];
                }else{
                    [weakSelf connectionBackWithType:0];
                }
            }];
    }
    }];
    
    self.centerManager.stateChangeCallback = ^(EasyCenterManager *manager, CBManagerState state) {
        NSLog(@"%ld",(long)state);
    };
  
}

需要注意的是蓝牙模块的生态

1.蓝牙搜索

开始蓝牙搜索 - 蓝牙搜索中 - 蓝牙搜索成功

2.蓝牙连接

开始蓝牙连接 - 蓝牙连接中 - 蓝牙连接成功

3.WIFI连接

开始WIFI连接 - WIFI连接中 - WIFI连接成功

4.设备联网

设备联网中 - 设备联网成功

不使用蓝牙时需要取消关闭
                [self.peripheral disconnectDevice];
                [self.centerManager stopScanDevice];
                [self.centerManager disConnectAllDevice];
                [self.centerManager removeAllScanFoundDevice];
连上蓝牙配网
- (void)deviceConnect:(EasyPeripheral *)peripheral error:(NSError *)error
        self.peripheral = peripheral;
        WEAKSELF;
        [self.peripheral discoverAllDeviceServiceWithCallback:^(EasyPeripheral *peripheral, NSArray<EasyService *> *serviceArray, NSError *error) {
            for (EasyService *tempS in serviceArray) {
                [tempS discoverCharacteristicWithCallback:^(NSArray<EasyCharacteristic *> *characteristics, NSError *error) {
                    for (EasyCharacteristic *tempC in characteristics) {
                        [tempC discoverDescriptorWithCallback:^(NSArray<EasyDescriptor *> *descriptorArray, NSError *error) {
                            for (EasyDescriptor *desc in descriptorArray) {
                                [desc readValueWithCallback:^(EasyDescriptor *descriptor, NSError *error) {
                                }];
                            }
queueMainStart
                            EasyService *tempS = weakSelf.peripheral.serviceArray[0] ;
                            EasyCharacteristic *tempC = tempS.characteristicArray[0];
                            weakSelf.characteristic = tempC;

                            NSLog(@"连上蓝 获取成功服务");
queueEnd
                        }];
                    }
                }];
            }
        }];
}

注意项:

写了一个NSMutableDictionary *foundDeviceDict如果不在扫描设备、已连接设备的集合中就加入其中,并通知外部调用者

之前通过NSArray来遍历数据,会存在偶现的闪退,定位后在这报错,直接遍历字典,因为key有时候返回的为nil
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary *)advertisementData RSSI:(NSNumber *)RSSI
{
//字典直接enumerateKeysAndObjectsUsingBlock遍历数据
    [self.foundDeviceDict enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        if ([key isEqualToString:peripheral.identifier.UUIDString]) {
            EasyPeripheral *tempP = self.foundDeviceDict[key];
            tempP.deviceScanCount++ ;
            existedIndex = tempP.deviceScanCount ;
            *stop = YES;    
        }
    }];
      //发现一个设备的回调
      //去掉重复搜索到的设备

}

绑定子设备的时候,已有不少麻烦的问题

  1. iOS的MAC地址和安卓的MAC的地址是不同的,倒叙的

  1. 部分子设备MAC地址需要自己手动截取一遍

由于系统限制,Android 上获取到的 deviceId 为设备 MAC 地址,iOS 上则为设备 uuid。因此 deviceId 不能硬编码到代码中。
[mac substringFromIndex:4]
NSString *macString = [NSString stringWithFormat:@"%@%@%@%@%@%@",[mac substringWithRange:NSMakeRange(10, 2)],
                                       [mac substringWithRange:NSMakeRange(8, 2)],
                                       [mac substringWithRange:NSMakeRange(6, 2)],
                                       [mac substringWithRange:NSMakeRange(4, 2)],
                                       [mac substringWithRange:NSMakeRange(2, 2)],
                                       [mac substringWithRange:NSMakeRange(0, 2)]
                ];
mac = [mac substringWithRange:NSMakeRange(mac.length - 12, 12)];
            NSString *macString = [NSString stringWithFormat:@"%@%@%@%@%@%@",[mac substringWithRange:NSMakeRange(10, 2)],
                                   [mac substringWithRange:NSMakeRange(8, 2)],
                                   [mac substringWithRange:NSMakeRange(6, 2)],
                                   [mac substringWithRange:NSMakeRange(4, 2)],
                                   [mac substringWithRange:NSMakeRange(2, 2)],
                                   [mac substringWithRange:NSMakeRange(0, 2)]
            ];

蓝牙产品在广播包中会以某个字节标识自己的类型,扫描到设备以后代理方法中会以字典的形式提供给我们。

iOS 8及以前kCBAdvDataManufacturerData这个数据提供的是scan response (SCAN_RSP),但是iOS 9及以后会把advertising packet (ADV_IND)scan response (SCAN_RSP)两部分合并在一起提供给了我们。所以不同版本的情况下我们获取kCBAdvDataManufacturerData会出现不同。

- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary<NSString *, id> *)advertisementData RSSI:(NSNumber *)RSSI
{
        id data = advertisementData[@"kCBAdvDataManufacturerData"];
}

官方的问答https://www.jianshu.com/p/e9f647f59eb6

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/390836.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

sheng的学习笔记-Actuator健康监控

前言在微服务系统里&#xff0c;对微服务程序的运行状况的跟踪和监控是必不可少的&#xff1b;例如GPE&#xff0c;TelegrafinfluxDB都提供了微服务体系监控的方案&#xff0c; ZIPKIN&#xff0c; Skywalking都提供了微服务云体系的APM的方案&#xff1b; 这些解决方案功能全面…

Thumbnailator快速入门

简介 Thumbnailator 是一个开源的 Java 项目&#xff0c;它提供了非常简单的 API 来对图片进行缩放、旋转以及加水印的处理。 有多简单呢&#xff1f;简单到一行代码就可以完成图片处理。形式如下&#xff1a; Thumbnails.of(new File("path/to/directory").listF…

IDEA搭建vue-cli | vue-router | 排错思路、Webpack、Axios、周期、路由、异步、重定向

&#x1f497;wei_shuo的个人主页 &#x1f4ab;wei_shuo的学习社区 &#x1f310;Hello World &#xff01; Vue.js概述 Vue 是一套用于构建用户界面的渐进式JavaScript框架。 与其它大型框架不同的是&#xff0c;Vue 被设计为可以自底向上逐层应用。Vue 的核心库只关注视图层…

IDEA集成Git就是这么简单

IDEA集成Git 文章目录IDEA集成Git配置Git环境配置Git的忽略文件①为什么需要配置忽略文件&#xff1f;②配置忽略文件③引用配置文件配置IDEA初始化项目添加到暂存区方式一&#xff1a;方式二&#xff1a;移除暂存区提交到本地库分支创建分支切换分支版本穿梭配置Git环境 配置…

应届大学生学什么技术好?哪些技术适合年轻人?

到了毕业季&#xff0c;应届大学生面临的就是就业问题&#xff0c;很多专业的大学生难以找到对口的工作&#xff0c;或是不得已随便就业&#xff0c;或者是学个技术高薪就业&#xff0c;那么&#xff0c;问题来了&#xff0c;应届大学生学什么技术好&#xff1f;哪些技术适合年…

XC7K160T-1FBG484I、XC7A100T-2CSG324I FPGA可编程门阵列 PDF规格书

1、XC7K160T-1FBG484I说明&#xff1a;Kintex-7 FPGA有-3、-2、-1、-1L和-2L速度等级&#xff0c;其中-3具有最高的性能。-2L器件被筛选为较低的最大静态功率&#xff0c;并且可以在较低的核心电压下运行&#xff0c;以获得比-2器件更低的动态功率。-2L工业(I)温度器件仅在VCCI…

C#教程03-- 数据类型

文章目录 C#数据类型值类型(Value types)引用类型对象(Object)类型动态(Dynamic)类型字符串(String)类型C#数据类型 在 C# 中,变量分为以下几种类型: 值类型(Value types) 引用类型(Reference types) 指针类型(Pointer types) 值类型(Value types) 值类型…

Bean的装配方式(xml和注解)

方式一&#xff1a;基于xml 掌握Bean基于XML的装配&#xff0c;能够使用XML装配方式对Bean进行装配 在基于XML的装配就是读取XML配置文件中的信息完成依赖注入&#xff0c;Spring容器提供了两种基于XML的装配方式&#xff0c;属性setter方法注入和构造方法注入。下面分另对这…

RocketMQ-02

1. 案例介绍 1.1 业务分析 模拟电商网站购物场景中的【下单】和【支付】业务 ###1&#xff09;下单 用户请求订单系统下单订单系统通过RPC调用订单服务下单订单服务调用优惠券服务&#xff0c;扣减优惠券订单服务调用调用库存服务&#xff0c;校验并扣减库存订单服务调用用户…

Vue2.0开发之——购物车案例-Footer组件封装(50)

一 概述 导入Footer子组件定义fullState计算属性把全选状态传递给Footer子组件实现全选功能 二 导入Footer子组件 2.1 App.vue中导入Footer组件 import Footer from "/components/Footer/Footer.vue";2.2 App.vue中注册Footer子组件 components: {Header,Goods,F…

云原生之使用Docker部署Gitblit服务器

云原生之使用Docker部署Gitblit服务器一、Gitblit介绍二、检查本地docker环境1.检查docker版本2.检查docker状态三、下载Gitblit镜像四、部署Gitblit应用1.创建部署目录2.创建Gitblit容器3.检查Gitblit容器状态4.检查Gitblit容器运行日志五、访问Gitblit首页1.访问Gitblit首页2…

LPDDR4x 的 学习总结(3) - SDRAM基本功能

上一节,我们重点介绍了array的存储结构。 本节介绍array周边的电路,对DDR的基本读写操作的相关功能模块的理解。 即通过哪些模块可以实现对ddr的基本读写。最简化的方式是把存储操作理解为行列选择&#xff0c;拆分为横竖两个纬度&#xff0c;最终实现对arrary进行读写。横向…

Kerberos 域委派攻击之约束性委派

CSDN自动博客文章迁移由于非约束性委派的不安全性&#xff0c;微软在 Windows Server 2003 中引入了约束委派。区别在于不会直接把 TGT 给服务&#xff0c;所发送的认证信息中包含了允许访问的服务&#xff0c;即不允许服务代表用户去访问其他服务。同时为了在 Kerberos 协议层…

1.数据结构前言

数据结构很重要&#xff01; 数据结构很重要&#xff01;&#xff01;&#xff01; 数据结构很重要&#xff01;&#xff01;&#xff01;&#xff01; 前言.思考 1.什么是数据结构 &#xff1f;&#xff08;What&#xff09; 2.为什么要学数据结构 &#xff1f;&#xff1f; (…

删除链表元素相关的练习

目录 一、移除链表元素 二、删除排序链表中的重复元素 三、删除排序链表中的重复元素 || 四、删除链表的倒数第 N 个结点 一、移除链表元素 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头…

python字典和集合——笔记

一、介绍 1、泛映射类型 collections.abc模块中有Mapping和MutableMapping这两个抽象基类&#xff0c;它们的作用是为dict和其他类似的类型定义形式接口&#xff08;在Python 2.6到Python 3.2的版本中&#xff0c;这些类还不属于collections.abc模块&#xff0c;而是隶属于coll…

【震撼发布】《致敬未来的攻城狮计划》| 文末赠书3本

《致敬未来的攻城狮计划》—— 文末有福利 摘要&#xff1a; 一个崭新的计划&#xff0c;寻找那群有志于向嵌入式发展的未来工程师&#xff01; 文章目录1 活动计划初衷2 活动计划形式3 活动计划收获4 活动计划要求5 活动计划时间6 活动计划致谢7 活动计划特别说明8 温馨提示9 …

Kerberos 域委派攻击之非约束性委派

CSDN文章自动迁移自博客在Windows 2000 Server 首次发布 Active Directory 时&#xff0c;Microsoft 必须提供一种简单的机制来支持用户通过 Kerberos 向 Web Server 进行身份验证并需要代表该用户更新后端数据库服务器上的记录的方案。这通常称为“Kerberos 双跳问题”&#x…

零入门kubernetes网络实战-20->golang编程syscall操作tun设备介绍

《零入门kubernetes网络实战》视频专栏地址 https://www.ixigua.com/7193641905282875942 本篇文章视频地址(稍后上传) 本篇文章主要是使用golang自带的syscall包来创建tun类型的虚拟网络设备。 注意&#xff1a; 目前只能使用syscall包来创建tun类型的虚拟设备。 tun虚拟网…

【RockerMQ】003-Windows 安装 RocketMQ

【RockerMQ】003-Windows 安装 RocketMQ 一、准备工作 1、环境要求 64位JDK 1.8;Maven 3.2.x;64位操作系统系统&#xff0c;本文档在Windows上安装 2、下载解压 下载地址 https://archive.apache.org/dist/rocketmq/5.1.0/ 下载目标 解压 到不含中文路径的目录下 环境变…