Android Thread、AsyncTask 与从 BLE onCharacteristicChanged() 调用的 IntentService

标签 android multithreading android-asynctask bluetooth-lowenergy intentservice

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

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

使用线程会导致此错误:GKI_exception out of buffers 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();
        }           
    }
}
...

最佳答案

But the docs say execute must be invoked on the UI thread, and I'm calling it from the BLE callback. Is that a problem? And how should I fix it?

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

Why does the Thread not work and how can I fix it?

我不能确切地说为什么你仍然看到错误,通过产生一系列私有(private)的不同步线程可能会导致其他令人头痛的问题。如果您想使用单个工作线程,更好的选择是使用单个 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);
    …
}

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

The IntentService never goes to the onHandleIntent(). What are the issues here?

您几乎不应该将顶级 Android 组件( Activity 、服务、内容提供程序、接收器)实现为内部类,因为它们也必须在您的 list 中声明(并且内部类的 XML 语法很丑陋) )。如果您的服务在 list 中没有匹配的条目,那么您将永远不会看到它启动。您可能想看看 docs on using services .

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

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

关于Android Thread、AsyncTask 与从 BLE onCharacteristicChanged() 调用的 IntentService,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28654576/

相关文章:

ruby - 在线程内部生成线程是标准做法还是不好的做法?

python - 如何组合线程?

java - 如何将整数数组从 AsyncTask 获取到同一 Activity 中的另一个方法?

android - 如何在没有 runOnUiThread() 的情况下在服务中创建/运行 AsyncTask

android - Rxjava : Insert only works with single but not with completable

java - 当使用 Home 而不是 Back 时,onActivityResult() 永远不会被调用

java swing keylistener 和线程

Android AsyncTask 实例或类

java - 如何在Mpandroid饼图中动态添加文本?

android - 如何将相机中的图像保存到内存中?