Android 实现断点续传

来自:http://www.2cto.com/kf/201401/269891.html


断点续传:

断点续传指的是在 下载 或上传时,将下载或上传任务(一个文件或一个压缩包)人为的划分为几个部分,每一个部分采用一个线程进行上传或下载,如果碰到网络故障,可以从已经上传或下载的部分开始继续上传下载以后未上传下载的部分,而没有必要从头开始上传下载。用途可以节省时间,提高速度。
测试服务器是否支持断点续传的方法:

浏览器测试:

看看某个请求的响应头信息里面是否包含:
Accept-Ranges bytes
比如请求hao123,相应头信息里面包含:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
Accept-Ranges bytes
Cache-Control max-age= 0
Content-Encoding gzip
Content-Length 59107
Content-Type text/html;charset=UTF- 8
Date Fri, 03 Jan 2014 02 : 40 : 27 GMT
Etag "1946829081"
Expires Fri, 03 Jan 2014 02 : 40 : 27 GMT
LFY cq01. 13
Last-Modified Fri, 03 Jan 2014 02 : 40 : 00 GMT
SFY cq01. 14
Server BWS/ 1.0
Set-Cookie hz= 0 ; path=/; domain=www.hao123.com
Vary Accept-Encoding

表明支持断点续传

curl测试:

curl –-range 100-200 http://192.168.9.155/johnny/testzip.zip > download_1
看下载文件的大小是否是100字节。
如果服务器支持断点续传,那么在请求头信息里面加上属性就可以了:
RANGE: bytes=1000-

这一行就是告诉服务器这个文件从1000字节开始传,前面就不要传了。

Java代码实现:

DownLoaderTask.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
package com.johnny.testdownloadbreakpoint;
 
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
 
import android.app.AlertDialog;
import android.app.AlertDialog.Builder;
import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.DialogInterface.OnCancelListener;
import android.content.DialogInterface.OnClickListener;
import android.os.AsyncTask;
import android.util.Log;
 
