1. 程式人生 > >Android開發——自動連線指定SSID的wifi熱點(不加密/加密)

Android開發——自動連線指定SSID的wifi熱點(不加密/加密)

最近在做一個專案,其中涉及到一塊“自動連線已存在的wifi熱點”的功能,在網上查閱了大量資料,五花八門,但其中一些說的很簡單,即不能實現傻瓜式的拿來就用,有些說的很詳細,但其中不乏些許錯誤造成功能無法實現,經過浣熊多方努力,終於成功將功能實現,遂將一點點小成就拿出來與大家分享。

在這篇文章中,作者定義了一個wifi工具類,其中存在著操作wifi的各種方法,其中有一些錯誤我以改正,正確的程式碼如下(建立一個名為WifiAdmin.java的檔案,以下程式碼中沒有包宣告和import,請自行新增):

public class WifiAdmin { 
    // 定義WifiManager物件  
    private WifiManager mWifiManager; 
    // 定義WifiInfo物件  
    private WifiInfo mWifiInfo; 
    // 掃描出的網路連線列表  
    private List<ScanResult> mWifiList; 
    // 網路連線列表  
    private List<WifiConfiguration> mWifiConfiguration; 
    // 定義一個WifiLock  
    WifiLock mWifiLock; 

 
    // 構造器  
    public WifiAdmin(Context context) { 
        // 取得WifiManager物件  
        mWifiManager = (WifiManager) context 
                .getSystemService(Context.WIFI_SERVICE); 
        // 取得WifiInfo物件  
        mWifiInfo = mWifiManager.getConnectionInfo(); 
    } 
 
    // 開啟WIFI  
    public void openWifi() { 
        if (!mWifiManager.isWifiEnabled()) { 
            mWifiManager.setWifiEnabled(true); 
        } 
    } 
 
    // 關閉WIFI  
    public void closeWifi() { 
        if (mWifiManager.isWifiEnabled()) { 
            mWifiManager.setWifiEnabled(false); 
        } 
    } 
 
    // 檢查當前WIFI狀態  
    public int checkState() { 
        return mWifiManager.getWifiState(); 
    } 
 
    // 鎖定WifiLock  
    public void acquireWifiLock() { 
        mWifiLock.acquire(); 
    } 
 
    // 解鎖WifiLock  
    public void releaseWifiLock() { 
        // 判斷時候鎖定  
        if (mWifiLock.isHeld()) { 
            mWifiLock.acquire(); 
        } 
    } 
 
    // 建立一個WifiLock  
    public void creatWifiLock() { 
        mWifiLock = mWifiManager.createWifiLock("Test"); 
    } 
 
    // 得到配置好的網路  
    public List<WifiConfiguration> getConfiguration() { 
        return mWifiConfiguration; 
    } 
 
    // 指定配置好的網路進行連線  
    public void connectConfiguration(int index) { 
        // 索引大於配置好的網路索引返回  
        if (index > mWifiConfiguration.size()) { 
            return; 
        } 
        // 連線配置好的指定ID的網路  
        mWifiManager.enableNetwork(mWifiConfiguration.get(index).networkId, 
                true); 
    } 
 
    public void startScan() { 
        mWifiManager.startScan(); 
        // 得到掃描結果  
        mWifiList = mWifiManager.getScanResults(); 
        // 得到配置好的網路連線  
        mWifiConfiguration = mWifiManager.getConfiguredNetworks(); 
    } 
 
    // 得到網路列表  
    public List<ScanResult> getWifiList() { 
        return mWifiList; 
    } 
 
    // 檢視掃描結果  
    public StringBuilder lookUpScan() { 
        StringBuilder stringBuilder = new StringBuilder(); 
        for (int i = 0; i < mWifiList.size(); i++) { 
            stringBuilder 
                    .append("Index_" + new Integer(i + 1).toString() + ":"); 
            // 將ScanResult資訊轉換成一個字串包  
            // 其中把包括:BSSID、SSID、capabilities、frequency、level  
            stringBuilder.append((mWifiList.get(i)).toString()); 
            stringBuilder.append("/n"); 
        } 
        return stringBuilder; 
    }
 
    // 得到MAC地址  
    public String getMacAddress() { 
        return (mWifiInfo == null) ? "NULL" : mWifiInfo.getMacAddress(); 
    } 
 
    // 得到接入點的BSSID  
    public String getBSSID() { 
        return (mWifiInfo == null) ? "NULL" : mWifiInfo.getBSSID(); 
    } 
 
    // 得到IP地址  
    public int getIPAddress() { 
        return (mWifiInfo == null) ? 0 : mWifiInfo.getIpAddress(); 
    } 
 
    // 得到連線的ID  
    public int getNetworkId() { 
        return (mWifiInfo == null) ? 0 : mWifiInfo.getNetworkId(); 
    } 
 
    // 得到WifiInfo的所有資訊包  
    public String getWifiInfo() { 
        return (mWifiInfo == null) ? "NULL" : mWifiInfo.toString(); 
    } 
 
    // 新增一個網路並連線  
    public void addNetwork(WifiConfiguration wcg) { 
	 int wcgID = mWifiManager.addNetwork(wcg); 
     boolean b =  mWifiManager.enableNetwork(wcgID, true); 
     System.out.println("a--" + wcgID);
     System.out.println("b--" + b);
    } 
 
