IT_Programming/Android_Java

[펌] Android AsyncTaskLoader에 대하여

JJun ™ 2014. 11. 5. 00:14



 출처: http://itzone.tistory.com/464



최근 비동기 처리 하기 위해서 Loader 클래스가 사용되는듯 합니다.

Android3.0 (API Level 11)에서 도입된 비동기 처리를 실시하는 클래스입니다.

자식 클래스로 AsyncTaskLoader 또는 CursorLoader가 정의되어 있습니다

Loader 클래스는 SupportPackage에도 포함되어 있기 때문에 어떤 Version에서든 이용이 가능합니다.


지금까지 AsyncTask로 실행했던 것을 AsyncTaskLoader에서 구현가능합니다.

리뉴얼 된 레퍼런스 페이지의 Loader 관련 정보는 여기에 집약되어 있습니다.


그 설명을 대충 보면 안드로이드 3.0 로더는 Activity 또는 Fragment 내의 비동기 데이터 불러오기를 

쉽게 해준다고 합니다. 


Loader들의 특징 : 


· They are available to every Activity and Fragment. 

· They provide asynchronous loading of data. 

· They monitor the source of their data and deliver new results when the content changes. 

· They automatically reconnect to the last loader 's cursor when being recreated after a configuration change. Thus, they don't need to re-query their data.



비동기 처리 위한 도구 같은데,

Fragment에도 비동기 처리를 사용할 수 있도록 새롭게 재설계한 클래스일까요?

기존에는 AsyncTask를 이용하여 비동기 처리를 실현하고 있었습니다.

AsyncTask의 설명은 다음과 같이 기술되어 있습니다

AsyncTask는 적절하고 타당한 UI 쓰레드의 사용을 가능하게 합니다. 이 클래스는 UI Thread 상에서 쓰레드나 핸들러 조작 없이 백그라운드 동작과 결과 발행을 수행하도록 해줍니다.


AsyncTask는 Handler와 Thread를 이용하지 않고 UI Thread를 이용하여 Background 작업을하고 있던 것
같습니다. Loader 클래스의 설명을 보면, UI Thread와 같은 단어는 보여지지 않기 때문에 (실행해봤더니 UI 처리를하면 예외를 뱉고 강종됩니다), 실행할 수있는 것은 동일하지만, 근본적으로 설계가 다른 클래스 인 것 같습니다



AsyncTask의 동작 이미지


같은 화면의 UI를 제어하는​​ 코드가 여러 클래스에 포함되어 있습니다.


AsyncTaskLoader 동작 이미지


같은 화면에서 표시하는 Progress와 비동기 처리 완료 후 View 갱신 처리 등 UI 제어에 대한 코드는
한자리에 있습니다. AsyncTaskLoader에서 결과만 LoaderCallback을 통해 통지됩니다.


다음으로 AsyncTaskLoader를 상속하여 구현 한 경우, 구체적으로 어떤 차이가 있는지를 살펴 보겠습니다.


구현 예제 (기본 바탕)


URL을 지정하여 Web 상에있는 이미지를 가져오고 그것을 ImageView에 표시하는 작업을 살펴​​ 보겠습니다






[AsyncTask를 사용하는 경우]


public class DownloadImageAsyncTaskHelper extends AsyncTask<string, string, bitmap> {

  

    @ Override

    protected Bitmap doInBackground(String... urls) {

         String url = urls[0];

           

         // Progress Dialog보기

         publishProgress("Loading from" + url);

         return HttpUtil.getBitmapHttpService(context, url);

     }

  

     @ Override

     protected void onProgressUpdate(String... progress) {

         // ProgressDialog에 문자열을 설정

         setProgressMessage(progress[0]);

     }

  

     @ Override

     protected void onPostExecute(Bitmap result) {

         // 비동기 적으로로드가 완료 이미지를 View에 표시

         showImage(result);

     }

}


비동기 처리에 필요한 초기 매개 변수, 경과를 나타내는 Progress 매개 변수, 결과의 형태 등 지정하는 것이 많네요. 또한 doInBackground()에서 실행되는 비동기 이미지 읽기와 showImage() setProgerssMessage() 등의 UI 업데이트 작업도 포함되어 있습니다. AsyncTask는 UI Thread에서 실행되기 때문에 UI의 변경도 가능하기 때문입니다.






