zoukankan      html  css  js  c++  java
  • Android下同时使用WIFI与3G网络


    http://blog.csdn.net/roger__wong/article/details/8603275


    编程时候发现当打开WIFI网络的时候系统默认就会把3G网络给关闭,估计是出于省电的原因才如此做的,但目前我需要同时使用这两个网络接口,经过一番研究找到了方法,故写成日志记录下来。

    1、Android网络机制。

    wifi的开启关闭首先由应用程序触发的事件开始(比如用户点击开启图标),然后传入framework层,framework层启动wpa_supplicant,同时建立socket连接与wpa_supplicant进行通信,包括对wpa_supplicant的控制与得到wpa_supplicant的回传结果。若连接上了一个无线网络,则调用dhcp服务申请ip地址(或者使用预先设置好的ip,不过大部分情况下是dhcp的)。若这一切正常则断开3g网络连接并关闭3g接口。

    因此从这个角度分析,若要同时启动3g和wifi则必须绕过framework层,手动完成上述所有操作。

    因此我找到的方法是在3g网络不断的情况下,手动启动wifi网络,连接指定ap并申请ip。

    2、操作步骤。

     (1) 准备工作

    因为牵扯到底层操作,所以root必不可少,在root的情况下进入adb shell,若adb daemon没有root使用adb root命令进而使adb shell获得root权限。

    原则上使用android里的终端软件也是可以的,但我没有测试。

    (2)加载wifi驱动。

    不加载驱动在netcfg里面根本看不到wlan0网卡,因此首先要加载驱动。

    一般情况下,驱动放在/system/lib/modules下,我的驱动模块是wlan.ko。

    insmod /system/lib/modules/wlan.ko。

    加载前:

    加载后:

    发现多出来了wlan0也就是wifi网卡

    (3)开启wifi网卡

    使用netcfg wlan0 up开启wifi网卡,使用ifconfig wlan0 up也同样可以。

    开启后:

    (4)连接ap

    启动wpa_supplicant并使之连接ap,在使用android系统时,如果开启wifi后范围内有一个之前连接过的wifi(而且这个wifi的ssid和密码还没改的话),那么我们的系统就会自动连接到这个wifi。因此手机里肯定是在某处记录者各个曾经连接过的wifi的ssid和密码。

    我的手机(小米2,miui开发版3.xxx)这个默认位置是/data/misc/wifi/wpa_supplicant.conf。当把这个配置文件传给wpa_supplicant启动的时候,就会自动尝试连接其中记录的wifi网络。

    使用命令wpa_supplicant -iwlan0 -c/data/misc/wifi/wpa_supplicant.conf -B 启动wpa_supplicant,-B代表后台运行,-i指定无线网卡。

    此时若一切正常就会连接上无线ap,使用dmesg查看信息得到:

    可以看到有连接成功的信息。

    (5)使用dhcp获取ip信息

    连接上了暂时还不能用,因为没有配置ip,使用dhcpcd wlan0配置ip信息,使用netcfg wlan0 dhcp也可以。

    若出现类似以下信息则说明成功。

    此时我们就在android下同时启用了wifi和3g。

    此时使用netcfg可以看到两个接口都有ip,说明我们的方法有效。

    当然最重要的是以上操作都可以在android用简单的程序来实现。

    使用Runtime.getRuntime().exec(new String[] { "su","your command" });来执行命令,之后拿到相应输出流可以得到执行的结果,根据结果再执行下一步命令即可。


    (6)更改路由表

    到此为止手机已经连接上了无线AP,但本机数据包还无法通过WiFi接口走,原因是没有配置路由导致OS并不知道什么数据包该从WiFi走,什么数据包该从3G走,所以最后一步就是配置路由,告诉OS若IP包的源地址是3G接口的IP,则从3G端口走,否则从WiFi端口走。配置过程及命令如下,主要是使用了了ip命令:

    ip route add table 5 via 192.168.1.1 dev wlan0 //添加一个路由表,若通过该路由表路由,则通过设备wlan0路由到地址192.168.1.1,也就是无线AP

    ip route add from 192.168.1.140 table 5 prio 30000 //添加路由项,若源地址是192.168.1.140,则通过路由表5进行路由,优先级为30000

    类似的,给3G接口也添加路由条目,优先级稍微低一点

    然后设置dns,因为WiFi接口没法使用3G接口的默认DNS

    setprop net.dns1 8.8.8.8

    接着删除默认路由表项

    ip route del default via 172.18.206.76 dev rmnet_usb0

    重新添加指向wifi的默认路由表

    ip route add default via 192.168.1.140 dev wlan0




  • 相关阅读:
    linux如何查看ip地址
    mybais-plus整合springboot自动代码生成。
    org.springframework.beans.factory.UnsatisfiedDependencyException 问题
    springboot中使用AOP做访问请求日志
    springboot集成swagger
    springboot中的跨域问题
    spring中的ApplicationListener
    spring中的BeanDefinitionRegistryPostProcessor
    spring中的BeanFactoryPostProcessor
    servlet中ServletContainerInitializer
  • 原文地址:https://www.cnblogs.com/ztguang/p/12646134.html
Copyright © 2011-2022 走看看