使用没有Internet连接的WiFi

时间:2017-02-19 16:29:35

标签: android

在我的应用中,我正在连接到拥有自己的WiFi网络的设备。 在android 6及更高版本中,如果我想连接到这个网络,即使没有互联网连接,系统会在几秒后问我。 只有在批准该消息后,我才能连接到我的设备。 我尝试以编程方式连接到网络,而不是强迫用户转到他的设置并每次手动连接。我使用以下代码连接到设备网络:

private void connectToWiFi(WifiManager wifiManager, String wifiName) {

    WifiConfiguration configuration = new WifiConfiguration();
    configuration.SSID = "\"" + wifiName + "\"";
    configuration.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
    wifiManager.addNetwork(configuration);
    List<WifiConfiguration> list = wifiManager.getConfiguredNetworks();
    for (WifiConfiguration i : list) {
        if (i.SSID != null && i.SSID.equals("\"" + wifiName + "\"")) {
            wifiManager.disconnect();
            wifiManager.enableNetwork(i.networkId, true);
            wifiManager.reconnect();
            break;
        }
    }
}

还试图强制该应用使用WiFi连接,而不是我正在使用的手机数据:

NetworkRequest.Builder builder;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        builder = new NetworkRequest.Builder();
        builder.addTransportType(NetworkCapabilities.TRANSPORT_WIFI);

        connectivityManager.requestNetwork(builder.build(), new ConnectivityManager.NetworkCallback() {
            @Override
            public void onAvailable(Network network) {
                String ssid = wifiManager.getConnectionInfo().getSSID();
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                    connectivityManager.bindProcessToNetwork(null);
                    if (ssid.equals("\"" + Prefs.getWifiName(PUFMainActivity.this) + "\"")) {
                        connectivityManager.bindProcessToNetwork(network);
                        connectivityManager.unregisterNetworkCallback(this);
                    }
                }
            }
        });

    }

虽然只要蜂窝数据处于活动状态,设备就无法连接。如果我禁用了蜂窝数据,那么它可以正常工作。 我需要知道是否有办法以编程方式执行我想要的操作而不告诉用户禁用他的蜂窝数据。

由于

3 个答案:

答案 0 :(得分:5)

  

虽然只要蜂窝数据处于活动状态,设备就不会   apear连接。如果我禁用Cellular数据,那么它可以工作   细

我遇到了类似的问题。默认解决方案对我不起作用,因此我使用bindSocket类中的Network方法,显然它工作得很好。我不知道这个解决方案是否适用于您的问题,但这里有一些代码:

ConnectivityManager connectivityManager = (ConnectivityManager) context.getApplicationContext().getSystemService(Context.CONNECTIVITY_SERVICE);
Network[] networks = connectivityManager.getAllNetworks();
for(Network network : networks){
   NetworkInfo networkInfo = connectivityManager.getNetworkInfo(network);

   if(networkInfo.getType() == ConnectivityManager.TYPE_WIFI){
       wifiNetwork = network; // Grabbing the Network object for later usage
   }
}

并在套接字之后:

if(android.os.Build.VERSION.SDK_INT >= 22) {
   wifiNetwork.bindSocket(socket);
}

<强>更新 您也可以使用NetworkCallback来抓取Network对象。

NetworkRequest.Builder builder;
builder = new NetworkRequest.Builder();
builder.addTransportType(NetworkCapabilities.TRANSPORT_WIFI);
ConnectivityManager connectivityManager = (ConnectivityManager) context.getApplicationContext().getSystemService(Context.CONNECTIVITY_SERVICE);
connectivityManager.requestNetwork(builder.build(), new ConnectivityManager.NetworkCallback() {
   @Override
   public void onAvailable(Network network) {
      wifiNetwork = network;
   }
});

答案 1 :(得分:1)

从棉花糖开始,您的代码看起来是正确的,但请注意,您的解决方案不适用于Lollipop(API级别21和22)。

问题可能是因为您可能试图强制应用使用之后连接WiFi网络的WiFi连接。

您需要将 NetworkCallback 之前设置为仅希望通过WiFi进行通信的WiFi网络。这是因为一旦连接到所需的网络,将调用 onAvailable(),并执行 bindProcessToNetwork()

考虑在my complete solution中使用相同的方法,但也支持Lollipop。我在Android版本5.1.1、6.0、6.0.1、7.1.1和8.1.0中进行了测试。

答案 2 :(得分:1)

对于那些仍在尝试解决此问题的人,基于

wifiNetwork.bindSocket(socket)

由@MayconPrado提供

是的,该套接字变量是提前声明的。该答案假定您正在通过网络发送UDP或TCP数据包。

部分但更完整的代码看起来像

val socket = DatagramSocket(somePortNumber)
wifiNetwork.bindSocket(socket)
val udpPacket = DatagramPacket(/** args specific to use case **/)
socket.send(udpPacket)

本来会对此发表评论,但我从未发布过SO,所以我有1个代表ATM。