public class DownLoaderTask extends AsyncTask< void , integer,= "" long = "" > {
     private final String TAG = "DownLoaderTask" ;
     private URL mUrl;
     private File mFile;
     private ProgressDialog mDialog;
     private int mProgress = 0 ;
     private ProgressReportingOutputStream mOutputStream;
     private Context mContext;
     public DownLoaderTask(String url,String out,Context context){
         super ();
         if (context!= null ){
             mDialog = new ProgressDialog(context);
             mContext = context;
         }
         else {
             mDialog = null ;
         }
         
         try {
             mUrl = new URL(url);
             String fileName = new File(mUrl.getFile()).getName();
             mFile = new File(out, fileName);
             Log.d(TAG, "out=" +out+ ", name=" +fileName+ ",mUrl.getFile()=" +mUrl.getFile());
         } catch (MalformedURLException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         
     }
     
     @Override
     protected void onPreExecute() {
         // TODO Auto-generated method stub
         //super.onPreExecute();
         if (mDialog!= null ){
             mDialog.setTitle( "Downloading..." );
             mDialog.setMessage(mFile.getName());
             mDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
             mDialog.setOnCancelListener( new OnCancelListener() {
                 
                 @Override
                 public void onCancel(DialogInterface dialog) {
                     // TODO Auto-generated method stub
                     cancel( true );
                 }
             });
             mDialog.show();
         }
     }
 
     @Override
     protected Long doInBackground(Void... params) {
         // TODO Auto-generated method stub
         if (!checkURL()){
             Log.e(TAG, "url can not access....." );
             return 0l;
         }
         return download();
     }
 
     @Override
     protected void onProgressUpdate(Integer... values) {
         // TODO Auto-generated method stub
         //super.onProgressUpdate(values);
         if (mDialog== null )
             return ;
         if (values.length> 1 ){
             int contentLength = values[ 1 ];
             if (contentLength==- 1 ){
                 mDialog.setIndeterminate( true );
             }
             else {
                 mDialog.setMax(contentLength);
             }
         }
         else {
             mDialog.setProgress(values[ 0 ].intValue());
         }
     }
 
     @Override
     protected void onPostExecute(Long result) {
         // TODO Auto-generated method stub
         //super.onPostExecute(result);
         if (mDialog!= null &&mDialog.isShowing()){
             mDialog.dismiss();
         }
         if (isCancelled())
             return ;
     }
 
     private long download(){
         URLConnection connection = null ;
         int bytesCopied = 0 ;
         try {
             connection = mUrl.openConnection();
             int length = connection.getContentLength();
             if (mFile.exists()&&length == mFile.length()){
                 Log.d(TAG, "file " +mFile.getName()+ " already exits!!" );
                 return 0l;
             }
             else if (mFile.exists()&&length > mFile.length()){
                 Log.d(TAG, "Downdoad from breakpoint : size is = " +mFile.length()+ ", total is " +length);
                 publishProgress( 0 ,length);
                 mProgress=( int ) mFile.length();
                 URLConnection connection1=mUrl.openConnection();
                 connection1.setRequestProperty( "RANGE" , "bytes=" + mFile.length() + "-" );
                 bytesCopied=copyFromBreakPoint(connection1.getInputStream(),mFile);
                 if (bytesCopied!=length&&length!=- 1 ){
                     Log.d(TAG, "Download incomplete bytesCopied=" +mFile.length()+ ", length" +length);
                 }
             }
             else if (!mFile.exists()){
                 mOutputStream = new ProgressReportingOutputStream(mFile);
                 publishProgress( 0 ,length);
                 bytesCopied =copy(connection.getInputStream(),mOutputStream);
                 if (bytesCopied!=length&&length!=- 1 ){
                     Log.d(TAG, "Download incomplete bytesCopied=" +bytesCopied+ ", length" +length);
                 }
                 mOutputStream.close();
             }
         } catch (IOException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         return bytesCopied;
     }
     private int copy(InputStream input, OutputStream output){
         byte [] buffer = new byte [ 1024 * 8 ];
         BufferedInputStream in = new BufferedInputStream(input, 1024 * 8 );
         BufferedOutputStream out  = new BufferedOutputStream(output, 1024 * 8 );
         int count = 0 ,n= 0 ;
         try {
             while ((n=in.read(buffer, 0 , 1024 * 8 ))!=- 1 ){
                 out.write(buffer, 0 , n);
                 count+=n;
                 
                 if (isCancelled()){
                     Log.e(TAG, "stop downloading ....." );
                     //out.flush();
                     //out.close();
                     //in.close();
                     break ;
                 }
                 
             }
             out.flush();
         } catch (IOException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         } finally {
             try {
                 out.close();
             } catch (IOException e) {
                 // TODO Auto-generated catch block
                 e.printStackTrace();
             }
             try {
                 in.close();
             } catch (IOException e) {
                 // TODO Auto-generated catch block
                 e.printStackTrace();
             }
         }
         return count;
     }
     private int copyFromBreakPoint(InputStream input, File output){
         byte [] buffer = new byte [ 1024 * 8 ];
         int count = 0 ,n= 0 ;
         try {
             RandomAccessFile fos = new RandomAccessFile(output, "rw" );
             
             fos.seek(output.length());
             while ( true ){
                 n=input.read(buffer);
                 if (n<= 0 ){
                     break ;
                 }
                 fos.write(buffer, 0 , n);
                 mProgress += n;
                 publishProgress(mProgress);
             }
             input.close();
             fos.close();
             
         } catch (FileNotFoundException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         } catch (IOException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         
         return 0 ;
     }
     private boolean checkURL(){
         boolean ret = false ;
         try {
             HttpURLConnection conn = (HttpURLConnection) mUrl.openConnection();
             if (conn.getResponseCode()== 200 )
                 ret= true ;
         } catch (IOException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         return ret;
     }
     private final class ProgressReportingOutputStream extends FileOutputStream{
 
         public ProgressReportingOutputStream(File file)
                 throws FileNotFoundException {
             super (file);
             // TODO Auto-generated constructor stub
         }
 
         @Override
         public void write( byte [] buffer, int byteOffset, int byteCount)
                 throws IOException {
             // TODO Auto-generated method stub
             super .write(buffer, byteOffset, byteCount);
             mProgress += byteCount;
             publishProgress(mProgress);
         }
         
     }
}</ void ,>

MainActivity.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
package com.johnny.testdownloadbreakpoint;
 
import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
 
public class MainActivity extends Activity {
 
     @Override
     protected void onCreate(Bundle savedInstanceState) {
         super .onCreate(savedInstanceState);
         setContentView(R.layout.activity_main);
         doDownLoadWork();
     }
 
     @Override
     public boolean onCreateOptionsMenu(Menu menu) {
         // Inflate the menu; this adds items to the action bar if it is present.
         getMenuInflater().inflate(R.menu.main, menu);
         return true ;
     }
     private void doDownLoadWork(){
         DownLoaderTask task = new DownLoaderTask( "http://192.168.9.155/johnny/H264+MP3.flv" , "/storage/emulated/legacy/" , this );
         //DownLoaderTask task = new DownLoaderTask("http://192.168.9.155/johnny/test.h264", getCacheDir().getAbsolutePath()+"/", this);
         task.execute();
     }
 
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值