<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    Calvin's Tech Space

    成于堅忍,毀于浮躁

       :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
    當(dāng)?shù)谝淮螁右粋€Android程序時,Android會自動創(chuàng)建一個稱為“main”主線程的線程。這個主線程(也稱為UI線程)很重要,因為它負(fù)責(zé)把事件分派到相應(yīng)的控件,其中就包括屏幕繪圖事件,它同樣是用戶與Andriod控件交互的線程。比如,當(dāng)你在屏幕上按下一個按鈕后,UI線程會把這個事件分發(fā)給剛按得那個按鈕,緊接著按鈕設(shè)置它自身為被按下狀態(tài)并向事件隊列發(fā)送一個無效(invalidate)請求。UI線程會把這個請求移出事件隊列并通知按鈕在屏幕上重新繪制自身。

     

    單線程模型會在沒有考慮到它的影響的情況下引起Android應(yīng)用程序性能低下,因為所有的任務(wù)都在同一個線程中執(zhí)行,如果執(zhí)行一些耗時的操作,如訪問網(wǎng)絡(luò)或查詢數(shù)據(jù)庫,會阻塞整個用戶界面。當(dāng)在執(zhí)行一些耗時的操作的時候,不能及時地分發(fā)事件,包括用戶界面重繪事件。從用戶的角度來看,應(yīng)用程序看上去像掛掉了。更糟糕的是,如果阻塞應(yīng)用程序的時間過長(現(xiàn)在大概是5秒鐘)Android會向用戶提示一些信息,即打開一個“應(yīng)用程序沒有相應(yīng)(application not responding)”的對話框。

     

    如果你想知道這有多糟糕,寫一個簡單的含有一個按鈕的程序,并為按鈕注冊一個單擊事件,并在事件處理器中調(diào)用這樣的代碼Thread.sleep(2000)。在按下這個按鈕這后恢復(fù)按鈕的正常狀態(tài)之前,它會保持按下狀態(tài)大概2秒鐘。如果這樣的情況在你編寫的應(yīng)用程序中發(fā)生,用戶的第一反應(yīng)就是你的程序運行很慢。

     

    現(xiàn)在你知道你應(yīng)該避免在UI線程中執(zhí)行耗時的操作,你很有可能會在后臺線程或工作者線程中執(zhí)行這些耗時的任務(wù),這樣做是否正確呢?讓我們來看一個例子,在這個例子中按鈕的單擊事件從網(wǎng)絡(luò)上下載一副圖片并使用ImageView來展現(xiàn)這幅圖片。代碼如下:

     
    Java代碼
    public void onClick( View v ) {   
            new Thread( new Runnable() {   
                public void run() {   
                    Bitmap b = loadImageFromNetwork();   
                    mImageView.setImageBitmap( b );   
                }          
             }).start();   
    }  

     

     這段代碼好像很好地解決了你遇到的問題,因為它不會阻塞UI線程。很不幸,它違背了單線程模型:Android UI操作并不是線程安全的并且這些操作必須在UI線程中執(zhí)行。在這段代碼片段中,在一個工作者線程中使用ImageView的方法,這回引起一些很古怪的問題。查處這個問題并修復(fù)這個bug會很困難而且也很耗時。

     

    Andriod提供了幾種在其他線程中訪問UI線程的方法。或許你已經(jīng)對其中的一些方式很熟悉,但下面是一個更全面的列表:
    Activity.runOnUiThread( Runnable )
    View.post( Runnable )
    View.postDelayed( Runnable, long )
    Hanlder

    上面的任何一個類或方法都可以修復(fù)我們前面代碼中出現(xiàn)的問題。

     
    Java代碼
    public void onClick( View v ) {   
            new Thread( new Runnable() {   
                    public void run() {   
                             final Bitmap b = loadImageFromNetwork();   
                             mImageView.post( new Runnable() {   
                                      mImageView.setImageBitmap( b );   
                              });   
                     }   
            }).start();   
    }  

     

    很不幸的是這些類或方法同樣會使你的代碼很復(fù)雜很難理解。然而當(dāng)你需要實現(xiàn)一些很復(fù)雜的操作并需要頻繁地更新UI時這會變得更糟糕。為了解決這個問題,Android 1.5提供了一個工具類:AsyncTask,它使創(chuàng)建需要與用戶界面交互的長時間運行的任務(wù)變得更簡單。

     

    在Android 1.0和1.1中具有與AsyncTask相同功能的類UserTask。它提供了完全一樣的API,你需要做的只是把它的代碼拷貝的你的程序中。

     

    AsyncTask的目標(biāo)是替你管理你的線程。前面的代碼可以很容易地使用AsyncTask重寫。

     
    Java代碼
    public void onClick( View v ) {   
         new DownloadImageTask().execute( "http://example.com/image.png" );   
    }   
     
    private class DownloadImageTask extends AsyncTask {   
         protected Bitmap doInBackground( String... urls ) {   
              return loadImageFormNetwork( urls[0] );   
         }   
     
         protected void onPostExecute( Bitmap result ) {   
             mImageView.setImageBitmap( result );   
         }   
    }  

     

    正如你看到的,使用AsyncTask必須要繼承它。使用AsyncTask非常重要的是:AsyncTask的實例必須在UI線程中創(chuàng)建而且只能被使用一次。你可以使用預(yù)讀AsyncTask的文檔來來了解如何使用這個類,下面大概地了解一下它是如何工作的:
    你可以使用泛型參數(shù)制定任務(wù)的參數(shù)、中間值(progress values)和任何的最終執(zhí)行結(jié)果
    doInBackground()方法會自動地在工作者線程中執(zhí)行
    onPreExecute()、onPostExecute()和onProgressUpdate()方法會在UI線程中被調(diào)用
    doInBackground()方法的返回值會被傳遞給onPostExecute()方法
    在doInBackground()方法中你可以調(diào)用publishProgress()方法,每一次調(diào)用都會使UI線程執(zhí)行一次onProgressUpdate()方法
    你可以在任何時候任何線程中取消這個任務(wù)

    除了官方的文檔,你可以閱讀Shelves和Photostream源代碼中的幾個復(fù)雜的示例。我強烈地推薦閱讀Shelves的源代碼,它會使你知道如何在配置更改之間持久化任務(wù)以及在activity被銷毀時正確的取消任務(wù)。

     

    不管是否使用AsyncTask,始終記住以下兩個關(guān)于單線程模型的準(zhǔn)則:不要阻塞UI線程以及一切Android UI操作都在UI線程中執(zhí)行。AsyncTask僅僅是使你能夠更容易地遵守這兩條準(zhǔn)則。

    轉(zhuǎn)自http://kimilv.javaeye.com/blog/382683
    posted on 2010-02-07 15:24 calvin 閱讀(250) 評論(0)  編輯  收藏 所屬分類: Android
    主站蜘蛛池模板: 亚洲 综合 国产 欧洲 丝袜 | 国产精品久久香蕉免费播放| 亚洲日本中文字幕| 水蜜桃视频在线观看免费播放高清 | 99久久国产免费中文无字幕| 亚洲一区二区三区无码中文字幕| 一级做a爰片久久毛片免费陪| 亚洲午夜精品一级在线播放放| 视频一区在线免费观看| 亚洲精品无码久久不卡| 一个人看的免费视频www在线高清动漫| 亚洲AV无码乱码在线观看| 人成免费在线视频| 亚洲成AV人片在线观看无码 | 午夜亚洲WWW湿好爽| 国产一级淫片a免费播放口之 | 78成人精品电影在线播放日韩精品电影一区亚洲| 国产无遮挡无码视频免费软件| 久久久久亚洲av无码专区导航| 国产电影午夜成年免费视频| 亚洲日本VA中文字幕久久道具| 永久免费AV无码网站在线观看| 理论亚洲区美一区二区三区| 自拍偷自拍亚洲精品被多人伦好爽 | 亚洲五月综合缴情在线观看| 老汉精品免费AV在线播放| 亚洲高清有码中文字| 又黄又爽一线毛片免费观看| 中文精品人人永久免费| 亚洲午夜精品在线| 亚洲AⅤ视频一区二区三区| 九九精品成人免费国产片| 亚洲人成网国产最新在线| 免费精品国产自产拍观看| 国产免费网站看v片在线| 亚洲人成网站在线观看播放青青| 国产成人精品高清免费| 光棍天堂免费手机观看在线观看| 亚洲av无码专区在线| 亚洲午夜福利精品无码| 114一级毛片免费|