Метод действия доступа из интерфейса обратного вызова AsyncTask - PullRequest
0 голосов
/ 14 января 2020

У меня есть несколько действий и одна asyncTask, которая использует тот же интерфейс для методов обратного вызова, как показано ниже

Интерфейс, используемый всеми

public interface AsyncTaskCallback {
    void onCookie(CookieManager cookieManager);
    void onResponse(String response);
    void onProgress(String... values);
    void onError(Exception e);
}

AsyncTask1 вызывается из всех действий следующим образом

public void exec_taskt() {
   alertDialog.SetText("Sending Request...");
   AsyncTask1 task1 = new AsyncTask1("https://stackoverflow.com");
   task1.setCookieManager(cookiejar);
   task1.setCallback(this);
   task1.execute();
}

Каждое действие также реализует интерфейс

@Override
public void onCookie(CookieManager cookieManager) {
    cookiejar = cookieManager;
}

@Override
public void onResponse(String response) {
    try {
        PostProc(response);
    }catch (Exception e){ // ERROR HERE
        e.printStackTrace();
    }
}

@Override
public void onProgress(String... values) {
    alertDialog.SetText(values[0]);
}

@Override
public void onError(Exception e) {
    ///SAME ERROR HERE TOO
    //Toast.makeText(getContext(), e.getMessage(), Toast.LENGTH_LONG).show();
    swipeRefreshLayout.setRefreshing(false);
}

private void PostProc(String response) {
    //the response string is parsed and displayed in a recyclerview in this method
    //this method is slightly different for each activity
}

AsyncTask1

public class AsyncTask1 extends AsyncTask<String, String, String> {
   private String address = "";
   private CookieManager mCookieManager;
   private Exception mException;
   private AsyncTaskCallback mCallback;


   public AsyncTask1 (String page) {
        this.address = page;

   }

   public void setCookieManager(CookieManager cm) {
    this.mCookieManager = cm;
   }

   public void setCallback(AsyncTaskCallback cb) {
     this.mCallback = (AsyncTaskCallback) cb;

   }

   @Override
protected String doInBackground(String... strings) {
  try{
        //all code here is executed without error 
        //code skipped for simplicity 
        // basically just loads the given url and then...
        publishProgress("Page Loaded");
        mCallback.onCookie(mCookieManager);
        mCallback.onResponse(response);

        return response;
  } catch (Exception e) {
        publishProgress("Error");
        e.printStackTrace();
        mCallback.onError(e);
        return "";
    }

 @Override
protected void onProgressUpdate(String... values) {
    Log.d(tag, TextUtils.join(",", values));
    mCallback.onProgress(values);
    super.onProgressUpdate(values);
}
}

Полученные ошибки помечены

ОШИБКА ЗДЕСЬ

в коде выше. и сообщение выглядит следующим образом:

android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.

Мой главный подозреваемый - метод PostProc() в действии, которое воспроизводит компонент пользовательского интерфейса. Но разве реализация интерфейса в деятельности не означает, что часть кода выполняется в основном потоке пользовательского интерфейса? Я знаю, что вы можете передать экземпляр действия в asyncTask и вызвать метод PostProc() из postexecute asyncTask, но я хотел бы изучить альтернативный путь к нему, поскольку к той же асинхронной задаче также обращаются другие действия. доходы одинаковы для всех видов деятельности. Только метод PostProc отличается.

Заранее спасибо

1 Ответ

1 голос
/ 14 января 2020

Вместо использования callback interface вы можете использовать android.os.Handler(), который связан с Looper.getMainLooper() для обновления представлений, например: -

AsyncTask1 task1 = new AsyncTask1(android.os.Handler(Looper.getMainLooper(),  msg -> { 
 // handle message sent from asyncTask
  return false;
}), url);

Затем в task1 отправьте сообщение, используя обработчик

public AsyncTask1 (Handler handler, String page) {
    this.address = page;
    this.handler = handler;
}

затем

Message message = new Message();
message.obj = uiData;
handler.sendMessage(message);

Обработчик onHandleMessage всегда будет выполняться на main looper!

...