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

news/2024/11/14 20:14:17/

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

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

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

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

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

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

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

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

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

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

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

目前的硬件设备想要链接网络使用的2.4G网络

首先注意一些注意事项

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

  1. 需要获取WIFI信息

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

在 iOS 13 当中,苹果增加了无线网络和蓝牙位置隐私保护,API 方面有所变化,并新增了控制选项,有助于在用户使用无线网络和蓝牙时,防止应用未经你同意而获取你的位置信息。
#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);};}//配网成功需要[self.peripheral disconnectDevice];[self.centerManager stopScanDevice];[self.centerManager disConnectAllDevice];[self.centerManager removeAllScanFoundDevice];
连上蓝牙配网
- (void)deviceConnect:(EasyPeripheral *)peripheral error:(NSError *)errorself.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) {}];}
queueMainStartEasyService *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.ppmy.cn/news/30125.html

相关文章

python日志模块,loggin模块

python日志模块&#xff0c;loggin模块loggin模块日志的格式处理器种类日志格式的参数使用loggin模块 logging库采用模块化方法&#xff0c;并提供了几类组件&#xff1a;记录器&#xff0c;处理程序&#xff0c;过滤器和格式化程序。 记录器&#xff08;Logger&#xff09;&a…

[C++]vector模拟实现

目录 前言&#xff1a; 1. vector结构 2. 默认成员函数 2.1 构造函数 无参构造&#xff1a; 有参构造&#xff1a; 有参构造重载&#xff1a; 2.2 赋值运算符重载、拷贝构造&#xff08;难点&#xff09; 2.3 析构函数&#xff1a; 3. 扩容 3.1 reserve 3.2 resize…

【C++】类和对象补充知识点

&#x1f3d6;️作者&#xff1a;malloc不出对象 ⛺专栏&#xff1a;C的学习之路 &#x1f466;个人简介&#xff1a;一名双非本科院校大二在读的科班编程菜鸟&#xff0c;努力编程只为赶上各位大佬的步伐&#x1f648;&#x1f648; 目录前言一、再谈构造函数1.1 构造函数体赋…

解析MySQL 8.0 OCP(1Z0-908)考试中一道大部分同学都会做错的题目

一个用户有下面的权限: mysql>SHOW GRANTS FOR jsmith;---------------------------------------------------------------------- | Grants for jsmith% | ----------------------------------------------------------…

【Linux】P4 Linux 权限 chmod chown

Linux 权限认知权限信息chmod 修改权限chown 修改用户与用户组认知权限信息 序号1&#xff1a;文件、文件夹权限控制信息&#xff1b; 权限控制信息一共有十位 第 1 位&#xff1a; - 表示文件&#xff0c;d 表示文件夹&#xff0c;l 表示软链接 第 2~4 位&#xff1a; 所属用…

MATLAB——数据及其运算

MATLAB数值数据数值数据类型的分类1&#xff0e;整型整型数据是不带小数的数&#xff0c;有带符号整数和无符号整数之分。表中列出了各种整型数据的取值范围和对应的转换函数。2&#xff0e;浮点型浮点型数据有单精度(single&#xff09;和双精度&#xff08;(double)之分&…

你不会工作1年了连枚举都还不知道吧?

&#x1f497;推荐阅读文章&#x1f497; &#x1f338;JavaSE系列&#x1f338;&#x1f449;1️⃣《JavaSE系列教程》&#x1f33a;MySQL系列&#x1f33a;&#x1f449;2️⃣《MySQL系列教程》&#x1f340;JavaWeb系列&#x1f340;&#x1f449;3️⃣《JavaWeb系列教程》…

2023年Wireshark数据包分析——wireshark0051.pcap

Wireshark数据包分析 任务环境说明: 服务器场景:FTPServer220223服务器场景操作系统:未知(关闭连接)FTP用户名:wireshark0051密码:wireshark0051从靶机服务器的FTP上下载wireshark0051.pcap数据包文件,找出黑客获取到的可成功登录目标服务器FTP的账号密码,并将黑客获…