斷點下載#
一、 思路
網絡上的一個文件,我們可以簡單的看成是一串超長字節,那么字節的長度就是文件的大小了。我們要做的就是把這串數據下載到我們本地!最簡單的就是一次性全部下載下來。但是有時候我們要暫停,然后繼續,這該怎么實現呢?其實我們只需要記錄下我們本地已經下載的長度,然后從這個長度開始繼續下載就好了!
效果圖
二、步驟
- 獲取文件的總長度
- 每次寫入文件的時候要記錄已經下載的長度
- 繼續下載的時候,從上次下載的地方繼續下載
三、實現
- 界面布局:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<ProgressBar
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
android:id="@+id/progressBar"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<Button
android:id="@+id/btn_start"
android:text="開始"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<Button
android:id="@+id/btn_stop"
android:text="暫停"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</LinearLayout>
- 新建一個Service,進行下載邏輯處理(別忘了在AndroidManifest中注冊喲):
public class DownloadService extends Service {
private MyBinder binder = null;
private boolean isPause = false;//是否暫停下載
@Nullable
@Override
public IBinder onBind(Intent intent) {
binder = new MyBinder();
return binder;
}
/**
* 自定義Binder
*/
class MyBinder extends Binder{
public void startDownload(){
DownloadService.this.startDownload();
}
public void pause(){
isPause = true;
}
}
/**
* 開始下載
*/
public void startDownload(){
isPause = false;
MyTask task = new MyTask("http://bmob-cdn-8240.b0.upaiyun.com/2017/01/23/5f2ea3409414490aa361c76608e0953e.apk",
Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getPath()+"/myPic.apk");
task.execute();//開始下載
}
private long length;//保存文件的總長度
private long start;//保存已經下載的長度
class MyTask extends AsyncTask<String,Integer,Integer>{
private final Integer PAUSE = -1;
private final Integer DONE = 1;
String str_url;//url地址
String path;//文件路徑+名稱
public MyTask(String url,String path){
str_url = url;
this.path = path;
}
@Override
protected Integer doInBackground(String... params) {
try {
//得到文件的總長度 單位是byte
length = getLength();
URL url = new URL(str_url);
URLConnection conn = url.openConnection();
//設置范圍:從上次下載的地方開始到結束
conn.setRequestProperty("Range", "bytes=" + start + "-" + length);
conn.setConnectTimeout(5*1000);
conn.setReadTimeout(5*1000);
//得到輸入流
InputStream is = conn.getInputStream();
RandomAccessFile file = new RandomAccessFile(path, "rwd");
file.setLength(length);//設置文件的大小
file.seek(start);//設置寫入的起點
byte[] b = new byte[1024];
int len;
while((len = is.read(b))!=-1){
file.write(b,0,len);//寫入文件
start+=len;//更新文件的長度
publishProgress((int)(start*1.0/length*100));//更新下載的進度
if(isPause){
return PAUSE;//用戶點了暫停
}
}
file.close();
is.close();
} catch (Exception e) {
e.printStackTrace();
}
return DONE;
}
/**
* 更新下載的進度
* @param values
*/
@Override
protected void onProgressUpdate(Integer... values) {
EventBus.getDefault().post(values[0]);
}
/**
* 完成
* @param integer
*/
@Override
protected void onPostExecute(Integer integer) {
if(integer == DONE){
Toast.makeText(DownloadService.this, "下載完成", Toast.LENGTH_SHORT).show();
}else {
Toast.makeText(DownloadService.this, "下載暫停", Toast.LENGTH_SHORT).show();
}
}
/**
* 得到文件的總長度
* @return
* @throws IOException
*/
public long getLength() throws IOException {
URL url = new URL(str_url);
URLConnection conn = url.openConnection();
length = conn.getContentLength();//獲得文件大小
return length;
}
}
}
- 在MainActivity中綁定服務,開始下載:
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
DownloadService.MyBinder binder;
ServiceConnection connection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
binder = (DownloadService.MyBinder) service;
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
@Bind(R.id.btn_stop)
Button btn_pause;
@Bind(R.id.btn_start)
Button btn_start;
@Bind(R.id.progressBar)
ProgressBar progressBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
EventBus.getDefault().register(this);
initView();
Intent intent = new Intent(this,DownloadService.class);
startService(intent);
bindService(intent,connection,BIND_AUTO_CREATE);
}
private void initView() {
btn_start.setOnClickListener(this);
btn_pause.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.btn_start:
binder.startDownload();
btn_start.setClickable(false);
btn_pause.setClickable(true);
break;
case R.id.btn_stop:
binder.pause();
btn_start.setText("繼續");
btn_start.setClickable(true);
btn_pause.setClickable(false);
break;
}
}
@Subscribe
public void onProgressUpdate(Integer p){
progressBar.setProgress(p);
}
}
這個小Demo真的很簡單,只要學會獲取網絡文件的長度,以及會使用RandomAccessFile類就可以了。