這篇文章主要介紹了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
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持神馬文庫。