当前位置: 首页 > news >正文

移动开发之Wifi列表获取功能

一、场景

业务需要通过App给设备配置无线网络连接,所以需要App获取附近的WiFi列表,并进行网络连接验证。

二、安卓端实现

1、阅读谷歌官网文档,关于Wifi 接口使用

https://developer.android.com/guide/topics/connectivity/wifi-scan?hl=zh-cn

文档的使用流程说的相当明了清晰,注册--扫描--获取。

但是其提到了关于Android 10 以上版本的特别说明, 而且看到代码中:

标明接口过期,但是实际调试使用,发现在10以上版本中也是能正常接收到广播获取扫描结果的。只要申请号对应的权限:

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />

权限当然也需要动态申请:

ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION},WIFI_REQUEST_FOR_PERMISSION);

Android13 权限额外需求:

https://developer.android.com/guide/topics/connectivity/wifi-permissions

NEARBY_WIFI_DEVICES

这个在实际调试过程中发现,加了和没加都能够获取到wifi列表数据。

另外需要注意的是,定位权限是一回事,手机系统有没有打开定义又是另外一回事,所以在使用此功能前要先判断定位开关是否打开:

// 通过GPS卫星定位,定位级别可以精确到街(通过24颗卫星定位,在室外和空旷的地方定位准确、速度快)
boolean gps = locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
// 通过WLAN或移动网络(3G/2G)确定的位置(也称作AGPS,辅助GPS定位。主要用于在室内或遮盖物(建筑群或茂密的深林等)密集的地方定位)
boolean network = locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER);
if (gps || network) {return true;
}

再者就是WLAN 的开关有没有打开,安卓10以下的可以直接通过代码设置,10以上的需要跳转到设置界面,引导用户打开:

