青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

life02

  C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
  197 隨筆 :: 3 文章 :: 37 評論 :: 0 Trackbacks
http://blog.csdn.net/furongkang/article/details/6838521


111
Android開發(fā)多線程斷點續(xù)傳下載器
分類: Android 
2011-10-01 23:14 931人閱讀 評論(8) 收藏 舉報

使用多線程斷點續(xù)傳下載器在下載的時候多個線程并發(fā)可以占用服務(wù)器端更多資源,從而加快下載速度,在下載過程中記錄每個線程已拷貝數(shù)據(jù)的數(shù)量,如果下載中斷,比如無信號斷線、電量不足等情況下,這就需要使用到斷點續(xù)傳功能,下次啟動時從記錄位置繼續(xù)下載,可避免重復(fù)部分的下載。這里采用數(shù)據(jù)庫來記錄下載的進度。

效果圖

       

 

斷點續(xù)傳

1.斷點續(xù)傳需要在下載過程中記錄每條線程的下載進度

2.每次下載開始之前先讀取數(shù)據(jù)庫,查詢是否有未完成的記錄,有就繼續(xù)下載,沒有則創(chuàng)建新記錄插入數(shù)據(jù)庫

3.在每次向文件中寫入數(shù)據(jù)之后,在數(shù)據(jù)庫中更新下載進度

4.下載完成之后刪除數(shù)據(jù)庫中下載記錄

Handler傳輸數(shù)據(jù)

這個主要用來記錄百分比,每下載一部分數(shù)據(jù)就通知主線程來記錄時間

1.主線程中創(chuàng)建的View只能在主線程中修改,其他線程只能通過和主線程通信,在主線程中改變View數(shù)據(jù)

2.我們使用Handler可以處理這種需求

   主線程中創(chuàng)建Handler,重寫handleMessage()方法

   新線程中使用Handler發(fā)送消息,主線程即可收到消息,并且執(zhí)行handleMessage()方法

動態(tài)生成新View

可實現(xiàn)多任務(wù)下載

1.創(chuàng)建XML文件,將要生成的View配置好

2.獲取系統(tǒng)服務(wù)LayoutInflater,用來生成新的View

   LayoutInflater inflater 
= (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);

3.使用inflate(int resource, ViewGroup root)方法生成新的View

4.調(diào)用當(dāng)前頁面中某個容器的addView,將新創(chuàng)建的View添加進來

示例

進度條樣式 download.xml
[html] view plaincopy

    
<?xml version="1.0" encoding="utf-8"?>  
    
<LinearLayout   
        xmlns:android
="http://schemas.android.com/apk/res/android"  
        android:layout_width
="fill_parent"  
        android:layout_height
="wrap_content"  
        
>  
        
<LinearLayout   
            android:orientation
="vertical"  
            android:layout_width
="fill_parent"  
            android:layout_height
="wrap_content"  
            android:layout_weight
="1"  
            
>  
            
<!--進度條樣式默認為圓形進度條,水平進度條需要配置style屬性,  
            
?android:attr/progressBarStyleHorizontal -->  
            
<ProgressBar  
                android:layout_width
="fill_parent"   
                android:layout_height
="20dp"  
                style
="?android:attr/progressBarStyleHorizontal"  
                
/>  
            
<TextView  
                android:layout_width
="wrap_content"   
                android:layout_height
="wrap_content"  
                android:layout_gravity
="center"  
                android:text
="0%"  
                
/>  
        
</LinearLayout>  
        
<Button  
            android:layout_width
="40dp"  
            android:layout_height
="40dp"  
            android:onClick
="pause"  
            android:text
="||"  
            
/>  
    
</LinearLayout>  

頂部樣式 main.xml
[html] view plaincopy

    
<?xml version="1.0" encoding="utf-8"?>  
    
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
        android:orientation
="vertical"  
        android:layout_width
="fill_parent"  
        android:layout_height
="fill_parent"  
        android:id
="@+id/root"  
        
>  
        
<TextView    
            android:layout_width
="fill_parent"   
            android:layout_height
="wrap_content"   
            android:text
="請輸入下載路徑"  
            
