多線程及AsyncTask

多線程及AsyncTask

一、android中的多線程

在Android中,當一個程序的第一個組件啓動時,android系統會爲其開闢一個新的線程,這就是main thread(主線程)。默認的情況下,一個程序的所有組件都運行在同一個線程中,及主線程,當我們通過某個組件來啓動另一個組件時,也是在主線程中進行。

二、Main Thread 和Worker Thread

在android中,通常將線程分爲兩種,一種是Main Thread,除了Main Thread以外的所有線程都可稱之爲Worker Thread。

當一個程序開始運行後,系統會爲其開闢一個新進程,即main thread,這個線程是最重要的線程,它用來加在UI界面,完成系統和用戶之間的交互,並將交互後的結果展示給用戶,所以Main Thread又被稱爲UI Thread。

Android系統默認不會給我們的應用程序組件創建一個額外的線程,所有的這些組件默認都是在同一個線程中運行。然而,某些時候當我們的應用程序需要完成一個耗時的操作的時候,例如訪問網絡或者是對數據庫進行查詢時,此時我們的UI Thread就會被阻塞。例如,當我們點擊一個Button,然後希望其從網絡中獲取一些數據,如果此操作在UI Thread當中完成的話,當我們點擊Button的時候,UI線程就會處於阻塞的狀態,此時,我們的系統不會調度任何其它的事件,更糟糕的是,當我們的整個現場如果阻塞時間超過5秒鐘(官方是這樣說的),這個時候就會出現 ANR (Application Not Responding)的現象,此時,應用程序會彈出一個框,讓用戶選擇是否退出該程序。對於Android開發來說,出現ANR的現象是絕對不能被允許的。

另外,由於我們的Android UI控件是線程不安全的,所以我們不能在UI Thread之外的線程當中對我們的UI控件進行操作。因此在Android的多線程編程當中,我們有兩條非常重要的原則必須要遵守:

  • 絕對不能在UI Thread當中進行耗時的操作,不能阻塞我們的UI Thread,例如網絡通信。
  • 不能在UI Thread之外的線程當中操縱我們的UI元素。

三、如何處理UI Thread 和 Worker Thread之間的通信

既然在Android當中有兩條重要的原則要遵守,那麼我們可能就有疑問了?我們既不能在主線程當中處理耗時的操作,又不能在工作線程中來訪問我們的UI控件,那麼我們比如從網絡中要下載一張圖片,又怎麼能將其更新到UI控件上呢?這就關係到了我們的主線程和工作線程之間的通信問題了。在Android當中,提供了兩種方式來解決線程直接的通信問題,一種是通過Handler的機制,還有一種就是今天要詳細講解的 AsyncTask 機制。

四、Async Task主要方法介紹

AsyncTask:異步任務,從字面上來說,就是在我們的UI主線程運行的時候,異步的完成一些操作。AsyncTask允許我們的執行一個異步的任務在後臺。我們可以將耗時的操作放在異步任務當中來執行,並隨時將任務執行的結果返回給我們的UI線程來更新我們的UI控件。通過AsyncTask我們可以輕鬆的解決多線程之間的通信問題。

怎麼來理解AsyncTask呢?通俗一點來說,AsyncTask就相當於Android給我們提供了一個多線程編程的一個框架,其介於Thread和Handler之間,我們如果要定義一個AsyncTask,就需要定義一個類來繼承AsyncTask這個抽象類,並實現其唯一的一個 doInBackgroud 抽象方法。要掌握AsyncTask,我們就必須要一個概念,總結起來就是: 3個泛型,4個步驟順序。

AsyncTask 時需要指定如下三個泛型參數。
AsyncTask <Params, Progress, Result>
    Params :啓動任務執行的輸入參數的類型。
    Progress :後臺任務完成的進度值的類型。
    Result :後臺執行任務完成後返回結果的類型。
使用 AsyncTask 要注意以下四個方法的順序和返回值

1.onPreExecute(): 這個方法是在執行異步任務之前的時候執行,並且是在UI Thread當中執行的,通常我們在這個方法裏做一些UI控件的初始化的操作,例如彈ProgressDialog。

2.doInBackground(Params… params): 在onPreExecute()方法執行完之後,會馬上執行這個方法,這個方法就是來處理異步任務的方法,Android操作系統會在後臺的線程池當中開啓一個worker thread來執行我們的這個方法,所以這個方法是在worker thread當中執行的,這個方法執行完之後就可以將我們的執行結果發送給我們的最後一個onPostExecute 方法,在這個方法裏,我們可以從網絡當中獲取數據等一些耗時的操作。
補充:d oInBackground(“param1”,“param2”,“param3”) ,或是doInBackground() 。實際上,在處理可變參數列表的時候,Java是轉化爲數組來處理的,比如前面的例子,doInBackground傳進來三個參數,此時params實際上是一個String[3],可以通過params[0]來引用傳進來的實參"param1"。當doInBackground()沒有提供實參時params就爲null。另外,對於可變參數列表的方法,我們可以直接傳遞一個數組代替,比如doInBackground(new String[]{“param1”, “param2”, “param3”}),效果和doInBackground(“param1”,“param2”,“param3”)是一樣的

