http://www.linuxidc.com/Linux/2013-12/93476.htm

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如果有,就找信号最强的,当最强的那个信号比当前的信号超过一定的值时,断开重连。当然这种做法还是效率不是高的。原因就是当你在不停的移动中,传过来的信号值其实和当前的实际值有一定的差异。

Android framework层实现实现wifi无缝切换AP的更多相关文章

  1. 怎样从C++代码直接訪问android framework层的WifiService

    说究竟,Java层的service就是就C++层的binder的封装.所以从原理上来讲通过C++代码直接訪问android framework层的service是全然可能的,这篇文章以訪问WifiSe ...

  2. Android Framework层Power键关机流程(一,Power长按键操作处理)

    一:Android处理Power按键长按操作 在Framework层中,Android4.x对Power键(KeyEvent.KEYCODE_POWER)的操作,我们从PhoneWindowManag ...

  3. Android Framework层Power键关机流程(二,关机流程)

    二,关机流程 从前一篇博文我们知道,当用户长按Power键时会弹出(关机.重启,飞行模式等选项)对话框,我们点击关机,则会弹出关机确认对话框.那么从选项对话框到关机确认对话框又是一个什么流程呢.下面我 ...

  4. 如何调试Android Framework?

    Linus有一句名言广为人知:Read the fucking source code. 但其实,要深入理解某个软件.框架或者系统的工作原理,仅仅「看」代码是远远不够的.就拿Android Frame ...

  5. MediaPlayer: 在不同控件之间实现视频的无缝切换的方法

    最近使用MediaPlayer + TextureView 实现了一个视频播放器,并且实现了它的横竖屏切换的效果,唯一美中不足的是在横竖屏切换的时候画面会卡顿一下,虽然也不影响播放,但是怕测试会报Bu ...

  6. Android源码剖析之Framework层升级版(窗口、系统启动)

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 看本篇文章之前,建议先查看: Android源码剖析之Framework层基础版 前面讲了frame ...

  7. Android源码剖析之Framework层基础版(窗口、linux、token、Binder)

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 关于Framework,就是应用层底下的控制层,离应用层最近,总想找个机会,写写WindowMang ...

  8. Android如何完全调试framework层代码

    1 之前写过一篇文章:<Android实现开机调试system_process> 2 google的eclipse插件ADT的已经能够很方便的调试Android的apk了,但是调试的时候应 ...

  9. Android主题换肤 无缝切换

    2016年7月6日 更新:主题换肤库子项目地址:ThemeSkinning,让app集成换肤更加容易.欢迎star以及使用,提供改进意见. 更新日志: v1.3.0:增加一键切换切换字体(初版)v1. ...

随机推荐

  1. Redis 在 LINUX 系统下 安装, 启动

    01, 下载  http://www.redis.cn/ ,  这里下再下来的是 redis-4.0.1.tar.gz 这个压缩包 02, 将压缩包放到 linux 系统中,  一般放在 usr/lo ...

  2. 深入浅出理解依赖注入这种由外部负责其依赖需求的行为,我们可以称其为 “控制反转(IoC)”

    原文地址: http://www.insp.top/learn-laravel-container ,转载务必保留来源,谢谢了! 这个组件现在可以很简单的获取到它所需要的服务,服务采用延迟加载的方式, ...

  3. Tocmat 启动错误 Port 8005 required by tomcat v7.0 server at localhost is already in use

    1: netstat -ano!findstr 8005 2: taskkill /pid 6476 /f

  4. AS3 localToGlobal、globalToLocal方法的总结

    (1).localToGlobal (point)   把point看成在(1)内部,计算出该point相当于stage的坐标 (2).globalToLocal (point)   point为全局 ...

  5. Django之 Model Field Options

    以下这些选项都是可选择的,非固定要求. 1)null,注意在CharField或者TextField里避免使用null,因为其存储的值是空字符串而不是NULL 2)blank该字段是否可以为空.如果为 ...

  6. python传参数

    Python参数传递采用的肯定是“传对象引用”的方式.这种方式相当于传值和传引用的一种综合.如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对 ...

  7. JSTL标签库学习记录2-fmt

    fmt的标签为辅助性功能标签 设置编码 <fmt:requestEncoding value=""> 国际化相关 <fmt:setLocale value=&qu ...

  8. 初涉定制linux系统之——rpm相关安装包的准备

    在上一篇博客http://www.cnblogs.com/dengtr/p/5543820.html#3634582 中介绍了如何定制Centos系统镜像,但其中有个问题,就是服务所依赖的安装包不在原 ...

  9. 103041000997维护的是周批,按周合并后再考虑最小采购批量、舍入值、然后回写到SAP系统

    描述:103041000997维护的是周批量,但最终没有按周批量来回写数据. 业务逻辑如下: 1.净需求考虑数量按周汇总(也有按日.按3天,具体 要根据物料主数据维护来判断) 2.第1点的结果再加上安 ...

  10. pidgin的未认证解决办法

    安全验证打开还是无法登陆,并且手机无法验证. 解决:开启手机二次验证,给pidgin设置专门的登陆账户密码,即可解决.