歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux編程 >> Linux編程

Android framework層實現實現wifi無縫切換AP

Android市場上有一款叫Wifijumper的軟件,實現相同ssid的多個AP之間根據wifi信號的強弱與閥值進行判斷,實現自動切換AP的功能。目前在android 4.2之前系統都沒有該功能,對於google來講,這是個相當簡單的問題,不明白為什麼一直都不支持該功能。鄙人之前在某個方案公司就遇到過客戶需要該功能。以下是鄙人實現的具體過程,希望對大家有些許的幫助。

 首先我們必須時刻監聽當前wifi的信號強度,那麼我們的手機連上wifi之後狀態蘭就會有wifi圖標出來,並且信號強度變化信號格也要跟隨變化,這部分工作是在SystemUI完成的。在目錄framework/base/packages/SystemUI/src/com/android/systemui/statusbar/policy/NetworkController.java這個類就是SystemUI監聽網絡連接狀態的Receiver。

在OnReceive方法中“

    if (action.equals(WifiManager.RSSI_CHANGED_ACTION)
                || action.equals(WifiManager.WIFI_STATE_CHANGED_ACTION)
                || action.equals(WifiManager.NETWORK_STATE_CHANGED_ACTION)) {
            updateWifiState(intent);
            refreshViews();
        }
調用了 updateWifiState(Intent intent)方法,那麼我們的切換功能就可以放到這個updateWifiState方法中:

  mWifiRssi = intent.getIntExtra(WifiManager.EXTRA_NEW_RSSI, -200);
            mWifiLevel = WifiManager.calculateSignalLevel(
                    mWifiRssi, WifiIcons.WIFI_LEVEL_COUNT);
+
+            if (mWifiConnected) { //判斷是否已鏈接wifi
+                WifiInfo wifiInfo = ((WifiManager) mContext
+                        .getSystemService(Context.WIFI_SERVICE))
+                        .getConnectionInfo();
+            //----
+                    if(wifiInfo != null && !isWorking){//當前wifi必須是能工作的
+                        String curentSSID = wifiInfo.getSSID();//獲取當前wifi的ssid
+                        String currentBssid = wifiInfo.getBSSID();
+                        Xlog.e(TAG,"updateWifiState curentSSID == "+curentSSID+" currentBssid = "+currentBssid+" mWifiRssi = "+mWifiRssi+" mWifiLevel = "+mWifiLevel );
+                        List<ScanResult> sameSSIDList = new ArrayList<ScanResult>();
+                        List<ScanResult> list = mWifiManager.getScanResults();//獲取當前掃描到的wifi列表
+                        if(list != null){
+                            for(ScanResult rt : list){
+                                Xlog.e(TAG, "list  ----------  onReceive():ScanResult = "+ rt);
+                                if(curentSSID.equals(rt.SSID)){//從wifi列表中獲取具有相同ssid的ap
+                                    sameSSIDList.add(rt);
+                                }
+                            }
+                        }
+                        if(sameSSIDList.size() >= 2){//判斷列表中存在相同ssid的ap,至少有2個
+                            ScanResult strongestRssi = sameSSIDList.get(0);
+                            for(int i = 1; i <= sameSSIDList.size()-1;i++){
+                                if(sameSSIDList.get(i).level > strongestRssi.level){//獲取信號最強的那個ap
+                                    strongestRssi = sameSSIDList.get(i);
+                                }
+                            }
+                            //int strongestLevel = WifiManager.calculateSignalLevel(
+                            //  strongestRssi.level, WifiIcons.WIFI_LEVEL_COUNT);
+                            Xlog.e(TAG, "strongestRssi = "+ strongestRssi +" mWifiRssi = "+mWifiRssi);
+                            //if(!currentBssid.equals(strongestRssi.BSSID) && (strongestLevel - mWifiLevel) >= 1){
+                            if((strongestRssi.level - mWifiRssi) >= 18){//這個18是一個信號閥值
+                                Xlog.e(TAG,"do reconnect now !!!!!!!!!!!!");
+                                //count++;
+                                //if(count > 1){
+                                    mWifiManager.disconnect();//先斷開當前wifi,再重新連接。這樣連接成功之後就是連接到了信號最強的那個ap了
+                                    mWifiManager.reconnect();
+                                    isWorking = true;
+                                    //count = 0;
+                                //}
+                            }
+                            //else{
+                            //    count = 0;
+                            //}
+                        }
+                    }
+            //----
+        }
+           

        }

邏輯其實很簡單,就是根據當前ssid去掃描的列表中找相同ssid的ap如果有,就找信號最強的,當最強的那個信號比當前的信號超過一定的值時,斷開重連。當然這種做法還是效率不是高的。原因就是當你在不停的移動中,傳過來的信號值其實和當前的實際值有一定的差異。

Copyright © Linux教程網 All Rights Reserved