3.onProgressUpdate(Progess… values): 這個方法也是在UI Thread當中執行的,我們在異步任務執行的時候,有時候需要將執行的進度返回給我們的UI界面,例如下載一張網絡圖片,我們需要時刻顯示其下載的進度,就可以使用這個方法來更新我們的進度。這個方法在調用之前,我們需要在 doInBackground 方法中調用一個 publishProgress(Progress) 的方法來將我們的進度時時刻刻傳遞給 onProgressUpdate 方法來更新。

4.onPostExecute(Result… result): 當我們的異步任務執行完之後,就會將結果返回給這個方法,這個方法也是在UI Thread當中調用的,我們可以將返回的結果顯示在UI控件上

使用 AsynvTask 時必須遵守如下規則:

必須在 UI 線程中創建 AsyncTask 的實例。
必須在 UI 線程中調用 AsyncTask 的execute() 方法。

爲什麼我們的AsyncTask抽象類只有一個 doInBackground 的抽象方法呢??原因是,我們如果要做一個異步任務,我們必須要爲其開闢一個新的Thread,讓其完成一些操作,而在完成這個異步任務時,我可能並不需要彈出要給ProgressDialog,我並不需要隨時更新我的ProgressDialog的進度條,我也並不需要將結果更新給我們的UI界面,所以除了 doInBackground 方法之外的三個方法,都不是必須有的,因此我們必須要實現的方法是 doInBackground 方法。

下載圖片的demo:

public class MainActivity extends AppCompatActivity {

    private Button button;
    private ImageView imageView;
    private ProgressDialog progressDialog;

    private String url="https://wx3.sinaimg.cn/mw690/005FQpaTgy1fowcuh0zinj30sg0g0h6s.jpg";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        button=findViewById(R.id.btn);
        imageView=findViewById(R.id.iv);

        progressDialog=new ProgressDialog(MainActivity.this);
        progressDialog.setTitle("Warning");
        progressDialog.setMessage("Downloading");
        progressDialog.setCancelable(false);
        progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);

        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                new MyAsyncTask().execute(url);
            }
        });

    }

    public  class MyAsyncTask extends AsyncTask<String, Integer, byte[]> {


        @Override
        protected void onPreExecute() {
            super.onPreExecute();

            progressDialog.show();
        }




        @Override
        protected byte[] doInBackground(String... strings) {
            HttpClient httpClient = new DefaultHttpClient();
            HttpPost httpPost = new HttpPost(strings[0]);
            byte[] image = new byte[]{};

            try
            {
                HttpResponse httpResponse = httpClient.execute(httpPost);
                HttpEntity httpEntity = httpResponse.getEntity();
                InputStream inputStream = null;
                ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
                if(httpEntity != null && httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK)
                {
                    //    得到文件的總長度
                    long file_length = httpEntity.getContentLength();
                    //    每次讀取後累加的長度
                    long total_length = 0;
                    int length = 0;
                    //    每次讀取1024個字節
                    byte[] data = new byte[1024];
                    inputStream = httpEntity.getContent();
                    while(-1 != (length = inputStream.read(data)))
                    {
                        //    每讀一次,就將total_length累加起來
                        total_length += length;
                        //    邊讀邊寫到ByteArrayOutputStream當中
                        byteArrayOutputStream.write(data, 0, length);
                        //    得到當前圖片下載的進度
                        int progress = ((int)(total_length/(float)file_length) * 100);
                        //    時刻將當前進度更新給onProgressUpdate方法
                        publishProgress(progress);
                    }
                }
                image = byteArrayOutputStream.toByteArray();
                inputStream.close();
                byteArrayOutputStream.close();
            }
            catch (Exception e)
            {
                e.printStackTrace();
            }
            finally
            {
                httpClient.getConnectionManager().shutdown();
            }
            return image;
        }
        @Override
        protected void onProgressUpdate(Integer... values)
        {
            super.onProgressUpdate(values);
            //    更新ProgressDialog的進度條
            progressDialog.setProgress(values[0]);
        }

        @Override
        protected void onPostExecute(byte[] result) {
            super.onPostExecute(result);
            //    將doInBackground方法返回的byte[]解碼成要給Bitmap
            Bitmap bitmap = BitmapFactory.decodeByteArray(result, 0, result.length);
            //    更新我們的ImageView控件
            imageView.setImageBitmap(bitmap);
            //    使ProgressDialog框消失
            progressDialog.dismiss();
        }
    }

}

本篇文章借鑑搬運於https://www.cnblogs.com/xiaoluo501395377/p/3430542.html
在此基礎上加入了自己的理解。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章