/>  
        
<LinearLayout   
            android:layout_width
="fill_parent"  
            android:layout_height
="wrap_content"  
            android:layout_marginBottom
="30dp"  
            
>  
            
<EditText  
                android:id
="@+id/path"  
                android:layout_width
="fill_parent"   
                android:layout_height
="wrap_content"   
                android:singleLine
="true"  
                android:layout_weight
="1"  
                
/>  
            
<Button  
                android:layout_width
="wrap_content"   
                android:layout_height
="wrap_content"   
                android:text
="下載"  
                android:onClick
="download"  
                
/>  
        
</LinearLayout>  
    
</LinearLayout>  
       

MainActivity.java
[java] view plaincopy

    
public class MainActivity extends Activity {  
        
private LayoutInflater inflater;  
        
private LinearLayout rootLinearLayout;  
        
private EditText pathEditText;  
      
        @Override  
        
public void onCreate(Bundle savedInstanceState) {  
            
super.onCreate(savedInstanceState);  
            setContentView(R.layout.main);  
      
            
//動態(tài)生成新View,獲取系統(tǒng)服務(wù)LayoutInflater,用來生成新的View  
            inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);  
            rootLinearLayout 
= (LinearLayout) findViewById(R.id.root);  
            pathEditText 
= (EditText) findViewById(R.id.path);  
      
            
// 窗體創(chuàng)建之后, 查詢數(shù)據(jù)庫是否有未完成任務(wù), 如果有, 創(chuàng)建進度條等組件, 繼續(xù)下載  
            List<String> list = new InfoDao(this).queryUndone();  
            
for (String path : list)  
                createDownload(path);  
        }  
      
        
/** 
         * 下載按鈕 
         * 
@param view 
         
*/  
        
public void download(View view) {  
            String path 
= "http://192.168.1.199:8080/14_Web/" + pathEditText.getText().toString();  
            createDownload(path);  
        }  
      
        
/** 
         * 動態(tài)生成新View 
         * 初始化表單數(shù)據(jù) 
         * 
@param path 
         
*/  
        
private void createDownload(String path) {  
            
//獲取系統(tǒng)服務(wù)LayoutInflater,用來生成新的View  
            LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);  
            LinearLayout linearLayout 
= (LinearLayout) inflater.inflate(R.layout.download, null);  
              
            LinearLayout childLinearLayout 
= (LinearLayout) linearLayout.getChildAt(0);  
            ProgressBar progressBar 
= (ProgressBar) childLinearLayout.getChildAt(0);  
            TextView textView 
= (TextView) childLinearLayout.getChildAt(1);  
            Button button 
= (Button) linearLayout.getChildAt(1);  
      
            
try {  
                button.setOnClickListener(
new MyListener(progressBar, textView, path));  
                
//調(diào)用當(dāng)前頁面中某個容器的addView,將新創(chuàng)建的View添加進來  
                rootLinearLayout.addView(linearLayout);  
            } 
catch (Exception e) {  
                e.printStackTrace();  
            }  
        }  
      
        
