移动数据打开时通过wifi(无互联网)发送数据

时间:2016-02-02 11:38:52

标签: android sockets android-wifi android-networking network-interface

我通过wifi(由设备生成)连接到硬件设备并通过套接字连接向其发送数据的开发和应用程序。 问题是,当移动数据(3G / 4G)被激活时,android会尝试通过它发送数据而不是通过设备生成的wifi发送数据,因为因为wifi没有互联网连接。 我正在考虑使用ConnectivityManager#setNetworkPreference()但它在api 21中已被弃用。

如何设置它使用设备生成的wifi而不是移动数据接口发送数据?

2 个答案:

答案 0 :(得分:14)

经过大量的研究和时间试图理解这一切后,我发现在Android 5.0(Lollipop)之前的版本中这是不可能的,操作系统一次只能保留一个网络界面并且应用程序不会控制这个。

为了在大于或等于Lollipop的版本上执行此操作,我执行了以下操作:

  1. 在进行套接字连接之前,检查android是否大于或等于Lollipop,如果不是你只是做你必须正常做的事情;
  2. 如果版本与Lollipop相同或更高,您需要执行以下操作:

    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.LOLLIPOP) {
        final ConnectivityManager manager = (ConnectivityManager) context
                .getSystemService(Context.CONNECTIVITY_SERVICE);
        NetworkRequest.Builder builder;
        builder = new NetworkRequest.Builder();
        //set the transport type do WIFI
        builder.addTransportType(NetworkCapabilities.TRANSPORT_WIFI);
        manager.requestNetwork(builder.build(), new ConnectivityManager.NetworkCallback() {
            @Override
            public void onAvailable(Network network) {
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                    manager.bindProcessToNetwork(network);
                } else {
                    //This method was deprecated in API level 23
                    ConnectivityManager.setProcessDefaultNetwork(network);
                }
                try {
                    //do a callback or something else to alert your code that it's ok to send the message through socket now
                } catch (Exception e) {
                    e.printStackTrace();
                }
                manager.unregisterNetworkCallback(this);
            }
        });
    }
    
  3. 完成后,您应该停止使用此方法绑定进程:

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        ConnectivityManager manager = (ConnectivityManager) InovePlugApplication.getContext()
                .getSystemService(Context.CONNECTIVITY_SERVICE);
        manager.bindProcessToNetwork(null);
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        ConnectivityManager.setProcessDefaultNetwork(null);
    }
    
  4. 这个答案让我无法理解如何做到https://stackoverflow.com/a/29837637/2550932

答案 1 :(得分:0)

在setProcessDefaultNetwork到wifi之后,网络将切换回移动设备,您无法通过这种方式发送大数据,因为以这种方式创建的所有套接字都将停止工作。我现在找不到完美的方式,这里仅供参考。

1.如果您可以获得root访问权限,请在连接到wifi之前执行此操作:

Process process = runtime.exec("su");
DataOutputStream dataOutputStream = new DataOutputStream(process.getOutputStream());
dataOutputStream.writeBytes("settings put global captive_portal_server 127.0.0.1\n");
dataOutputStream.writeBytes("exit\n");
dataOutputStream.flush();
int status = process.waitFor();

发送数据后删除它:

dataOutputStream.writeBytes("settings delete global captive_portal_server\n");

2.如果您可以获得WRITE_SECURE_SETTINGS权限:

Settings.Global.putString(getContentResolver(),"captive_portal_server","127.0.0.1");

或者:

Settings.Global.putInt(getContentResolver(),"captive_portal_detection_enabled",0);