Android线程与AsyncTask对比从BLE onCharacteristicChanged()调用的IntentService

时间:2015-02-22 04:30:26

标签: android multithreading android-asynctask bluetooth-lowenergy intentservice

我有一个Android应用程序,我从中收到BLE数据(通过通知每62毫秒)。该应用程序可以通过BufferedWriter将数据保存到文件中。在每个onCharacteristicChanged()回调时,如果用户启用了文件保存,我会调用AsyncTask,Thread或IntentService来执行文件写入。

AsyncTask似乎工作正常。但是文档说必须在UI线程上调用execute,并且我从BLE回调中调用它。那是问题吗?我该如何解决?

使用Thread导致此错误:GKI_exception超出缓冲区https://code.google.com/p/android/issues/detail?id=65455(除了我的代码不扫描但接收通知),如果文件保存很长,我需要关闭Nexus 7(应用程序和BLE)变得完全没有反应)。为什么线程不起作用,我该如何解决?

IntentService永远不会进入onHandleIntent()。这有什么问题?

以下是一些代码:

...
_context = this.getApplicationContext();
...
private BluetoothGattCallback mGattCallback = new BluetoothGattCallback() {
...
@Override
public void onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
...
int mode = 1;
if (mode==0) // Asynctask
    new doFileWriteTask().execute(strBuild.toString());
else if (mode==1)     // Thread
{
    final String str = strBuild.toString();
    new Thread(new Runnable() {
        public void run() {
           try {
               _writer.write(str);
           } catch (Exception e) {
               e.printStackTrace();
           }
        }
    }).start();
}
else if (mode==2)   // intentService
{
    Intent mServiceIntent = new Intent(_context, writeFileService.class);
    mServiceIntent.putExtra("foo", strBuild.toString());
    startService(mServiceIntent);
}
}
...
};

private class doFileWriteTask extends AsyncTask<String, Void, Void> {
@Override
protected Void doInBackground(String... strings) {
    try {
        _writer.write(strings[0]);              
    } catch (Exception e) {
        e.printStackTrace();
    }
    return null;
}

private class writeFileService extends IntentService {
    public writeFileService() {
        super("writeFileService");
    }

    @Override
    protected void onHandleIntent(Intent workIntent) {
        String dataString = workIntent.getStringExtra("foo"); 
        try {
            _writer.write(dataString);
        } catch (Exception e) {
           e.printStackTrace();
        }           
    }
}
...

1 个答案:

答案 0 :(得分:2)

  

但是文档说必须在UI线程上调用execute,并且我从BLE回调中调用它。那是问题吗?我该如何解决?

框架在调用它的同一个线程上触发AsyncTask回调方法(假定是主线程)。它并没有真正影响后台工作,但是如果你开始尝试使用onPostExecute()之类的话,你会发现问题。 AsyncTask可能不是从您无法控制的主题中调用的最佳选择。

  

为什么线程不起作用以及如何解决?

我无法确切地说出为什么你仍然会看到错误,通过产生一系列私有的非同步线程可能会导致其他问题。如果您想使用单个工作线程,更好的选择是使用一个HandlerThread,您可以使用Handler从事件回调中发布,例如:

…
_workerThread = new HandlerThread("Worker");
_workerThread.start();
_handler = new Handler(_workerThread.getLooper(), new Handler.Callback() {
        @Override
        public boolean handleMessage(Message msg) {
            String str = (String) msg.obj;
            _writer.write(str);

            return true;
        }
});
…

@Override
public void onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
    …
    Message msg = Message.obtain(_handler, 0, strBuild.toString());
    _handler.sendMessage(msg);
    …
}

该解决方案的代码相当多,但考虑到写入频率,这可能是最有效的选择。

  

IntentService永远不会进入onHandleIntent()。这里有什么问题?

你几乎不应该将顶级Android组件(活动,服务,内容提供者,接收者)实现为内部类,因为它们也必须在清单中声明(并且内部类的XML语法很丑陋) )。如果您的服务在清单中没有匹配的条目,那么您将永远不会看到它开始。您可能需要查看docs on using services

至少,作为内部类编写的Service必须public static才能正常工作。否则框架无法看到它并且无法使用默认构造函数实例化它(非静态内部类与构造函数混乱)。除非你现在在try / catch中调用startService(),否则我很惊讶它在您尝试此操作时不会崩溃。

IntentService可能是您三个选择中最简单的选择,因为它是最分离的,并且框架将处理排队工作并在完成所有传入工作时拆除线程。