private final class MyListener implements OnClickListener {  
            
private ProgressBar progressBar;  
            
private TextView textView;  
            
private int fileLen;  
            
private Downloader downloader;  
            
private String name;  
              
            
/** 
             * 執(zhí)行下載 
             * 
@param progressBar //進度條 
             * 
@param textView //百分比 
             * 
@param path  //下載文件路徑 
             
*/  
            
public MyListener(ProgressBar progressBar, TextView textView, String path) {  
                
this.progressBar = progressBar;  
                
this.textView = textView;  
                name 
= path.substring(path.lastIndexOf("/"+ 1);  
      
                downloader 
= new Downloader(getApplicationContext(), handler);  
                
try {  
                    downloader.download(path, 
3);  
                } 
catch (Exception e) {  
                    e.printStackTrace();  
                    Toast.makeText(getApplicationContext(), 
"下載過程中出現(xiàn)異常"0).show();  
                    
throw new RuntimeException(e);  
                }  
            }  
              
            
//Handler傳輸數(shù)據(jù)  
            private Handler handler = new Handler() {  
                @Override  
                
public void handleMessage(Message msg) {  
                    
switch (msg.what) {  
                        
case 0:  
                            
//獲取文件的大小  
                            fileLen = msg.getData().getInt("fileLen");  
                            
//設(shè)置進度條最大刻度:setMax()  
                            progressBar.setMax(fileLen);  
                            
break;  
                        
case 1:  
                            
//獲取當(dāng)前下載的總量  
                            int done = msg.getData().getInt("done");  
                            
//當(dāng)前進度的百分比  
                            textView.setText(name + "\t" + done * 100 / fileLen + "%");  
                            
//進度條設(shè)置當(dāng)前進度:setProgress()  
                            progressBar.setProgress(done);  
                            
if (done == fileLen) {  
                                Toast.makeText(getApplicationContext(), name 
+ " 下載完成"0).show();  
                                
//下載完成后退出進度條  
                                rootLinearLayout.removeView((View) progressBar.getParent().getParent());  
                            }  
                            
break;  
                    }  
                }  
            };  
      
            
/** 
             * 暫停和繼續(xù)下載 
             
*/  
            
public void onClick(View v) {  
                Button pauseButton 
= (Button) v;  
                
if ("||".equals(pauseButton.getText())) {  
                    downloader.pause();  
                    pauseButton.setText(
"");  
                } 
else {  
                    downloader.resume();  
                    pauseButton.setText(
"||");  
                }  
            }  
        }  
    }  