[AsyncTaskLoader를 이용하는 경우]


마찬가지로 URL을 지정하고 Web에서 이미지를 가져 와서 ImageView에 표시하는 작업을 구현 봅니다


public class DownloadImageAsyncTaskLoaderHelper extends AsyncTaskLoader<bitmap> {

  

    private String imageUrl = "";

    private Context Context = null;

      

    public DownloadImageAsyncTaskLoaderHelper(Context context, String url) {

        super (context);

          

        this.imageUrl = url;

        this.context = context;

    }

  

    @Override

    public Bitmap loadInBackground() {

         return HttpUtil.getBitmapHttpService(context, url);

    }

      

    @Override

    protected void onStartLoading() {

        forceLoad();

    }

}


Progress 표시 비동기 처리 종료시 View 업데이트 등의 UI 제어 코드가 없어 정돈된 느낌입니다. UI Thread가 얽혀 있지 않기 때문에 UI 변경은 여기서 할 수 없습니다. 만약 AsyncTask처럼 UI 변경하면 UI Thread 이외 View 변경 때 Throw되는 항상 예외가 발생합니다.


Caused by : android.view.ViewRoot $ CalledFromWrongThreadException : only the original thread that created a view hierarchy can touch its views.

또한 AsyncTask에 비해 다른 것은 결과의 형식만 지정되어 있습니다. 

그럼, 어디서 종료시 View 업데이트 등의 UI 업데이트를 실시하면 좋을까요?


AsyncTaskLoader 결과를 받으려면 LoaderCallbacks 이라는 Interface를 구현 해주는 것으로, 비동기 처리 결과를 받을 수 있습니다. 이 때 주의해야 할 점이 있습니다. SupportPakcage를 이용할 경우, Loader를 이용하는 방법(getSupportLoaderManager)이 "FragmentActivity"에 포함되어 있기 때문에 Activity에서 실행하려면 "FragmentActivity"을 상속 해 줄 필요가 있습니다


public class MainActivity extends FragmentActivity Implements LoaderCallbacks<bitmap> {

    // Loader의 초기화에서 시작까지

    public void startAsyncLoadImage(String url) {

        Bundle args = New Bundle();

        args.putString("url", url);

        getSupportLoaderManager().initLoader(0, args, this);     // onCreateLoader가 호출

          

        // 여러 Loader를 동시에 움직일 경우에는 첫 번째 인수를 고유 ID를 해 줄 필요가 있습니다.

        // GridView 등에 표시하는 이미지를 비동기로 단번에 취득하는 경우 나

    }

      

    @Override

    public Loader<bitmap> onCreateLoader(int ID, Bundle args) {         

        // 비동기 처리를 수행하는 Loader를 생성합니다.

        // 여기를 전환주는 것만으로 다양한 비동기 처리에 대응할 수 있습니다.

        if (args! = null) {

            String url = args.getString("url");

            return New DownloadImageAsyncTaskLoaderHelper(this, url);

        }

    }

      

    @Override

    public void onLoadFinished(Loader<bitmap> arg0, Bitmap arg1) {     

        // 비동기 처리가 끝나면 불립니다.

        // 이번은 Download가 완료 이미지를 ImageView에 표시합니다.

        ImageView imageView = (ImageView)findViewById(R.id.imageview);

        Drawable iconImage = New BitmapDrawable(getResources(), bmp);

        imageView.setImageDrawable(iconImage);

        imageView.invalidate();

    }

      

    @Override

    public void onLoaderReset(Loader<bitmap> arg0) {

          

    }

}


비동기 처리 부분과 UI 업데이트 부분을 명확하게 나눌 수 있습니다. 

UI의 제어에 관련되는 부분이한데 모아져 코드가 분산되지 않기 때문에 기존보다 역할 분담이 

명확하게 되었습니다.



정리

AsyncTaskLoader를 사용하면 UI 제어 및 비동기 처리의 역할 분담을 명확하게 나눌 수있다.

SupportPackage에 들어 있기 때문에, 지금이라도 사용 가능.

무엇보다 최근의 비동기 처리 구현이므로 멋지다. (아마도)