int wifiState = wifiManager.getWifiState();
if (WifiManager.WIFI_STATE_ENABLED != wifiState){if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {ToastUtil.makeText(mContext,"请打开WiFi开关");startActivity(new Intent(Settings.ACTION_WIFI_SETTINGS));//startActivity(new Intent(Settings.Panel.ACTION_WIFI));finish();}else {wifiManager.setWifiEnabled(true);}
}

2、网络连接验证调试

 为了确认用户输入的密码是正确的,所以想对网络连接进行验证测试,此时就发现安卓10 上下版本的接口差异了:

API29 以下

https://developer.android.com/reference/android/net/wifi/WifiManager#addNetwork(android.net.wifi.WifiConfiguration)

添加
wifiManager.addNetwork
使能
wifiManager.enableNetwork
移除,此处如果是设置里面原有保存的,则无法移除,需要引导
wifiManager.removeNetwork

AndroidQ 以后:

https://developer.android.com/guide/topics/connectivity/wifi-suggest?hl=zh-cn

/*** Android API 29 之后的wifi连接验证* @param ssid  账号* @param pwd   密码, 目前都是用 WAP2 方式*/
@RequiresApi(api = Build.VERSION_CODES.Q)
private void connectWifiAfterQ(String ssid, String pwd){WifiNetworkSpecifier.Builder builder = new WifiNetworkSpecifier.Builder();builder.setSsid(ssid);builder.setWpa2Passphrase(pwd);WifiNetworkSpecifier wifiNetworkSpecifier = builder.build();NetworkRequest.Builder networkRequestBuilder1 = new NetworkRequest.Builder();networkRequestBuilder1.addTransportType(NetworkCapabilities.TRANSPORT_WIFI);//networkRequestBuilder1.addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET);if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {networkRequestBuilder1.setNetworkSpecifier(wifiNetworkSpecifier);}NetworkRequest networkRequest = networkRequestBuilder1.build();ConnectivityManager cm = (ConnectivityManager)getApplicationContext().getSystemService(Context.CONNECTIVITY_SERVICE);// 自己独有的callback中响应事件ConnectivityManager.NetworkCallback networkCallback = newConnectivityManager.NetworkCallback() {@Overridepublic void onAvailable(Network network) {super.onAvailable(network);Log.d(TAG, "onAvailable:" + network);//让本App能够使用到此网络,此时系统其他应用是无法联网的,不知道是不是bugif (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {cm.bindProcessToNetwork(network);}}@Overridepublic void onLost(Network network) {Log.d(TAG, "The application no longer has a default network. The last default network was " + network);}@Overridepublic void onCapabilitiesChanged(Network network, NetworkCapabilities networkCapabilities) {Log.d(TAG, "The default network changed capabilities: " + networkCapabilities);}@Overridepublic void onLinkPropertiesChanged(Network network, LinkProperties linkProperties) {Log.d(TAG, "The default network changed link properties: " + linkProperties);}@Overridepublic void onUnavailable() {super.onUnavailable();Log.d(TAG, "onUnavailable:");EventBus.getDefault().post("onUnavailable");}};cm.requestNetwork(networkRequest, networkCallback);
}

这种安卓10以上版本的则需要使用suggestion方式去请求网络,但是目前调试发现一个问题就是通过App成功连接到网络之后,手机系统的其他应用则不能通过WiFi联网,而官网给出的移除网络API使用不生效:

https://developer.android.com/reference/android/net/wifi/WifiManager#removeNetworkSuggestions(java.util.List%3Candroid.net.wifi.WifiNetworkSuggestion%3E)

查找了一通资料,最后貌似好像看到说是谷歌系统的bug,至今可能还没有修复:

https://issuetracker.google.com/issues/140398818/resources

在国内的华米OV几大机型上都测试了,具有系统其他应用不能通过WIFI上网的问题,由此影响用户体验,所以最终不进行网络连接测试,改为上报wifi名称和密码,由设备自己去验证网络连接,然后在APP中展示联网效果,由此来看,其他的IoT设备,例如百度音响,是否也是这么实现,并没有通过App来改变手机本身系统的WiFi连接。

但是安卓10以下版本是无此问题的。

参考实现demo:

https://github.com/zly394/WifiListDemo

https://github.com/lilongweidev/Android13Wifi

三、苹果端实现

​​​​​​https://developer.apple.com/documentation/networkextension/wi-fi_configuration/

刚开始看文档,以为会很简单,调用几个接口即可实现。

 后来才知道这个所谓的热点助手才能实现获取列表功能,而且这个接口的使用权限要单独申请:
https://developer.apple.com/contact/request/hotspot-helper/

不出意外,这个申请果然被拒了,苹果认为只有运营商或者网络设备制造商才有需要此功能。

Thank you for your interest in the NEHotspotHelper API. This API is not designed for the use you’ve identified, so this request cannot be approved.

The NEHotspotHelper API is meant to be used by hotspot network implementers to connect their users to the internet via a large aggregated network of Wi-Fi Hotspots that they manage.

NEHotspotHelper was designed to facilitate internet hotspot network connections and is not appropriate for apps trying to do IoT accessory integration, Wi-Fi location, or other low-level Wi-Fi tasks like signal strength. Specifically, NEHotspotHelper does not let your app initiate a local Wi-Fi scan, or access iOS’ internal list of nearby SSIDs.  

Many perceived uses of NEHotspotHelper, such as the configuration of an IoT accessory, can be accomplished with NEHotspotConfiguration, which does not require an Apple-approved entitlement.

For information about enabling your app to configure an IoT accessory, please see the following article: Configuring a Wi-Fi Accessory to Join the User’s Network

For a complete explanation of Wi-Fi management APIs on iOS, please see Technote TN3111: iOS Wi-Fi API Overview.

For further technical assistance please visit the Apple Developer Forums.

Thank You,

Apple Developer Relations

于是只能退而求其次,通过用户手动输入WiFi名称和密码,App端来进行网络连接校验,这其中也遇到了权限问题:

-(NSString *)getCurrentWifi{
    NSString * wifiName = @"";
    CFArrayRef wifiInterfaces = CNCopySupportedInterfaces();
    if (!wifiInterfaces) {
        wifiName = @"";
    }
    NSArray *interfaces = (__bridge NSArray *)wifiInterfaces;
    for (NSString *interfaceName in interfaces) {
        CFBridgingRetain(interfaceName);
        CFDictionaryRef dictRef = CNCopyCurrentNetworkInfo((__bridge CFStringRef)(interfaceName));
        if (dictRef) {
            NSDictionary *networkInfo = (__bridge NSDictionary *)dictRef;
            wifiName = [networkInfo objectForKey:(__bridge NSString *)kCNNetworkInfoKeySSID];
            NSLog(@"接口查询当前连接的wifi名称为: %@", wifiName);
            CFRelease(dictRef);
        }
    }
    CFRelease(wifiInterfaces);
    NSLog(@"最终确认当前连接的wifi名称为: %@", wifiName);
    return wifiName;
}
 

就是如上这段代码刚开始始终是获取不到当前连接的WiFi名称的,而网络上查询的资料大部分都是用此方法。后面无意中在发现一个告警日志:


sent invalid result code [1] for Wi-Fi information request

通过网络搜索,原来需要开启capacity Access wifi information .

而不只是引入这两个框架即可:

//连接wifi的框架
#import <NetworkExtension/NetworkExtension.h>
//获取当前wifi的框架
#import <SystemConfiguration/CaptiveNetwork.h>

并且这个方法单独调用也是获取不到当前Wifi名称的,而是需要NEHotspotConfigurationManager 这个请求回调里面执行才可。(不知道是不是也只能查询到自己发起请求连接的WiFi,还是怎么回事)

 NEHotspotConfiguration * configuration = [[NEHotspotConfiguration alloc] initWithSSID:wifiName passphrase:passwd isWEP:NO];
    [[NEHotspotConfigurationManager sharedManager] applyConfiguration:configuration completionHandler:^(NSError * _Nullable error) {
        
        if(error != nil){
            NSLog(@"apply config error=%@ code=%ld", error.description, (long)error.code);
        }else{
            NSLog(@"apply config success ? wifiName=%@, passwd=%@", wifiName, passwd);
        }
        //有时无法加入WIFI,没有返回error
        if ([[self getCurrentWifi] isEqualToString:wifiName]) {
            if (error) {
                //无法加入网络,需移除
                [[NEHotspotConfigurationManager sharedManager] removeConfigurationForSSID:wifiName];
                if(error.code == NEHotspotConfigurationErrorAlreadyAssociated){
                    //上报账号密码,这个是正确的连接, 应该不会出现这个逻辑了
                    NSLog(@"WiFi之前已经连接成功,不应该走这个逻辑了,前面每次都先移除了");
                    //[self ConfigSuccessBack];
                }else{
                    [self alertConfigInfoError:NSLocalizedString(@"wifiConfigAccountPasswordError", nil)];
                }
                
            }else{
                //连接wifi成功
                NSLog(@"连接WiFi成功");
                [self ConfigSuccessBack];
            }
        }else{
            //无法加入网络,需移除
            [[NEHotspotConfigurationManager sharedManager] removeConfigurationForSSID:wifiName];
            [self alertConfigInfoError:NSLocalizedString(@"wifiConfigAccountPasswordError", nil)];
        }
    }];
 

总之一番操作之后,确认需要添加的Capacity 有如下三个:

+       com.apple.developer.networking.HotspotConfiguration

+       com.apple.developer.networking.networkextension

+       com.apple.developer.networking.wifi-info

最终苹果端这边还是可以成功验证网络是否正常连接,而且不影响系统重其他应用上网。

一个看似很简单的功能,前前后后,各种零散问题分析查阅调试验证,安卓的各种机型和版本,苹果的各种权限和邮件回复,让这个功能还是弄了蛮久,关键是这种功能涉及敏感安全,为了防止滥用,系统平台随着自己的不断完善发展,对其限制要求会越来越严格。

相关文章:

移动开发之Wifi列表获取功能

一、场景 业务需要通过App给设备配置无线网络连接&#xff0c;所以需要App获取附近的WiFi列表&#xff0c;并进行网络连接验证。 二、安卓端实现 1、阅读谷歌官网文档&#xff0c;关于Wifi 接口使用 https://developer.android.com/guide/topics/connectivity/wifi-scan?hl…...

MyBatisPlus - 实体类 的 常用注解

TableName(“表名”) 假设 表名是 book&#xff0c;实体类类名是 Book MyBatisPlus会进行自动映射 但如果 表名是 tab_book&#xff0c;实体类类名是 Book 那么MyBatisPlus就无法进行自动映射&#xff0c;需要我们使用 TableName注解 去指定实体类对应的表 如下 TableNa…...

vue3+ts+elementui-plus二次封装树形表格实现不同层级展开收起的功能

一、TableTreeLevel组件 <template><div classmain><div class"btns"><el-button type"primary" click"expandLevel(1)">展开一级</el-button><el-button type"primary" click"expandLevel(2…...

Qt之切换语言的方法(传统数组法与Qt语言家)

http://t.csdn.cn/BVigB 传统数组法&#xff1a; 定义一个字符串二维数组&#xff0c; QString weekStr[2][7] {"星期一","星期二","星期三","星期四","星期五","星期六","星期日",\ "Monday&…...

qt root start faild

深入解析chown -r root:root命令_笔记大全_设计学院 ffmpeg第五弹&#xff1a;QtSDLffmpeg视频播放演示_txp玩Linux的博客-CSDN博客...

数据结构—串

4.1串 4.1.1串的定义 串&#xff08;String&#xff09;——零个或多个任意字符组成的有限序列 S"a1 a2...an"串的定义——几个术语 子串&#xff1a;串中任意个连续字符组成的子序列称为该串的子串 例如&#xff0c;“abcde”的子串有&#xff1a; “ ”、“a”、…...

hive 全量表、增量表、快照表、切片表和拉链表

全量表&#xff1a;记录每天的所有的最新状态的数据&#xff0c;增量表&#xff1a;记录每天的新增数据&#xff0c;增量数据是上次导出之后的新数据。快照表&#xff1a;按日分区&#xff0c;记录截止数据日期的全量数据切片表&#xff1a;切片表根据基础表&#xff0c;往往只…...

数据结构07:查找[C++][B树Btree]

图源&#xff1a;文心一言 考研对于B树的要求重点在推理手算的部分&#xff0c;只参考王道论坛咸鱼老师的视频就可以了&#xff1b;若时间非常充裕的小伙伴&#xff0c;也可以往下滑了解一下代码~&#x1f95d;&#x1f95d; 备注&#xff1a; 这次的代码是从这里复制的&…...

在CSDN学Golang云原生(Kubernetes集群管理)

一&#xff0c;Node的隔离与恢复 在 Kubernetes 集群中&#xff0c;Node 的隔离与恢复通常可以通过以下方式实现&#xff1a; 使用 Taints 和 Tolerations 实现隔离 Taints 和 Tolerations 是 Kubernetes 中用于节点调度的机制。通过给节点添加 taints&#xff08;污点&…...

WPF实战学习笔记18-优化设计TodoView

文章目录 优化设计TodoView修复新增项目无法编辑问题增加了对完成状态的区分增加了选项卡删除功能更新删除请求URI添加删除命令并初始化UI添加删除按钮更改控制器 增加查询结果为空的图片增加转换器修改UI添加资源、命名空间 添加相关元素 增加了根据状态查询的功能Mytodo.Serv…...

Python版day59

503. 下一个更大元素 II 给定一个循环数组 nums &#xff08; nums[nums.length - 1] 的下一个元素是 nums[0] &#xff09;&#xff0c;返回 nums 中每个元素的 下一个更大元素 。 数字 x 的 下一个更大的元素 是按数组遍历顺序&#xff0c;这个数字之后的第一个比它更大的数&…...

[SQL挖掘机] - 算术运算符

在 sql 中&#xff0c;算术运算符主要用于执行数值计算操作&#xff0c;并且在查询语句中具有重要的地位。下面是算术运算符在 sql 中的一些作用和地位&#xff1a; 进行数值计算&#xff1a;算术运算符可以对数值类型的数据进行加减乘除等数值计算操作。例如&#xff0c;可以…...

机器学习基础 数据集、特征工程、特征预处理、特征选择 7.27

机器学习基础 1. 数据集 2. 特征工程 3. 学习分类 4. 模型 5. 损失函数 6. 优化 7. 过拟合 8. 欠拟合数据集 又称资料集、数据集合或者资料集合&#xff0c;是一种由数据所组成的集合特征工程 1. 特征需求 2. 特征设计 3. 特征处理特征预处理、特征选择、特征降维 4. 特征验…...

Sass 常用的功能!

Sass 常用功能 Sass 功能有很多&#xff0c;这边只列举一些比较常用的。 嵌套规则 (Nested Rules) Sass 允许将一套 CSS 样式嵌套进另一套样式中&#xff0c;内层的样式将它外层的选择器作为父选择器。 编译前 .box {.box1 {background-color: red;}.box2 {background-col…...

chmod命令详细使用说明

chmod命令详细使用说明 chmod是Unix和类Unix系统上用于更改文件或目录权限的命令。它是"change mode"的缩写。在Linux和其他类Unix操作系统中&#xff0c;文件和目录具有权限位&#xff0c;用来控制哪些用户可以访问、读取、写入或执行它们。chmod命令允许用户修改这…...

ICC2如何计算Gate Count?

我正在「拾陆楼」和朋友们讨论有趣的话题&#xff0c;你⼀起来吧&#xff1f;知识星球入口 我们认为gate count等于standard cell(非physical only)总面积 / 最小驱动二输入与非门面积。 ICC2没有专门的命令去报告gate count&#xff0c;只能自己计算&#xff0c;使用report_d…...

Qtday3作业

作业 头文件 #ifndef WIDGET_H #define WIDGET_H#include <QPushButton> #include <QTextToSpeech> #include <QWidget> #include <QDebug> #include <QTimer> //定时器类 #include <QTime> //时间类 #include <QTimerEvent>…...

全球程序员需要知道的50+网址,有多少你第一次听说?

作为程序员&#xff0c;需要知道的50网址&#xff0c;有多少你第一次听说 GitHub (github.com): 最大的代码托管平台&#xff0c;开源项目和代码分享的社区。程序员可以在这里找到各种有趣的项目&#xff0c;参与开源贡献或托管自己的代码。 Stack Overflow (stackoverflow.co…...

Matlab中实现对一幅图上的局部区域进行放大

大家好&#xff0c;我是带我去滑雪&#xff01; 局部放大图可以展示图像中的细节信息&#xff0c;使图像更加直观和精美&#xff0c;此次使用magnify工具实现对绘制的figure选择区域绘制&#xff0c;图像效果如下&#xff1a; 1、基本图像绘制 这里选择绘制一个散点图&#xff…...

mysql-速成补充

目录 1.演示事务 ​编辑 1.1 read-uncommitted 1.2 read-committed 1.3 repeatable read 1.4 幻读 1.5 serializable 1.6 savepoint 2 变量 2.1 语法 2.2 举例 3 存储过程和函数 3.1 特点和语法 3.2 举例 4.函数 4.1 语法 4.2 举例 5 流程控制 5.1 分…...

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站&#xff0c;会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后&#xff0c;网站没有变化的情况。 不熟悉siteground主机的新手&#xff0c;遇到这个问题&#xff0c;就很抓狂&#xff0c;明明是哪都没操作错误&#x…...

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…...

CTF show Web 红包题第六弹

提示 1.不是SQL注入 2.需要找关键源码 思路 进入页面发现是一个登录框&#xff0c;很难让人不联想到SQL注入&#xff0c;但提示都说了不是SQL注入&#xff0c;所以就不往这方面想了 ​ 先查看一下网页源码&#xff0c;发现一段JavaScript代码&#xff0c;有一个关键类ctfs…...

golang循环变量捕获问题​​

在 Go 语言中&#xff0c;当在循环中启动协程&#xff08;goroutine&#xff09;时&#xff0c;如果在协程闭包中直接引用循环变量&#xff0c;可能会遇到一个常见的陷阱 - ​​循环变量捕获问题​​。让我详细解释一下&#xff1a; 问题背景 看这个代码片段&#xff1a; fo…...

java 实现excel文件转pdf | 无水印 | 无限制

文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...

从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)

