线程挂在WifiManager.enableNetwork()上

时间:2011-10-31 17:29:40

标签: android wifimanager

我在调用WifiManager.enableNetwork()时看到我的调用线程在本机代码中挂起。到目前为止,我只能在运行Android 3.2.1的Motorola Xoom平板电脑上重现这个问题。我已经在其他几款手机和平板电脑上测试过(全部运行Froyo或Gingerbread)并且没有看到问题。 Xoom是我必须测试的唯一双核设备(我已经在2个不同的Xoom上重现了这个问题),所以我觉得在与WifiManager接口时,我遇到了一些非常微妙的Android线程要求。我的调用线程挂起的堆栈跟踪是:

    BinderProxy.transact(int, Parcel, Parcel, int) line: not available [native method]
    IWifiManager$Stub$Proxy.enableNetwork(int, boolean) line: 513
    WifiManager.enableNetwork(int, boolean) line: 587

我的应用程序正在尝试连接到已知的wifi接入点,执行一些测试,然后将设备重新连接到其原始接入点(如果之前已连接)。在建立连接之前,我们已经验证了wifi已启用,我们已执行扫描以验证是否找到了我们的接入点SSID。建立连接的代码在AsyncTask中运行,如下所示:

... 
private WifiManager mWifiManager;
private List<WifiConfiguration> mConfiguredNets = new ArrayList<WifiConfiguration>();
private Object mConnectMonitor = new Object();
private NetworkInfo.State mNetworkState = State.UNKNOWN;

private final BroadcastReceiver mConnectionStateReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context inContext, final Intent inIntent) {
        final String action = inIntent.getAction();
        if (WifiManager.NETWORK_STATE_CHANGED_ACTION.equals(action)) {
            NetworkInfo ni =
                (NetworkInfo)inIntent.getParcelableExtra(WifiManager.EXTRA_NETWORK_INFO);
            State state = ni.getState();
            if (state == State.CONNECTED) {
                synchronized (mConnectMonitor) {
                    mNetworkState = state;
                    mConnectMonitor.notify();
                }   
            }   
        }   
    }
};  

public void runninInAsyncTask(Context activityContext, int networkID) {

    mWifiManager = (WifiManager)activityContext.getSystemService(Context.WIFI_SERVICE);

    // Register our broadcast receiver to get network state change events
    IntentFilter ifilter = new IntentFilter();
    ifilter.addAction(WifiManager.NETWORK_STATE_CHANGED_ACTION);
    activityContext.registerReceiver(mConnectionStateReceiver, ifilter);

    // Get a list of our currently configured networks so we can re-enable
    // them after connecting to the desired network
    mConfiguredNets = mWifiManager.getConfiguredNetworks();

    // Enable our network and disable all others
    mWifiManager.enableNetwork(networkId, true);

    // Start the reconnection process to connect to our desired  network
    synchronized (mConnectMonitor) {
        mWifiManager.reconnect();
        mConnectMonitor.wait(60000);
        if (mNetworkState != State.CONNECTED) {
            Log.e(TAG, "Problems connecting to desired network!");
        }
        else {
            Log.e(TAG, "Successfully connected to desired network!");
        }
    }               

    // Re-enable all of our previously configured networks
    for (WifiConfiguration wifiConfig : mConfiguredNets)
    {               
        if (wifiConfig.status != Status.ENABLED) {
            mWifiManager.enableNetwork(wifiConfig.networkId, false);
        }
    }               
}
...

此代码基于Android Gingerbread开源代码中的Wifi设置菜单代码。有什么关于调用我丢失的WifiManager.enableNetwork()吗?是否必须在特定线程上运行?我已经尝试确保在UI线程上调用enableNetwork()(通过将逻辑移动到广播接收器)。这似乎有点帮助,但我仍然能够重现悬挂。也许这是Honeycomb特有的东西?现在,这2个Xoom是我唯一可用于测试的Honeycomb设备,所以它们是我唯一拥有的数据点。

1 个答案:

答案 0 :(得分:1)

这确实是特定于3. *(似乎)的固件问题。

我见过这种情况发生在Asus Transformer TF101和索尼Tablet S上(均为3. *,就在不久前)。

从3.0开始,有一些用于连接WiFi的新API,不需要批量使用enableNetwork(以启用除当前网络之外的所有网络)。

有关这些API的更多信息,我可以从4.0源代码中收集:

  • 他们标有“@hide”
  • 设置应用
  • 使用它们
  • 从4.1
  • 开始,它们仍未记录在案
  • 他们在3. *和4. * runtime
  • 之间有所改变

我的建议是尝试通过反射使用这些API。由于它们由“设置”应用使用,因此可以正常使用。