Android類FileDownloadList分析_Android

來源:互聯網
上載者:User

先上代碼,再來分析

public class FileDownloadList {  /**上下文*/ private Context mContext; /**請求對象*/ private BaseRequestLims fileRequest = null; /**進度條對話方塊*/ private AlertDialog progressDialog = null; /**進度條控制項變數*/ private ProgressBar mProgress; /**百分比顯示控制項*/ private TextView mProgressPercent;  private File localFile = null; /**接收HttpHelper中擷取到檔案大小後發送的廣播,確定檔案大小*/ private DownLoadReceiver receiver; /**檔案大小*/ private long fileLength = -1L; /**是否登入廣播標誌*/ private boolean castFlag = false; /**是否顯示進度條標誌*/ private boolean showDialog = false; /**檔案下載完的回調介面*/ private Runnable mCallback = null;  private Handler mHandler = new Handler(){  @Override  public void handleMessage(Message msg) {   super.handleMessage(msg);   int tempSize = (int)localFile.length();   if(tempSize < fileLength){//檔案下載中        if(showDialog){//顯示了進度條的情況下,更新進度條          int progress = (int)((Double.valueOf(tempSize) / Double.valueOf(fileLength)) * 100);          mProgress.setProgress(tempSize);          mProgressPercent.setText(progress + "%");        }      }else{//下載檔案完畢    if(castFlag){//如登入廣播,登出廣播      mContext.unregisterReceiver(receiver);      castFlag = false;      }    if(showDialog){     mProgress.setProgress((int)fileLength);     mProgressPercent.setText("100%");     progressDialog.dismiss();    }              if(mCallback != null){     try{      Thread.sleep(500);      mCallback.run();     }catch (Exception e) {      e.printStackTrace();     }    }   }  } };  /**  * 構造器  * @param activity  */  /**  * 構造器  * @param activity  * @param showDialog 顯示進度條標誌  */ public FileDownloadList(Context context, boolean showDialog){  mContext = context;  this.showDialog = showDialog;  fileRequest = new BaseRequestLims(context,ClientServiceType.FILE_DOWN);  fileRequest.setMethodType(BaseRequestLims.METHOD_TYPE_POST);  fileRequest.setContext(mContext);     }  public BaseRequestLims getFileRequest(){  return fileRequest; }  /**  * 通過關聯類別型來下載檔案  * @param fileName 檔案名稱或檔案在伺服器上的相對路徑加名稱  * @param saveDir 儲存在本地的檔案目錄  * @param saveName 儲存在本地的檔案名稱  * @param gllx 關聯類別型  * @param callback 下載後的處理線程  */ public void downloadFile(String fileName, String saveDir, String saveName, Runnable callback){  if(callback != null){   mCallback = callback;  }    File saveDirFile = new File(saveDir);    //judge the save dir path exist or not   if(!saveDirFile.exists()){   saveDirFile.mkdirs();  }  localFile = new File(saveDir,saveName);    if(localFile.isDirectory()){   new AlertDialog.Builder(mContext).setTitle("提示").setMessage("the save file is directory").show();   return;  }  if(fileRequest.getServiceType()==null){   fileRequest.setServiceType(ClientServiceType.FILE_DOWN);  }  fileRequest.addParameter("fpath", fileName);  fileRequest.addParameter("fname", saveName);  fileRequest.setStreamPath(localFile.getAbsolutePath());  fileRequest.setStream(true);  if(localFile.exists()){   if(localFile.length() == 0){    invokeFile(fileRequest);   }else{    //檔案存在直接開啟    if(showDialog)     buildProgressDialog().show();    mHandler.sendMessage(mHandler.obtainMessage());   }  }else{   invokeFile(fileRequest);  } }  /**  * 進入檔案下載子線程  * @param request  */ private void invokeFile(final BaseRequestLims request){  try{   if(showDialog){    buildProgressDialog().show();   }   receiver = new DownLoadReceiver();   IntentFilter filter = new IntentFilter();   filter.addAction("SAVE_DOWNLOAD_FILE");   mContext.registerReceiver(receiver, filter);   castFlag = true;   //下載的子線程   new Thread(){    @Override    public void run() {     super.run();     HttpHelper.invoke(request);    }   }.start();  }catch (Exception e) {   e.printStackTrace();  } }  /**  * 建立進度對話方塊  * @return  */ private AlertDialog buildProgressDialog(){  AlertDialog.Builder builder = new Builder(mContext);  builder.setTitle("正在下載檔案,請稍候...");  RelativeLayout container = new RelativeLayout(mContext);  mProgress = new ProgressBar(mContext);  mProgress.setId("progress".hashCode());  BeanUtils.setFieldValue(mProgress, "mOnlyIndeterminate", Boolean.valueOf(false));  mProgress.setIndeterminate(false);  LayerDrawable layerDrawable = (LayerDrawable)mContext.getResources().getDrawable(android.R.drawable.progress_horizontal);  ClipDrawable clipDrawable = (ClipDrawable)layerDrawable.getDrawable(2);  clipDrawable.setColorFilter(Color.parseColor("#32B5E5"), Mode.SRC_IN);  mProgress.setProgressDrawable(layerDrawable);  mProgress.setPadding(0, 0, 0, 0);  mProgress.setIndeterminateDrawable(    mContext.getResources().getDrawable(android.R.drawable.progress_indeterminate_horizontal));  mProgressPercent = new TextView(mContext);  mProgressPercent.setId("percent".hashCode());  mProgressPercent.setText("0%");  mProgressPercent.setTextSize(18);    int containerPadding = DimensionUtils.dip2Px(mContext, 10);  container.setPadding(containerPadding, containerPadding, containerPadding, containerPadding);    LayoutParams progressLayoutParams = new LayoutParams(    LayoutParams.MATCH_PARENT, DimensionUtils.dip2Px(mContext, 4));  progressLayoutParams.addRule(RelativeLayout.CENTER_VERTICAL);  progressLayoutParams.addRule(RelativeLayout.LEFT_OF, mProgressPercent.getId());  mProgress.setLayoutParams(progressLayoutParams);    LayoutParams percentLayoutParams = new LayoutParams(    LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);  percentLayoutParams.addRule(RelativeLayout.ALIGN_PARENT_RIGHT);  percentLayoutParams.addRule(RelativeLayout.CENTER_VERTICAL);  mProgressPercent.setLayoutParams(percentLayoutParams);    container.addView(mProgressPercent);  container.addView(mProgress);  builder.setView(container);  builder.setNegativeButton("取消", new OnClickListener() {    @Override   public void onClick(DialogInterface dialog, int which) {    dialog.dismiss();   }  });    progressDialog = builder.create();  return progressDialog; }  class DownLoadReceiver extends BroadcastReceiver{  @Override  public void onReceive(Context context, Intent intent) {   //顯示進度條   fileLength = intent.getLongExtra("FILE_LENGTH", -1);   if(showDialog){    mProgress.setMax((int)fileLength);   }   //更新進度條的線程   new Thread(){    @Override    public void run() {     super.run();     while(true){      try{       Thread.sleep(500);      }catch (Exception e) {       e.printStackTrace();      }      mHandler.sendMessage(mHandler.obtainMessage());      //擷取下載檔案的大小      int loadedSize = (int)localFile.length();      if(loadedSize >= fileLength){       break;      }     }    }   }.start();  } }  public DownLoadReceiver getReciver() {  return receiver; }}

它的邏輯:

建立一個FileDownloadList對象後,就可以直接使用該下述方法來實現下載功能。

downloadFile(String fileName, String saveDir, String saveName, Runnable callback)

在實現上是這麼個意思:

1.在當前上下文,開啟下載線程。當擷取到要下載的檔案的大小時,發送一個廣播過來(這部分沒有展示在上述代碼中)。

2.在當前上下文中,註冊一個廣播監聽器,監聽廣播標識為SAVE_DOWNLOAD_FILE的廣播。首次監聽到發出來的廣播後,首次發送過來的廣播,包含了要下載的檔案的大小資訊,然後就每隔5毫秒檢測本地檔案的大小,直到本地檔案的大小(loadedSize)大於等於要下載的檔案(fileLength)大小時,退出該迴圈。

在不斷檢測的過程中,通過mHandler.sendMessage(mHandler.obtainMessage()); ,讓UI線程更新進度條。

下載線程,會不斷將伺服器返回的資料流,寫到本地檔案中,所以,本地檔案的大小會不斷變化,直到,它的大小跟要下載的檔案的大小相等時,就退出這個不斷檢測本地檔案大小的線程。

其它沒有在上述代碼中表現出來的內容(在其它部分的代碼中):

1.在invokeFile( final BaseRequestLims request)方法中,開了一個如下的下載線程.該下載線程,會將伺服器返回的檔案流,寫到本地檔案(localFile)中;然後,它還會發送一個標識為SAVE_DOWNLOAD廣播,包含的資訊有要下載檔案的檔案大小fileLength。

//下載的子線程   new Thread(){    @Override    public void run() {     super.run();     HttpHelper.invoke(request);    }   }.start();

上述代碼存在的問題:

1.上下文,使用的是某個Activity,如果發生系統調用了該Activity的onDestroy()時,下載線程還沒有完成,也就意味著,loadedSize的大小還是小於fileLength。從而,那個不斷檢測本地檔案大小的線程就一直在執行著。

即是檢測本地檔案大小的線程和下載線程還在執行著:

檢測本地檔案大小的線程:

new Thread(){    @Override    public void run() {     super.run();     while(true){      try{       Thread.sleep(500);      }catch (Exception e) {       e.printStackTrace();      }      mHandler.sendMessage(mHandler.obtainMessage());      //擷取下載檔案的大小      int loadedSize = (int)localFile.length();      if(loadedSize >= fileLength){       break;      }     }    }   }.start();

下載線程:

new Thread(){        @Override        public void run() {          super.run();          HttpHelper.invoke(request);        }      }.start();

那麼,會出現什麼問題呢?

1).我可以確定的就是,mContext會出現泄漏。

2). DownLoadReceiver不能正常被取消註冊。

分析,待續。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.