Android 連接匿名WiFi的示例代碼

這篇文章主要介紹了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

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持神馬文庫。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章