При вызове BluetoothGatt.connect() возникает перехват DeadObjectException после сброса чипа Bluetooth на Android 4.3. Bluetooth BLE

Я занимаюсь разработкой приложения, которое использует преимущества нового API Bluetooth BLE из Android 4.3.

Я использовал Samsung BLE Stack для Android 4.2, и он работает нормально, даже если стабильность может быть лучше.

Теперь с 4.3 для каждого клиентского соединения у нас есть экземпляр BluetoothGatt учебный класс.

То есть я подключаюсь к устройству BLE, вызывая

BluetoothGatt gatt = device.connectGatt(this, true, callbacks);

Эти BluetoothGatt объекты - это те, которые используются для реального взаимодействия с устройством.

Поскольку я хочу подключиться к нескольким устройствам и взаимодействовать с ними, я кеширую BluetoothGatt случаи в HashMap определяется как частная собственность моей службы:

private HashMap<String, BluetoothGatt> devicesGatts 
    = new HashMap<String, BluetoothGatt>();

Ключи - это адреса устройств.

Затем, когда я хочу подключиться к устройству, BluetoothGatt экземпляр из этого HashMap:

public BluetoothGatt connectGatt(final BluetoothDevice device){
    if(device == null){
        return null;
    }
    String adr = device.getAddress();
    BluetoothGatt gatt = devicesGatts.get(adr);
    if(gatt == null){
        gatt = device.connectGatt(this, true, mGattCallbacks);
    } else {
        BluetoothDevice gattDevice = gatt.getDevice();
        if(gattDevice != null && adr.equals(gattDevice.getAddress())){
                gatt.connect(); // PROBLEM APPEARS HERE
        } else {
            gatt = device.connectGatt(this, true, mGattCallbacks);
        }
    }
    devicesGatts.put(adr, gatt);
    return gatt;
}

Проблема в том, что иногда, если устройство пытается подключиться через его кеширование BluetoothGatt например, позвонив gatt.connect()Я получаю сообщение об ошибке (не фатальное) называется DeadObjectException,

Это происходит, когда я приостанавливаю приложение и перезагружаю чип Bluetooth, а затем возобновляю приложение.

Я не удивлен, что не могу использовать кэшированный экземпляр после того, как это произошло, но я хотел бы знать, как я мог бы перехватить это исключение или обнаружить его до того, как оно произойдет.

Вот трассировка стека:

02-18 10:43:51.884: E/BluetoothGatt(23312): android.os.DeadObjectException
02-18 10:43:51.884: E/BluetoothGatt(23312):     at android.os.BinderProxy.transact(Native Method)
02-18 10:43:51.884: E/BluetoothGatt(23312):     at android.bluetooth.IBluetoothGatt$Stub$Proxy.clientConnect(IBluetoothGatt.java:841)
02-18 10:43:51.884: E/BluetoothGatt(23312):     at android.bluetooth.BluetoothGatt.connect(BluetoothGatt.java:759)
02-18 10:43:51.884: E/BluetoothGatt(23312):     at MYSERVICE.connectGatt(...)

Исключение вызывается собственным методом, но не генерируется классами стека Bluetooth.

Как я мог поймать это?

1 ответ

Решение

Я нашел обходной путь.

Я просто сбрасываю HashMap, когда чип Bluetooth выключен.

IntentFilter filter = new IntentFilter();
filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
this.registerReceiver(bluetoothStatusChangeReceiver, filter);

private final BroadcastReceiver bluetoothStatusChangeReceiver
= new BroadcastReceiver() {

    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction();
        if(action.equals(BluetoothAdapter.ACTION_STATE_CHANGED)){
            if(intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1) 
                    == BluetoothAdapter.STATE_OFF){
                devicesGatts.clear();
                resetBluetooth();
            } else if(intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1) 
                    == BluetoothAdapter.STATE_ON){
                initBluetooth();
            }
        }
    }
}

Функции initBluetooth и resetBluetooth позволяют мне сбросить BluetoothGattServer а также BluetoothManager экземпляров.

Другие вопросы по тегам