多线程下载与断点续传

本文介绍了一种基于Android平台的断点续传下载技术实现方法。通过使用多线程进行分段下载,并利用文件记录已下载部分,实现了下载过程中断后可从断点继续的功能。文中详细展示了依赖库引入、所需权限配置、UI布局定义及核心代码逻辑。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

依赖:compile 'com.loopj.android:android-async-http:1.4.9'
权限:
    <uses-permission android:name="android.permission.INTERNET"></uses-permission>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
布局文件:
<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=".MainActivity" >

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="下载地址" />

    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal" >

        <Button
            android:id="@+id/bt_download"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:onClick="download"
            android:text="下载" />

        <Button
            android:id="@+id/bt_pause"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:onClick="pause"
            android:enabled="false"
            android:text="暂停" />
    </LinearLayout>

    <ProgressBar
        android:id="@+id/pb"
        style="@android:style/Widget.ProgressBar.Horizontal"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/tv_info"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="下载:0%" />

</LinearLayout>
/**
 * 1. 类的用途  断点续传
 * 2. @author forever
 * 3. @date 2017/11/8 16:13
 */
public class OtherActivity 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(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_other);
//找到控件
        pb = (ProgressBar) findViewById(R.id.pb);
        tv_info = (TextView) findViewById(R.id.tv_info);
        bt_download = (Button) findViewById(R.id.bt_download);
        bt_pause = (Button) findViewById(R.id.bt_pause);
        //数据的回显
        //确定下载的文件
        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){
                            //如果暂停下载   就直接return
                            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);
        }
}


资源下载链接为: https://pan.quark.cn/s/9e7ef05254f8 行列式是线性代数的核心概念,在求解线性方程组、分析矩阵特性以及几何计算中都极为关键。本教程将讲解如何用C++实现行列式的计算,重点在于如何输出分数形式的结果。 行列式定义如下:对于n阶方阵A=(a_ij),其行列式由主对角线元素的乘积,按行或列的奇偶性赋予正负号后求和得到,记作det(A)。例如,2×2矩阵的行列式为det(A)=a11×a22-a12×a21,而更高阶矩阵的行列式可通过Laplace展开或Sarrus规则递归计算。 在C++中实现行列式计算时,首先需定义矩阵类或结构体,用二维数组存储矩阵元素,并实现初始化、加法、乘法、转置等操作。为支持分数形式输出,需引入分数类,包含分子和分母两个整数,并提供整数、浮点数的转换以及加、减、乘、除等运算。C++中可借助std::pair表示分数,或自定义结构体并重载运算符。 计算行列式的函数实现上,3×3及以下矩阵可直接按定义计算,更大矩阵可采用Laplace展开或高斯 - 约旦消元法。Laplace展开是沿某行或列展开,将矩阵分解为多个小矩阵的行列式乘积,再递归计算。在处理分数输出时,需注意避免无限循环和除零错误,如在分数运算前先约简,确保分子分母互质,且所有计算基于整数进行,最后再转为浮点数,以避免浮点数误差。 为提升代码可读性和可维护性,建议采用面向对象编程,将矩阵类和分数类封装,每个类有明确功能和接口,便于后续扩展如矩阵求逆、计算特征值等功能。 总结C++实现行列式计算的关键步骤:一是定义矩阵类和分数类;二是实现矩阵基本操作;三是设计行列式计算函数;四是用分数类处理精确计算;五是编写测试用例验证程序正确性。通过这些步骤,可构建一个高效准确的行列式计算程序,支持分数形式计算,为C++编程和线性代数应用奠定基础。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值