java - 异步任务上的多个notifyObservers()不起作用

标签 java android multithreading asynchronous android-asynctask

我使用 AsyncTask 通过互联网获取和上传数据。我想在 Activity 上显示任务的当前状态。为此,我使用观察者,它将当前状态发送到我的 Activity 。这是我的 Background.java 代码的一小段:

private class Syncing extends AsyncTask<String, Void, String> {

        @Override
        protected String doInBackground(String... params) {
            sync.init();
            if (Sync.isOnline()) {
                setChanged();
                notifyObservers(ObserverStatus.UPLOAD_PENDING);
                sync.upload(); //Upload data
                setChanged();
                notifyObservers(ObserverStatus.DOWNLOAD_PENDING);
                sync.download(); //Download Data
                setChanged();
                notifyObservers(ObserverStatus.SYNC_COMPLETED);
            } else {
                setChanged();
                notifyObservers(ObserverStatus.OFFLINE);
            }
            return "Executed";
        }

        @Override
        protected void onPostExecute(String result) {
            setChanged();
            notifyObservers(ObserverStatus.FINISHED);
        }

        @Override
        protected void onPreExecute() {
        }

        @Override
        protected void onProgressUpdate(Void... values) {
        }
    }

Syncing 类嵌入在Background 类中。这段代码工作没有任何问题,直到我使用第二个 notifyObservers() 调用。因此,在这种情况下,如果用户在线并启动任务,第一个 notifyObservers() 将被调用,但第二个、第三个......不会被调用,并抛出以下错误:

05-10 21:50:21.563 16265-16285/net.myapp.example E/AndroidRuntime: FATAL EXCEPTION: AsyncTask #1
Process: net.myapp.example, PID: 16265
java.lang.RuntimeException: An error occurred while executing doInBackground()
     at android.os.AsyncTask$3.done(AsyncTask.java:325)
     at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:354)
     at java.util.concurrent.FutureTask.setException(FutureTask.java:223)
     at java.util.concurrent.FutureTask.run(FutureTask.java:242)
     at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:243)
     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133)
     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:607)
     at java.lang.Thread.run(Thread.java:761)
 Caused by: android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
     at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6892)
     at android.view.ViewRootImpl.invalidateChildInParent(ViewRootImpl.java:1083)
     at android.view.ViewGroup.invalidateChild(ViewGroup.java:5205)
     at android.view.View.invalidateInternal(View.java:13656)
     at android.view.View.invalidate(View.java:13620)
     at android.view.View.invalidate(View.java:13604)
     at android.widget.ImageView.setImageDrawable(ImageView.java:531)
     at net.myapp.example.MainActivity.update(MainActivity.java:239)
     at java.util.Observable.notifyObservers(Observable.java:161)
     at net.myapp.example.sync.Background$Syncing.doInBackground(Background.java:37)
     at net.myapp.example.sync.Background$Syncing.doInBackground(Background.java:27)
     at android.os.AsyncTask$2.call(AsyncTask.java:305)
     at java.util.concurrent.FutureTask.run(FutureTask.java:237)
     at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:243) 
     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133) 
     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:607) 
     at java.lang.Thread.run(Thread.java:761) 

在我的 Activity 中的第 238 行,将替换一个图标,您也可以在错误日志中看到。但它会发生在第 37 行。这是第二个 notifyObservers() 将被调用的地方。

我该如何解决这个问题?

最佳答案

我现在找到了解决此问题的一个选项。我现在使用onProgressUpdate。所以我添加了以下功能:

@Override
protected void onProgressUpdate(ObserverStatus... values) {
    super.onProgressUpdate(values);
    setChanged();
    notifyObservers(values[0]);
}

现在我可以通过 publishProgress() 方法发送进度。幸运的是,它工作得很好。

@Override
protected String doInBackground(String... params) {
    sync.init();
    if (Sync.isOnline()) {
        publishProgress(ObserverStatus.UPLOAD_PENDING);
        sync.upload(); //Upload data
        publishProgress(ObserverStatus.DOWNLOAD_PENDING);
        sync.download(); //Download Data
        publishProgress(ObserverStatus.SYNC_COMPLETED);
    } else {
        setChanged();
        notifyObservers(ObserverStatus.OFFLINE);
    }
    return "Executed";
}

感谢您的回答@jereksel。

关于java - 异步任务上的多个notifyObservers()不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43901837/

相关文章:

java - 如何使用 Asterisk 调用 Skype Out 电话?

java - volatile 关键字有什么用?

java - Thread.sleep 对等待 Windows 完成文件创建/移动的可运行任务的影响

java - 按键绑定(bind)没有响应

java - else语句是语法错误?

android - 在不接触系统的情况下使用自定义 BT 适配器名称在 Android 中进行 BLE 广告

android - ACRA 可以用于图书馆项目吗?

c# - 关于 Java 中的 synchronized 关键字(和 C# 的锁)的几个问题

java - 如何在 Android 中使用 cordova 创建一个基本的应用程序类

android - 如何在不发疯的情况下使用 ARM DS-5 在 eclipse 中调试 Android native 库?