Downloader.java
[java] view plaincopy

    
public class Downloader {  
      
        
private int done;  
        
private InfoDao dao;  
        
private int fileLen;  
        
private Handler handler;  
        
private boolean isPause;  
      
        
public Downloader(Context context, Handler handler) {  
            dao 
= new InfoDao(context);  
            
this.handler = handler;  
        }  
        
/** 
         * 多線程下載 
         * 
@param path 下載路徑 
         * 
@param thCount 需要開啟多少個線程 
         * 
@throws Exception 
         
*/  
        
public void download(String path, int thCount) throws Exception {  
            URL url 
= new URL(path);  
            HttpURLConnection conn 
= (HttpURLConnection) url.openConnection();  
            
//設(shè)置超時時間  
            conn.setConnectTimeout(3000);  
            
if (conn.getResponseCode() == 200) {  
                fileLen 
= conn.getContentLength();  
                String name 
= path.substring(path.lastIndexOf("/"+ 1);  
                File file 
= new File(Environment.getExternalStorageDirectory(), name);  
                RandomAccessFile raf 
= new RandomAccessFile(file, "rws");  
                raf.setLength(fileLen);  
                raf.close();  
                  
                
//Handler發(fā)送消息,主線程接收消息,獲取數(shù)據(jù)的長度  
                Message msg = new Message();  
                msg.what 
= 0;  
                msg.getData().putInt(
"fileLen", fileLen);  
                handler.sendMessage(msg);  
                  
                
//計算每個線程下載的字節(jié)數(shù)  
                int partLen = (fileLen + thCount - 1/ thCount;  
                
for (int i = 0; i < thCount; i++)  
                    
new DownloadThread(url, file, partLen, i).start();  
            } 
else {  
                
throw new IllegalArgumentException("404 path: " + path);  
            }  
        }  
      
        
private final class DownloadThread extends Thread {  
            
private URL url;  
            
private File file;  
            
private int partLen;  
            
private int id;  
      
            
public DownloadThread(URL url, File file, int partLen, int id) {  
                
this.url = url;  
                
this.file = file;  
                
this.partLen = partLen;  
                
this.id = id;  
            }  
      
            
/** 
             * 寫入操作 
             
*/  
            
public void run() {  
                
// 判斷上次是否有未完成任務(wù)  
                Info info = dao.query(url.toString(), id);  
                
if (info != null) {  
                    
// 如果有, 讀取當(dāng)前線程已下載量  
                    done += info.getDone();  
                } 
else {  
                    
// 如果沒有, 則創(chuàng)建一個新記錄存入  
                    info = new Info(url.toString(), id, 0);  
                    dao.insert(info);  
                }  
      
                
int start = id * partLen + info.getDone(); // 開始位置 += 已下載量  
                int end = (id + 1* partLen - 1;  
      
                
try {  
                    HttpURLConnection conn 
= (HttpURLConnection) url.openConnection();  
                    conn.setReadTimeout(
3000);  
                    
//獲取指定位置的數(shù)據(jù),Range范圍如果超出服務(wù)器上數(shù)據(jù)范圍, 會以服務(wù)器數(shù)據(jù)末尾為準  
                    conn.setRequestProperty("Range""bytes=" + start + "-" + end);  
                    RandomAccessFile raf 
= new RandomAccessFile(file, "rws");  
                    raf.seek(start);  
                    
//開始讀寫數(shù)據(jù)  
                    InputStream in = conn.getInputStream();  
                    
byte[] buf = new byte[1024 * 10];  
                    
int len;  
                    
while ((len = in.read(buf)) != -1) {  
                        
if (isPause) {  
                            
//使用線程鎖鎖定該線程  
                            synchronized (dao) {  
                                
try {  
                                    dao.wait();  
                                } 
catch (InterruptedException e) {  
                                    e.printStackTrace();  
                                }  
                            }  
                        }  
                        raf.write(buf, 
0, len);  
                        done 
+= len;  
                        info.setDone(info.getDone() 
+ len);  
                        
// 記錄每個線程已下載的數(shù)據(jù)量  
                        dao.update(info);   
                        
//新線程中用Handler發(fā)送消息,主線程接收消息  
                        Message msg = new Message();  
                        msg.what 
= 1;  
                        msg.getData().putInt(
"done", done);  
                        handler.sendMessage(msg);  
                    }  
                    in.close();  
                    raf.close();  
                    
// 刪除下載記錄  
                    dao.deleteAll(info.getPath(), fileLen);   
                } 
catch (IOException e) {  
                    e.printStackTrace();  
                }  
            }  
        }  
      
        
//暫停下載  
        public void pause() {  
            isPause 
= true;  
        }  
        
//繼續(xù)下載  
        public void resume() {  
            isPause 
= false;  
            
//恢復(fù)所有線程  
            synchronized (dao) {  
                dao.notifyAll();  
            }  
        }  
    }  

 

 

Dao:

 

DBOpenHelper:
[java] view plaincopy

    
public class DBOpenHelper extends SQLiteOpenHelper {  
      
        
public DBOpenHelper(Context context) {  
            
super(context, "download.db"null1);  
        }  
      
        @Override  
        
public void onCreate(SQLiteDatabase db) {  
            db.execSQL(
"CREATE TABLE info(path VARCHAR(1024), thid INTEGER, done INTEGER, PRIMARY KEY(path, thid))");  
        }  
      
        @Override  
        
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {  
        }  
      
    }  


InfoDao:
[java] view plaincopy

    
public class InfoDao {  
        
private DBOpenHelper helper;  
      
        
public InfoDao(Context context) {  
            helper 
= new DBOpenHelper(context);  
        }  
      
        
public void insert(Info info) {  
            SQLiteDatabase db 
= helper.getWritableDatabase();  
            db.execSQL(
"INSERT INTO info(path, thid, done) VALUES(?, ?, ?)"new Object[] { info.getPath(), info.getThid(), info.getDone() });  
        }  
      
        
public void delete(String path, int thid) {  
            SQLiteDatabase db 
= helper.getWritableDatabase();  
            db.execSQL(
"DELETE FROM info WHERE path=? AND thid=?"new Object[] { path, thid });  
        }  
      
        
public void update(Info info) {  
            SQLiteDatabase db 
= helper.getWritableDatabase();  
            db.execSQL(
"UPDATE info SET done=? WHERE path=? AND thid=?"new Object[] { info.getDone(), info.getPath(), info.getThid() });  
        }  
      
        
public Info query(String path, int thid) {  
            SQLiteDatabase db 
= helper.getWritableDatabase();  
            Cursor c 
= db.rawQuery("SELECT path, thid, done FROM info WHERE path=? AND thid=?"new String[] { path, String.valueOf(thid) });  
            Info info 
= null;  
            
if (c.moveToNext())  
                info 
= new Info(c.getString(0), c.getInt(1), c.getInt(2));  
            c.close();  
      
            
return info;  
        }  
      
        
public void deleteAll(String path, int len) {  
            SQLiteDatabase db 
= helper.getWritableDatabase();  
            Cursor c 
= db.rawQuery("SELECT SUM(done) FROM info WHERE path=?"new String[] { path });  
            
if (c.moveToNext()) {  
                
int result = c.getInt(0);  
                
if (result == len)  
                    db.execSQL(
"DELETE FROM info WHERE path=? "new Object[] { path });  
            }  
        }  
      
        
public List<String> queryUndone() {  
            SQLiteDatabase db 
= helper.getWritableDatabase();  
            Cursor c 
= db.rawQuery("SELECT DISTINCT path FROM info"null);  
            List
<String> pathList = new ArrayList<String>();  
            
while (c.moveToNext())  
                pathList.add(c.getString(
0));  
            c.close();  
            
return pathList;  
        }  
      
    }  



posted on 2012-04-02 14:50 life02 閱讀(1751) 評論(2)  編輯 收藏 引用 所屬分類: Android開發(fā)

評論

# re: Android開發(fā)多線程斷點續(xù)傳下載器 (轉(zhuǎn)載) 2012-04-02 14:52 life02
http://download.csdn.net/download/suijing/3114607  回復(fù)  更多評論
  

# re: Android開發(fā)多線程斷點續(xù)傳下載器 (轉(zhuǎn)載) 2012-09-13 15:29 25Age
這個handler是不是只能放在外面?
還有哪些地方 。  回復(fù)  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲视频欧美视频| 欧美在线免费视屏| 亚洲美女网站| 亚洲日本精品国产第一区| 亚洲第一久久影院| 亚洲经典视频在线观看| 日韩亚洲欧美在线观看| 99精品久久久| 亚洲特色特黄| 午夜电影亚洲| 久久久精品国产免费观看同学| 久久久噜噜噜久久中文字免| 久久亚洲一区二区| 欧美成人高清| 欧美色欧美亚洲另类二区 | 国内精品国产成人| 国产综合av| 亚洲高清不卡在线| 99视频精品全国免费| 亚洲无限av看| 久久av在线看| 欧美激情国产精品| 亚洲精品一区二区在线观看| 亚洲视频一区二区| 欧美一级片在线播放| 麻豆精品视频| 欧美视频中文字幕| 国产亚洲欧美日韩精品| 亚洲高清不卡av| 亚洲在线日韩| 久久影视精品| 亚洲欧洲精品一区二区精品久久久 | 亚洲欧美在线x视频| 欧美在线三级| 欧美高清一区| 国产精品欧美久久| 亚洲成人资源| 亚洲一区二区三区乱码aⅴ蜜桃女| 午夜一区在线| 欧美激情在线| 亚洲一二三区精品| 免费成人av在线看| 国产精品欧美激情| 亚洲三级视频在线观看| 亚洲午夜极品| 男男成人高潮片免费网站| 99一区二区| 久久久蜜桃精品| 欧美午夜精品久久久久免费视| 国产视频一区二区三区在线观看| 亚洲美女毛片| 久久九九精品99国产精品| 亚洲精品免费电影| 久久久噜噜噜久久中文字幕色伊伊 | 性做久久久久久免费观看欧美| 免费高清在线视频一区·| 国产精品久久久久久亚洲调教| **欧美日韩vr在线| 先锋影音国产精品| 亚洲精品免费看| 久久亚洲视频| 国产视频一区二区在线观看| 亚洲视频一二区| 蘑菇福利视频一区播放| 亚洲欧美激情诱惑| 欧美日韩影院| 亚洲精品男同| 欧美 日韩 国产一区二区在线视频 | 欧美成人精品| 欧美一区国产二区| 国产精品国产三级国产aⅴ无密码| 最新中文字幕一区二区三区| 久久久亚洲精品一区二区三区| 一区二区三区高清| 欧美巨乳在线| 亚洲激情成人在线| 久热精品视频| 久久本道综合色狠狠五月| 国产精品美女久久| 亚洲一区精彩视频| 最新国产拍偷乱拍精品| 欧美a级大片| 在线观看欧美激情| 久久久精品日韩欧美| 亚洲欧美精品suv| 国产精品swag| 亚洲一区日韩| 99精品免费| 欧美日韩美女一区二区| 99精品视频免费观看视频| 欧美电影免费观看网站| 久久精品人人做人人爽电影蜜月| 国产日韩一区欧美| 欧美中文字幕视频| 羞羞色国产精品| 国产日韩欧美91| 欧美在线视频在线播放完整版免费观看| 一区二区冒白浆视频| 欧美三级视频在线| 在线亚洲一区观看| 亚洲美女少妇无套啪啪呻吟| 欧美极品在线视频| 日韩午夜视频在线观看| 亚洲精品一区二区三区99| 欧美日韩成人激情| 亚洲尤物视频在线| 亚洲自拍三区| 国产欧美在线看| 久久久九九九九| 久久亚洲色图| 亚洲精品视频在线| 日韩午夜免费| 国产精品久久久久久久久久尿 | 久久久久久久性| 激情久久综合| 欧美大尺度在线观看| 嫩草成人www欧美| 亚洲毛片av| 99re66热这里只有精品4| 国产精品超碰97尤物18| 欧美一区久久| 欧美在线视频一区二区三区| 亚洲国产mv| 亚洲精品影院| 欧美日韩精品欧美日韩精品| 亚洲在线第一页| 欧美在线视频日韩| 亚洲欧洲在线看| 99精品欧美一区二区三区| 国产欧美日韩综合精品二区| 美日韩精品免费观看视频| 欧美啪啪一区| 欧美怡红院视频| 美女日韩在线中文字幕| 一本色道久久综合狠狠躁篇怎么玩| 亚洲色无码播放| 国产揄拍国内精品对白| 亚洲国产成人久久综合一区| 国产精品麻豆成人av电影艾秋| 篠田优中文在线播放第一区| 久久欧美中文字幕| 在线视频精品一区| 久久成人免费电影| 一区二区三区欧美视频| 午夜精品久久久久影视| 亚洲欧洲日产国码二区| 亚洲欧美成人| 亚洲精品日韩在线观看| 亚洲欧美国产一区二区三区| 91久久精品www人人做人人爽| 在线亚洲美日韩| 亚洲国产欧美精品| 亚洲综合色丁香婷婷六月图片| 亚洲国产精品热久久| 一区二区欧美激情| 亚洲黄色精品| 欧美一区二区三区免费观看视频 | 一区二区毛片| 在线观看不卡av| 亚洲影院免费| 99re热精品| 久久久噜噜噜久久人人看| 亚洲小少妇裸体bbw| 久久综合伊人77777蜜臀| 亚洲欧美中文日韩在线| 蜜臀av一级做a爰片久久| 欧美一级理论性理论a| 欧美黑人多人双交| 久久人人97超碰人人澡爱香蕉 | 亚洲国产人成综合网站| 亚洲欧美一区二区视频| 一本色道久久综合精品竹菊 | 欧美激情成人在线| 国产一区二区三区高清在线观看| 99精品国产热久久91蜜凸| 亚洲国产欧美在线人成| 欧美一区二区三区免费看| 亚洲自拍电影| 欧美精品一区二区三区四区| 美女网站在线免费欧美精品| 国产日本欧美一区二区三区在线| 亚洲久久一区| 亚洲精品一区中文| 老**午夜毛片一区二区三区| 久久久久99| 国产亚洲精品久久久久动| 一区二区三区免费观看| 亚洲视频碰碰| 欧美日韩精品在线观看| 亚洲国产精品成人| 亚洲高清资源综合久久精品| 久久精品99| 久久一二三国产| 国产主播一区二区三区| 午夜精品福利电影| 久久国产精品亚洲77777| 国产欧美精品日韩| 亚洲欧美日韩一区在线| 午夜精品久久久久| 国产精品一二三|