设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile&#xff0c;新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...

Python ROS2【机器人中间件框架】 简介

销量过万TEEIS德国护膝夏天用薄款 优惠券冠生园 百花蜂蜜428g 挤压瓶纯蜂蜜巨奇严选 鞋子除臭剂360ml 多芬身体磨砂膏280g健70%-75%酒精消毒棉片湿巾1418cm 80片/袋3袋大包清洁食品用消毒 优惠券AIMORNY52朵红玫瑰永生香皂花同城配送非鲜花七夕情人节生日礼物送女友 热卖妙洁棉…...

推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材)

推荐 github 项目:GeminiImageApp(图片生成方向&#xff0c;可以做一定的素材) 这个项目能干嘛? 使用 gemini 2.0 的 api 和 google 其他的 api 来做衍生处理 简化和优化了文生图和图生图的行为(我的最主要) 并且有一些目标检测和切割(我用不到) 视频和 imagefx 因为没 a…...

PHP 8.5 即将发布:管道操作符、强力调试

前不久&#xff0c;PHP宣布了即将在 2025 年 11 月 20 日 正式发布的 PHP 8.5&#xff01;作为 PHP 语言的又一次重要迭代&#xff0c;PHP 8.5 承诺带来一系列旨在提升代码可读性、健壮性以及开发者效率的改进。而更令人兴奋的是&#xff0c;借助强大的本地开发环境 ServBay&am…...

WebRTC从入门到实践 - 零基础教程

WebRTC从入门到实践 - 零基础教程 目录 WebRTC简介 基础概念 工作原理 开发环境搭建 基础实践 三个实战案例 常见问题解答 1. WebRTC简介 1.1 什么是WebRTC&#xff1f; WebRTC&#xff08;Web Real-Time Communication&#xff09;是一个支持网页浏览器进行实时语音…...