Android 连接匿名WiFi的示例代码
前言
因为开发需要在应用内部实现wifi连接,结合网上的资料,实现连接wifi的还是比较简单,但是对于连接匿名wifi,却鲜有提及,所以在此分享下。
基本使用
首先介绍下wifi开发相关的一些基础概念和工具类等,如果对wifi已经有过接触的同学可以直接跳过看下一节。
1.权限
android中要使用系统功能一般都要申请权限,这里wifi需要的权限有
<uses-permission android:name="android.permission.change_network_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.access_wifi_state"/> <uses-permission android:name="android.permission.access_fine_location"/> // 需要系统权限 [定位权限]
因为可以利用wifi进行定位,所以这里需要申请定位权限,在6.0以上设备,定位权限需要主动申请。
2.api
类名 | 功能 |
---|---|
wifimanager | wifi统一管理类,进行各种wifi操作 |
wifiinfo | 描述当前连接的wifi热点信息 |
wificonfiguration | wifi网络配置信息 |
scanresult | 描述扫描出的wifi热点的信息 |
wifimanager类是framework层暴露的api,用来管理wifi。
val mwifimanager = mcontext.getsystemservice(context.wifi_service) as wifimanager
通过他可以得到:1.已经配置的网络列表。2.当前连接的wifi。3.扫描到的wifi。4.以及一些常量表示广播的意图等
scanresult类用于存放wifi扫描结果信息,主要有以下内容:
属性 | 描述 |
---|---|
ssid | 描述wifi热点的名称,就是大家搜索到的直接名称,如chinanet |
bssid | 姑且理解成热点的mac地址,但实际有所不同 |
networkid | 数字型的id |
level | 描述wifi信号强弱的值,值是负数,绝对值越小,信号越强 |
capabilities | 如加密方式,如wep |
3.监听设备wifi状态的改变
wifi状态的改变是会导致广播事件的发生。
val filter = intentfilter() filter.addaction(wifimanager.wifi_state_changed_action) //监听wifi状态改变 filter.addaction(wifimanager.scan_results_available_action) //监听扫描到wifi列表改变
private val mreceiver = object : broadcastreceiver() { override fun onreceive(context: context, intent: intent) { val action = intent.action if (textutils.isempty(action)) return when (action) { wifimanager.wifi_state_changed_action -> { } wifimanager.scan_results_available_action -> { } } } }
wifimanager之中有当前状态的enum类型,可以看下表:
名称 | 描述 |
---|---|
wifi_state_disabling | wifi正在关闭 |
wifi_state_disabled | wifi关闭 |
wifi_state_enabling | wifi正在开启 |
wifi_state_enabled | wifi开启 |
wifi_state_unknown | wifi未知 |
连接普通wifi
连接wifi我大致分为以下几步:
- 获取想要连接wifi热点的ssid、加密方式信息,和用户输入的密码
- 根据上述信息来创建wificonfigruation对象
- 调用wifimanager的方法,传入wificonfigruation,完成wifi连接
public static void connectnewwifi(context mcontext, wificonfiguration wificonfiguration) { wifimanager wifimanager = (wifimanager) mcontext.getapplicationcontext().getsystemservice(context.wifi_service); int networkid = wifimanager.addnetwork(wificonfiguration); wifimanager.enablenetwork(networkid, true); }
所以重点是怎样创建wificonfigruation,不同的加密方式的wifi,创建过程也不太一样:
public wificonfiguration createwificonfig(string ssid, @wificiphertype int wificiphertype, string password) { wificonfiguration wificonfiguration = new wificonfiguration(); wificonfiguration.ssid = converttoquotedstring(ssid); switch (wificiphertype) { case wificiphertype.security_none: wificonfiguration.allowedkeymanagement.set(wificonfiguration.keymgmt.none); break; case wificiphertype.security_wep: wificonfiguration.allowedkeymanagement.set(keymgmt.none); wificonfiguration.allowedauthalgorithms.set(authalgorithm.open); wificonfiguration.allowedauthalgorithms.set(authalgorithm.shared); if (!textutils.isempty(password)) { int length = password.length(); // wep-40, wep-104, and 256-bit wep (wep-232?) if ((length == 10 || length == 26 || length == 58) && password.matches("[0-9a-fa-f]*")) { wificonfiguration.wepkeys[0] = password; } else { wificonfiguration.wepkeys[0] = '"' + password + '"'; } } break; case wificiphertype.security_psk: wificonfiguration.allowedkeymanagement.set(keymgmt.wpa_psk); if (!textutils.isempty(password)) { if (password.matches("[0-9a-fa-f]{64}")) { wificonfiguration.presharedkey = password; } else { wificonfiguration.presharedkey = '"' + password + '"'; } } break; case wificiphertype.security_eap: wificonfiguration.allowedkeymanagement.set(keymgmt.wpa_eap); wificonfiguration.allowedkeymanagement.set(keymgmt.ieee8021x); wificonfiguration.enterpriseconfig = new wifienterpriseconfig(); int eapmethod = 0; int phase2method = 0; wificonfiguration.enterpriseconfig.seteapmethod(eapmethod); wificonfiguration.enterpriseconfig.setphase2method(phase2method); if (!textutils.isempty(password)) { wificonfiguration.enterpriseconfig.setpassword(password); } break; default: break; } return wificonfiguration; }
至此wifi就连接完成了,然后可以在广播中获取连接结果。相应的wifi配置信息会被保存在/data/misc/wifi/wpa_supplicant.conf中:
network={ ssid="test" psk="88888888" key_mgmt=wpa-psk disabled=1 id_str="%7b%22creatoruid%22%3a%221000%22%2c%22configkey%22%3a%22%5c%22test%5c%22wpa_psk%22%7d" }
连接匿名wifi
匿名wifi相较于普通wifi,不同之处在于不会广播其ssid,所以就不能被直接扫描到,需要我们输入wifi的ssid来主动进行扫描,先来看下匿名wifi的配置信息:
network={ ssid="test2" scan_ssid=1 bssid=56:28:f8:fa:f8:a0 psk="44444444" key_mgmt=wpa-psk disabled=1 id_str="%7b%22creatoruid%22%3a%221000%22%2c%22configkey%22%3a%22%5c%22test2%5c%22wpa_psk%22%7d" }
可以看到,多了一个scan_ssid属性,查看wificonfiguration,确实有一个属性可以设置
/** * this is a network that does not broadcast its ssid, so an * ssid-specific probe request must be used for scans. */ public boolean hiddenssid;
所以在创建wificonfiguration的时候多设置下这个属性就行了:
configuration.hiddenssid = true
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
上一篇: 详解C# 中Session的用法
下一篇: 详解git的分支与合并的两种方法