多執行緒斷點續傳顯示進度條和百分比
簡介
複製即可用使用簡單
這是一個簡單使用的支援暫停和繼續多執行緒下載顯示進度條和百分比進度的多
首先新增依賴
compile 'com.loopj.android:android-async-http:1.4.9'
許可權
記得如果是模擬器的話需要手動設定允許操作記憶體
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES"/>
佈局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.wyj.test.ddxc.LaoshiDDXCActivity">
<ProgressBar
android:id="@+id/pb"
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>
<TextView
android:id="@+id/tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<Button
android:id="@+id/btn_start"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="download"
android:text="開始"/>
<Button
android:id="@+id/btn_stop"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="pause"
android:text="停止"/>
</LinearLayout>
以下時全部程式碼
其中註釋還是挺全的
package com.wyj.test.ddxc;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.os.SystemClock;
import android.support.v7.app.AppCompatActivity;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
import com.loopj.android.http.HttpGet;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.InputStream;
import java.io.RandomAccessFile;
import cz.msebera.android.httpclient.Header;
import cz.msebera.android.httpclient.HttpResponse;
import cz.msebera.android.httpclient.client.HttpClient;
import cz.msebera.android.httpclient.client.methods.HttpHead;
import cz.msebera.android.httpclient.impl.client.DefaultHttpClient;
public class LaoshiDDXCActivity extends AppCompatActivity {
protected static final String TAG = "OtherActivity";
//下載執行緒的數量
private final static int threadsize = 3;
protected static final int SET_MAX = 0;
public static final int UPDATE_VIEW = 1;
private ProgressBar pb;
private Button bt_download;
private Button bt_pause;
private TextView tv_info;
//顯示進度和更新進度
private Handler mHandler = new Handler(){
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case SET_MAX://設定進度條的最大值
int filelength = msg.arg1;
pb.setMax(filelength);
break;
case UPDATE_VIEW://更新進度條 和 下載的比率
int len = msg.arg1;//新下載的長度
pb.setProgress(pb.getProgress()+len);//設定進度條的刻度
// 進度百分比
int max = pb.getMax();//獲取進度的最大值
int progress = pb.getProgress();//獲取已經下載的資料量
// 下載:30 總:100
int result = (progress*100)/max;
tv_info.setText("下載:"+result+"%");
break;
default:
break;
}
};
};
String uri = "http://c.hiphotos.baidu.com/image/pic/item/b90e7bec54e736d1e51217c292504fc2d46269f3.jpg";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_laoshi_ddxc);
//找到控制元件
pb = (ProgressBar) findViewById(R.id.pb);
tv_info = (TextView) findViewById(R.id.tv);
bt_download = (Button) findViewById(R.id.btn_start);
bt_pause = (Button) findViewById(R.id.btn_stop);
//資料的回顯
//確定下載的檔案
String name = getFileName(uri);
File file = new File(Environment.getExternalStorageDirectory(), name);
if (file.exists()){//檔案存在回顯
//獲取檔案的大小
int filelength = (int) file.length();
pb.setMax(filelength);
try {
//統計原來所有的下載量
int count = 0;
//讀取下載記錄檔案
for (int threadid = 0; threadid < threadsize; threadid++) {
//獲取原來指定執行緒的下載記錄
int existDownloadLength = readDownloadInfo(threadid);
count = count + existDownloadLength;
}
//設定進度條的刻度
pb.setProgress(count);
//計算比率
int result = (count * 100) / filelength;
tv_info.setText("下載:" + result + "%");
} catch (Exception e) {
e.printStackTrace();
}
}
}
//暫停
private boolean flag = false;//是否在下載
public void pause(View v){
flag = false;
bt_download.setEnabled(true);
bt_pause.setEnabled(false);
}
//下載
public void download(View v){
flag = true; //是在下載
bt_download.setEnabled(false);//一點選變成不可點選
bt_pause.setEnabled(true);//一點選變成可點選
new Thread(){//子執行緒
public void run() {
try {
//獲取伺服器上檔案的大小
HttpClient client = new DefaultHttpClient();
HttpHead request = new HttpHead(uri);
HttpResponse response = client.execute(request);
//response 只有響應頭 沒有響應體
if(response.getStatusLine().getStatusCode() == 200){
Header[] headers = response.getHeaders("Content-Length");
String value = headers[0].getValue();
//檔案大小
int filelength = Integer.parseInt(value);
Log.i(TAG, "filelength:"+filelength);
//設定進度條的最大值
Message msg_setmax = Message.obtain(mHandler, SET_MAX, filelength, 0);
msg_setmax.sendToTarget();
//處理下載記錄檔案
for(int threadid=0;threadid<threadsize;threadid++){
//對應的下載記錄檔案
File file = new File(Environment.getExternalStorageDirectory(),threadid+".txt");
//判斷檔案是否存在
if(!file.exists()){
//建立檔案
file.createNewFile();
}
}
//在sdcard建立和伺服器大小一樣的檔案
String name = getFileName(uri);
File file = new File(Environment.getExternalStorageDirectory(),name);
//隨機訪問檔案
RandomAccessFile raf = new RandomAccessFile(file, "rwd");
//設定檔案的大小
raf.setLength(filelength);
//關閉
raf.close();
//計算每條執行緒的下載量
int block = (filelength%threadsize == 0)?(filelength/threadsize):(filelength/threadsize+1);
//開啟三條執行緒執行下載
for(int threadid=0;threadid<threadsize;threadid++){
new DownloadThread(threadid, uri, file, block).start();
}
}
} catch (Exception e) {
e.printStackTrace();
}
};
}.start();
}
//執行緒下載類
private class DownloadThread extends Thread{
private int threadid;//執行緒的id
private String uri;//下載的地址
private File file;//下載檔案
private int block;//下載的塊
private int start;
private int end;
public DownloadThread(int threadid, String uri, File file, int block) {
super();
this.threadid = threadid;
this.uri = uri;
this.file = file;
this.block = block;
//計算下載的開始位置和結束位置
start = threadid * block;
end = (threadid + 1)*block -1;
try {
//讀取該條執行緒原來的下載記錄
int existDownloadLength = readDownloadInfo(threadid);
//修改下載的開始位置 從新下載
start = start + existDownloadLength;
} catch (Exception e) {
e.printStackTrace();
}
}
//下載 狀態碼:200是普通的下載 206是分段下載 Range:範圍
@Override
public void run() {
super.run();
try {
RandomAccessFile raf = new RandomAccessFile(file, "rwd");
//跳轉到起始位置
raf.seek(start);
//分段下載
HttpClient client = new DefaultHttpClient();
HttpGet request = new HttpGet(uri);
request.addHeader("Range", "bytes:"+start+"-"+end);//新增請求頭
HttpResponse response = client.execute(request);
if(response.getStatusLine().getStatusCode() == 200){
InputStream inputStream = response.getEntity().getContent();
//把流寫入到檔案
byte[] buffer = new byte[1024];
int len = 0;
while((len = inputStream.read(buffer)) != -1){
//如果暫停下載 點選暫停 false 就直接return 點選下載true接著下載
if(!flag){
return;//標準執行緒結束
}
//寫資料
raf.write(buffer, 0, len);
//讀取原來下載的資料量 這裡讀取是為了更新下載記錄
int existDownloadLength = readDownloadInfo(threadid);//原來下載的資料量
//計算最新的下載
int newDownloadLength = existDownloadLength + len;
//更新下載記錄 從新記錄最新下載位置
updateDownloadInfo(threadid, newDownloadLength);
//更新進度條的顯示 下載的百分比
Message update_msg = Message.obtain(mHandler, UPDATE_VIEW, len, 0);
update_msg.sendToTarget();
//模擬 看到進度條動的效果
SystemClock.sleep(50);
}
inputStream.close();
raf.close();
Log.i(TAG, "第"+threadid+"條執行緒下載完成");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
/**
* 讀取指定執行緒的下載資料量
* @param threadid 執行緒的id
* @return
* @throws Exception
*/
public int readDownloadInfo(int threadid) throws Exception{
//下載記錄檔案
File file = new File(Environment.getExternalStorageDirectory(),threadid+".txt");
BufferedReader br = new BufferedReader(new FileReader(file));
//讀取一行資料
String content = br.readLine();
int downlength = 0;
//如果該檔案第一次建立去執行讀取操作 檔案裡面的內容是 null
if(!TextUtils.isEmpty(content)){
downlength = Integer.parseInt(content);
}
//關閉流
br.close();
return downlength;
}
/**
* 更新下載記錄
* @param threadid
* @param newDownloadLength
*/
public void updateDownloadInfo(int threadid,int newDownloadLength) throws Exception{
//下載記錄檔案
File file = new File(Environment.getExternalStorageDirectory(),threadid+".txt");
FileWriter fw = new FileWriter(file);
fw.write(newDownloadLength+"");
fw.close();
}
/**
* 獲取檔案的名稱
* @param uri
* @return
*/
private String getFileName(String uri){
return uri.substring(uri.lastIndexOf("/")+1);
}
}
相關推薦
多執行緒斷點續傳顯示進度條和百分比
簡介 複製即可用使用簡單 這是一個簡單使用的支援暫停和繼續多執行緒下載顯示進度條和百分比進度的多 首先新增依賴 compile 'com.loopj.android:android-async-http:1.4.9' 許可權 記得如果是模擬器的話
XUtils開源框架的使用(HttpUtils支援多執行緒斷點續傳)
XUtils專案下載地址:https://github.com/wyouflf/xUtils •XUtils中包含的四大模組: 1、DbUtils模組 2、ViewUtils模組 3、HttpUtils模組: ◦支援同步,非同步方式的請求; ◦支援大檔案上傳,上傳大檔案不會oom
多執行緒斷點續傳(二)
一、 學習內容 1、 多檔案下載列表的顯示 2、 啟動多個執行緒分段下載 二、 多執行緒下載原理簡介 假設要分3個執行緒下載一個100位元組的檔案:從頭到尾,每個執行緒下載一段 三、 學習點 1、 Adapter的getCount() 2、 A
多執行緒斷點續傳(一)
一、 學習內容 1、 基本UI定義 2、 資料庫的操作 3、 Service的啟動 4、 Activity給service傳遞引數 5、 使用廣播回傳資料到Activity 6、 執行緒和Handler 7、 網路操作:檔案的寫入,網路往本地磁碟寫入 二、 網路下
OkHttp實現多執行緒斷點續傳下載,單例模式下多工下載管理器,一起拋掉sp,sqlite的輔助吧
最近專案需要使用到斷點下載功能,筆者比較喜歡折騰,想方設法拋棄SharedPreferences,尤其是sqlite作記錄輔助,改用臨時記錄檔案的形式記錄下載進度,本文以斷點下載為例。先看看demo執行效果圖: 斷點續傳:記
Android FTP 多執行緒斷點續傳下載\上傳
最近在給我的開源下載框架Aria增加FTP斷點續傳下載和上傳功能,在此過程中,爬了FTP的不少坑,終於將功能實現了,在此把一些核心功能點記錄下載。 FTP下載原理 FTP單執行緒斷點續傳 FTP和傳統的HTTP協議有所不同,由於FTP沒有所謂的標頭
C#實現http多執行緒斷點續傳下載檔案
using System; using System.Collections.Generic; using System.IO; using System.Threading; using System.Net; using u8 = System.Byte; using
Java多執行緒斷點續傳下載
package com.example.threadpool; import java.io.File; import java.io.FileInputStream; import java.io.InputStream; import java.io.RandomAc
java實現FTP多執行緒斷點續傳,上傳下載!
package com.ftp; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io
android 多執行緒斷點續傳下載 三
今天跟大家一起分享下android開發中比較難的一個環節,可能很多人看到這個標題就會感覺頭很大,的確如果沒有良好的編碼能力和邏輯思維,這塊是很難搞明白的,前面2次總結中已經為大家分享過有關技術的一些基本要領,我們先一起簡單回顧下它的基本原理。什麼是多執行緒下載?多執行緒下載其
多工多執行緒斷點續傳框架-總序
功能點: 1、支援多工下載 2、支援多執行緒斷點下載 3、下載進度更新 | 暫停 | 取消 | 恢復 4、支援後臺下載,notification更新 5、網路切換智慧判斷 6、下載佇列優先順序 提煉技術點: 1、斷點續傳 Range (Ran
【Android開發經驗】關於“多執行緒斷點續傳下載”功能的一個簡單實現和講解
上班第一天,在技術群裡面和大家閒扯,無意中談到了關於框架的使用,一個同學說為了用xUtils的斷線續傳下載功能,把整個庫引入到了專案中,在google的官方建議中,是非常不建議這種做法的,集合框架雖然把很多功能整合起來,但是程式碼越多,出現問題的可能越大,而且無形之中
輕量級多執行緒斷點續傳下載框架
我又來了,一個月寫了三個小框架我也是屌屌的。 一般的小專案,遇到下載的問題時都是簡單的開一個執行緒然後通過流的方式來實現。少量的下載,檔案也比較小的的時候,這樣的方式都是OK的。但是如果真要做一款下載為主要功能的app的時候,或者專案中涉及大量下載任務的時候,
Android實現網路多執行緒斷點續傳下載
本示例介紹在Android平臺下通過HTTP協議實現斷點續傳下載。 我們編寫的是Andorid的HTTP協議多執行緒斷點下載應用程式。直接使用單執行緒下載HTTP檔案對我們來說是一件非常簡單的事。那麼,多執行緒斷點需要什麼功能? 1.多執行緒下載,
android多執行緒斷點續傳原理解析
在下載大檔案的時候,我們往往要使用多執行緒斷點續傳,保證資料的完整性下面我來解析一下多執行緒斷點續傳的原理首先說多執行緒,我們要多執行緒下載一個大檔案,就有開啟多個執行緒,多個connection,既然是一個檔案分開幾個執行緒來下載,那肯定就是一個執行緒下載一個部分,不能重複那麼我們這麼確定一個執行緒下載一部
擼了個多執行緒斷點續傳下載器,我從中學習到了這些知識
> 文章已經收錄在 [Github.com/niumoo/JavaNotes](https://github.com/niumoo/JavaNotes) ,更有 Java 程式設計師所需要掌握的核心知識,歡迎Star和指教。 > 歡迎關注我的[公眾號](https://github.com/n
圖片上傳顯示進度條和預覽圖的前端實現之進度條篇
很久沒有寫部落格了,是最近忙著找實現,另外工作也很忙,閒下來整理一下最近做的東西,覺得還是有點可寫的。 不知道有沒有朋友在工作中碰到這樣的需求,需要實現一個元件,能實現圖片上傳功能,同時,在圖片未上傳完成時,要顯示進度條和相應的圖片預覽圖。 如圖所示,上傳過
Android 多線程斷點續傳同時下載多個大文件
總結 orm acc ast mil view inpu lib 設置 最近學習在Android環境中一些網絡請求方面的知識,其中有一部分是關於網絡下載方面的知識。在這裏解析一下自己寫的demo,總結一下自己所學的知識。下圖為demo的效果圖,仿照一些應用下載商城在List
android多執行緒斷點下載
多執行緒斷電xia下載,通過設定執行緒的數量,動態獲取下載檔案執行緒的個數,這是本人用於練習所寫demo,註釋很詳細,用於初學者參考使用。 MainActivity.java頁面 package com.dahui.download; import java.io.Buf
ios開發網路-大檔案的多執行緒斷點下載
說明:本文介紹多執行緒斷點續傳。專案中使用了蘋果自帶的類,實現了同時開啟多條執行緒下載一個較大的檔案。 因為實現過程較為複雜, 實現思路:下載 開始,建立一個與要下載檔案大小相同的檔案(如果要下載100M,那麼就在沙盒建立一個100M的檔案,然後計算每一段的下載量,開啟多條執行緒下載各段的資料,分