    // 斷開指定ID的網路  
    public void disconnectWifi(int netId) { 
        mWifiManager.disableNetwork(netId); 
        mWifiManager.disconnect(); 
    } 
 
//然後是一個實際應用方法,只驗證過沒有密碼的情況:
 
    public WifiConfiguration CreateWifiInfo(String SSID, String Password, int Type) 
    { 
          WifiConfiguration config = new WifiConfiguration();   
           config.allowedAuthAlgorithms.clear(); 
           config.allowedGroupCiphers.clear(); 
           config.allowedKeyManagement.clear(); 
           config.allowedPairwiseCiphers.clear(); 
           config.allowedProtocols.clear(); 
          config.SSID = "\"" + SSID + "\"";   
          
          WifiConfiguration tempConfig = this.IsExsits(SSID);           
          if(tempConfig != null) {  
        	  mWifiManager.removeNetwork(tempConfig.networkId);  
          }
          
          if(Type == 1) //WIFICIPHER_NOPASS
          { 
               config.wepKeys[0] = ""; 
               config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE); 
               config.wepTxKeyIndex = 0; 
          } 
          if(Type == 2) //WIFICIPHER_WEP
          { 
              config.hiddenSSID = true;
              config.wepKeys[0]= "\""+Password+"\""; 
              config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.SHARED); 
              config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.CCMP); 
              config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.TKIP); 
              config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.WEP40); 
              config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.WEP104); 
              config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE); 
              config.wepTxKeyIndex = 0; 
          } 
          if(Type == 3) //WIFICIPHER_WPA
          { 
          config.preSharedKey = "\""+Password+"\""; 
          config.hiddenSSID = true;   
          config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.OPEN);   
          config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.TKIP);                         
          config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.WPA_PSK);                         
          config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.TKIP);                    
          //config.allowedProtocols.set(WifiConfiguration.Protocol.WPA);  
          config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.CCMP);
          config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.CCMP);
          config.status = WifiConfiguration.Status.ENABLED;   
          }
           return config; 
    } 
    
    private WifiConfiguration IsExsits(String SSID)  
    {  
        List<WifiConfiguration> existingConfigs = mWifiManager.getConfiguredNetworks();  
           for (WifiConfiguration existingConfig : existingConfigs)   
           {  
             if (existingConfig.SSID.equals("\""+SSID+"\""))  
             {  
                 return existingConfig;  
             }  
           }  
        return null;   
    }
  
}
//分為三種情況:1沒有密碼2用wep加密3用wpa加密

改動主要集中在CreateWifiInfo這個方法中,並且添加了一個私有方法:

(1)將與方法的第三個引數有關的變數都改成int型,或者使用原作者的列舉型(存在bug需要改正),但列舉會在後續的開發中遇到些困難;

(2)在if(type == 3)中註釋掉“config.allowedProtocols.set(WifiConfiguration.Protocol.WPA);”,並新增“

config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.CCMP); config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.CCMP

);“這兩句,否則當wifi熱點需要輸入密碼時,無法加入網路。

(3)在程式碼末尾添加了方法IsExsits,原因在於如果按照網上介紹的方法成功加入指定的wifi後,都會在終端的wifi列表中新新增一個以該指定ssid的網路,所以每執行一次程式,列表中就會多一個相同名字的ssid。而該方法就是檢查wifi列表中是否有以輸入引數為名的wifi熱點,如果存在,則在CreateWifiInfo方法開始配置wifi網路之前將其移除,以避免ssid的重複:

WifiConfiguration tempConfig = this.IsExsits(SSID);           

          if

(tempConfig != null) {  

          mWifiManager.removeNetwork(tempConfig.networkId);  

          }

以上便是wifi工具類的建立,之後就可以在其他部分例項化這個類,呼叫其中的方法完成加入指定ssid的wifi熱點,還是先上程式碼吧,建立一個名為Test_wifiActivity.java的檔案(同上,沒有包含包宣告和import語句):

public class Test_wifiActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        WifiAdmin wifiAdmin = new WifiAdmin(this);
        wifiAdmin.openWifi();
        wifiAdmin.addNetwork(wifiAdmin.CreateWifiInfo("XXX", "XXX", 3));
    }
}

很簡單,如果是接入wifi,大體上只涉及到openWifi(開啟wifi)、CreateWifiInfo(配置wifi網路資訊)和addNetwork(新增配置好的網路並連線),對CreateWifiInfo進行簡單的說明:第一引數是SSID的名稱;第二個引數是指定SSID網路的密碼,當不需要密碼是置空(”“);第三個引數是熱點型別:1-無密碼 / 2-WEP密碼驗證(未測試)/ 3-WAP或WAP2 PSK密碼驗證。

最後就是在Manifest中新增相應的許可權了:

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

如果按照上述的方法進行開發,就可以很傻瓜的通過改變Test_wifiActivity.java中的CreateWifiInfo方法的三個入口引數實現加入指定SSID的wifi熱點了,無論該熱點是否需要密碼認證。

以上就是我對於自動連線指定SSID的wifi熱點的學習心得,由於水平有限有些地方可能介紹錯誤,希望大家多多批評指正!