`

断点下载功能

阅读更多
  • 项目中要用到下载功能,发现一个断点下载的文章,还不错,归案备份。

由于Linux操作系统的兴起和Java 语言的日渐成熟,使用Java 语言实现 一个跨平台的、外观一致的下载 工具软件已成为可能。网络蚂蚁是大家非常熟知的下载 工具软件,我采用Java 语言实现 了类似网络蚂蚁的基本功能的软件Jants。本文介绍了一些技术实现 要点。

 

单线程直接获取网络文件

 

单线程直接获取网络文件的关键点是获取网络文件,以确定基本方法的正确性。它的初始代码的内容比较简单,可以利用HTTP的基本知识进行设计。它的基本原理是:连接网络地址,打开连接并获取输入流,从输入流中读取数据。实现 代码(测试过程中使用的)如下:

 

 

int data;//从输入流中获取数据URL url=new URL("http://www.sohu.com");//创建连接的地址HttpURLConnection connection=url.openConnection();//打开连接int responsCode=connection. getResponseCode();    //返回Http的响应状态码InputStream input=connection.getInputStream();    //获取输入流System.out.println("responseCode:"+responseCode);    While((data=input.read())!=-1)    System.out.println(data);//将获取的数据打到屏幕显示出来

 

单线程断点续传

 

大家常常有这样的体会:下载 一个几十兆的文件时突然断线,结果前功尽弃。可以使用断点续传解决这个问题。

 

基本原理

 

这里主要介绍一下断点续传的原理。断点续传的原理很简单,只是在HTTP请求上和一般的下载 有所不同。

 

所谓断点续传,也就是要从文件已经下载 的地方开始继续下载 。所以,在客户端传给Web服务器的时候,要多加一条信息——下载 的起始位置,且服务器返回的HTTP状态代码也从200转变为206。

 

上述要点,可以使用Java 语言中的HttpURLConnection类中的setRequestProperty()方法来设置。

 

断点续传的关键步骤

 

1.实现 提交断点续传下载 的起始位置

 

Java 的Net包中提供了这种功能,代码如下:

 

 

URL url = new URL("http://www.mydomain.com/down.zip ";); HttpURLConnection httpConnection=(HttpURLConnection) url.openConnection();     //设置断点续传的开始位置     httpConnection.setRequestProperty("RANGE","bytes=10000");     //设置请求信息    httpConnection.setRequestProperty("GET","/down.zip HTTP/1.1");    //设置接受信息httpConnection.setRequestProperty("Accept","image/gif,image/x-xbitmap,application/msword,*/*");    //设置连接信息    httpConnection.setRequestProperty("Connection","Keep-Alive");     //获得输入流 InputStream input = httpConnection.getInputStream();

 

从输入流中取出的字节流,就是down.zip文件从10000字节开始的字节流。

 

2.保存获得的字节流到文件中

 

由于文件的下载 涉及到断点续传,因此,在保存文件的时候,需要对文件进行随机读写。特别是在多线程下载 的过程中,需要在写文件之前在文件中定位。

 

Java 的IO包中的RandomAccessFile类可以满足这种设计需求。该类在文件中定位指针时,用到的方法是seek(Long)。

 

操作相当简单。假设从10000字节处开始保存文件,代码如下:

 

 

RandomAccess oSavedFile = new  RandomAccessFile("down.zip","rw");     //创建随机文件    long nPos = 10000;     //定位文件指针到nPos位置     oSavedFile.seek(nPos);     byte[] b = new byte[1024];     int nRead;     //从输入流中读入字节流,然后写到文件中     while((nRead=input.read(b,0,1024)) > 0)     { //input为网络输入流      oSavedFile.write(b,0,nRead);     }

 

3.保存已经下载 的文件的长度值

 

由于在每次断开连接时都要保存已下载 文件的长度,且应进行永久保存,因此将它保存到文件介质中。这里采用的是对象序列化的方法——将要保存的内容序列化到一个临时文件中。代码如下:

 

 

long nStart;   //记录已经下载 的字节数File tempFile=new File(“donwzip.tmp”);    //创建临时文件    FileOutputStream file=new FileOutputStream(tempFile);     //创建文件输出流  ObjectOutputStream serialize=new ObjectOutputStream(file);    //创建文件序列化流    serialize.writeObject(nStart);//序列化    serialize.flush();//刷新序列化流    file.close();//关闭文件输出流    serialize.close();//关闭序列化流

 

多线程的断点续传

 

加入断点续传以后,下载 速度还没能得到提高。为防止已下载 文件数据的丢失,也为提高网络文件的下载 速度,可在其中加入多线程。虽然前两步已经基本实现 ,似乎再加入多线程时应该比较容易,但是并非如此。在多线程的编程过程中,需要考虑到线程的同步与互斥。由于是多线程进行断点续传,还要考虑记录多个断点位置,且记录断点位置时也要考虑同步互斥等问题。所有这些都使得这一步比较复杂。

 

同步的考虑

 

同步的基本思想是避免多个线程访问同一个资源时出现问题。由于多线程对同一个文件资源进行读写,因此,为了避免出现错误,要进行读写控制——即同步。Java 中使用synchronized实现 线程之间的同步。Java 是面向对象的语言,它的资源是以对象的形式表现的。因此,Java 同步机制的作用就是力图避免对“对象”的访问冲突。

 

对需要同步的方法或代码段进行标记以实现 同步,需要用到关键字synchronized。系统使用synchronized关键字声明的方法就是为其设置特殊的标记。这个标记起着信号量的作用,每当调用该方法时,Java 的运行系统都将进行检查,以确认此标记的状态,看相应的代码是否已经被调用执行。如没有执行,系统将把这个内部标记授予调用代码的线程,方法运行结束后,标记被释放。在标记被释放之前,任何其它的对象不得调用此方法。

 

主要的同步代码如下(在下载 数据保存入文件中时使用):

 

 

public synchronized int write(byte[] b,int nStart,int len){                     int n=-1;                try{                        rf.write(b,nStart,len); //调用另一个类的方法,向文件中写入数据                        n=len;                }catch(IOException ioe){                        ioe.printStackTrace();                }                return n;        }

 

在保存已下载 的字节数时,由于多个断点位置在不同的线程中记录,所以必须在所有线程都结束时才能保存。为此,解决的办法是再开出一个线程,用以持续监测是否所有的线程都已经结束。若结束,保存所有的断点位置;否则,继续监测。同样,在文件下载 的线程中,需要设置标志位以记录线程是否结束。基本代码如下(在监测线程中使用):

 

 

stop=false;while(!stop){          if(Utility.bStop[0] && Utility.bStop[1] && Utility.bStop[2] && Utility.bStop[3] && Utility.bStop[4]){            System.out.println("Serialize...");            Utility.serializeOut();            //调用序列化函数以保存断点位置于文件中            javaants.setStopFalse();            javaants.setStartTrue();            stop=true;            }

 

断点数据的记录

 

笔者使用的是一个静态数组以记录断点位置。由于有多个断点位置,可采用一个函数进行统一保存:

 

 

public static void serializeOut(){          try{         File tempFile=new File(getTempFileName()+"."+"tmp");               FileOutputStream file=new FileOutputStream(tempFile);             ObjectOutputStream serialize=new ObjectOutputStream(file);         for(int i=0;i<5;i++){           serialize.writeObject(String.valueOf(nStart[i]));            serialize.writeObject(String.valueOf(nStop[i]));         }         serialize.flush();         file.close();         serialize.close();      }catch(Exception e){               System.out.println(e.toString());          }    }

 

图形界面

 

该文件下载 系统的图形界面与流行下载 软件——网络蚂蚁很相似。图形界面的实现 ,使用了Swing包。

发布

 

使用jar命令将所有的文件下载 系统的.class文件打包为javaants.jar文件,并在其中加入Menifest.mf文件,指定Main-Class。

 

 

打包代码: jar cfv *.* javaants.jar运行代码:javaw -classpath javaants.jar Main

分享到:
评论

相关推荐

    java web 网站,实现网页直接上传文件,强悍支持断点下载功能

    java web 网站,实现网页直接上传文件,强悍支持断点下载功能, 今天公司,做项目的时候突然客户需要,在页面实现上传文件,并且还得支持断点续传的功能,郁闷啊,只得花时间去弄了,一般现在网上说的都是用FTP的...

    Android之实现断点下载功能源码.zip

    在Android开发中,实现断点下载功能是一项常见的需求,它允许用户在任意时刻暂停下载,然后在需要时继续从上次中断的地方开始。断点下载在移动网络环境下特别有用,因为这样可以节省用户的流量并提高用户体验。本文...

    一键封装断点下载功能(参考xUtils)

    断点下载功能是网络编程中的一个重要技术,它允许用户在中断下载后,从上次停止的地方继续下载,而不需要重新开始。这种技术对于处理大文件,尤其是网络不稳定的情况,显得尤为重要。这里我们关注的是如何抽离xUtils...

    c#断点下载功能,实现文件断点续传

    简单的C#现在功能实现。 c#断点下载功能,实现文件断点续传

    IDHTTP断点下载,类似迅雷断点下载

    在这里,我们将深入探讨如何使用IDHTTP组件实现类似迅雷的断点下载功能。 首先,了解IDHTTP组件。IDHTTP是Indy库的一部分,提供了HTTP协议的基本操作,如发送GET和POST请求,处理响应,以及下载和上传文件。要实现...

    Android OkHttp断点下载

    在Android开发中,OkHttp和RxJava的结合可以实现高效的网络请求处理,其中包括断点下载功能。断点下载是指在用户暂停下载后,可以从上次停止的位置继续下载,避免了重新开始下载大文件的不便。本篇文章将深入探讨...

    android通知栏,断点下载

    本文将深入探讨如何在Android应用中实现断点下载功能,并基于提供的GitHub项目"yinglovezhuzhu/AndroidDownloader"进行讲解。该项目的作者提供了一个完整的断点下载框架,我们可以通过学习和修改这个开源代码来实现...

    FTP断点上传与断点下载

    在大型文件传输时,断点上传和断点下载功能显得尤为重要,因为它们允许用户在文件传输中断后从上次停止的位置继续,而无需重新开始整个过程,这大大提高了效率并节省了时间。 **断点上传** 断点上传是指在文件传输...

    retrofit实现多线程断点下载,可暂停,开始

    在Android开发中,为了提供更好的用户体验,我们常常需要实现文件的多线程断点下载功能。这不仅可以提高下载速度,还能允许用户在任何时候暂停或继续下载。本篇将重点讲解如何利用Retrofit库和GreenDao数据库来实现...

    Android文件断点下载

    在Android平台上,文件断点下载是一项重要的功能,尤其对于大文件的下载,它允许用户在任意...在实际项目中,你可以参考此类的代码来理解并实现断点下载功能。记住,良好的异常处理和状态管理是保证下载稳定性的关键。

    易语言HTTP断点下载

    在本主题“易语言HTTP断点下载”中,我们将深入探讨如何使用易语言实现HTTP协议的断点下载功能,以及如何结合进度条展示下载进度。 HTTP是互联网上应用最为广泛的一种网络协议,用于从Web服务器传输超文本到本地...

    iOS AFNetWorking实现文件断点下载

    在本教程中,我们将深入探讨如何使用AFNetworking 3.0实现文件的断点下载功能,这是一种允许用户暂停和恢复下载过程的技术,尤其在处理大文件时非常有用。 首先,了解AFNetworking 3.0的基础知识是必要的。...

    springMvc实现文件断点下载

    在这个特定的项目中,我们关注的是如何使用Spring MVC来实现文件的断点下载功能。断点下载是允许用户在中断下载后从上次停止的地方继续下载的技术,这对于大文件下载尤其有用,因为它可以提高用户体验并减少网络资源...

    OKHttp多线程断点下载

    【OKHttp多线程断点下载】是一种在Android或Java应用中实现高效文件下载的方法,它结合了OKHttp网络库...通过分析和理解这些代码,开发者可以学习如何在实际项目中实现OKHttp多线程断点下载功能,提升应用的下载体验。

    android AsyncTask 断点下载实例

    本实例将探讨如何利用AsyncTask实现断点下载功能,这是一种允许用户在暂停后继续下载大文件的技术,极大地提高了用户体验。 首先,我们需要了解AsyncTask的结构。AsyncTask包含三个泛型参数:Params, Progress, ...

    http断点下载

    这种技术允许用户在任何时间中断下载,并在后续连接时从上次中断的位置继续,而不需要重新开始整个下载过程。这尤其适用于下载大型文件,如软件...在实际开发中,可以参考这些源代码来构建自己的HTTP断点下载功能。

    Android高级应用源码-Android断点下载.zip

    通过分析和理解这个“Android高级应用源码-Android断点下载.zip”的源代码,开发者可以深入学习如何在Android中实现高效、可靠的断点下载功能,提升应用程序的用户体验。同时,这也是提升Android开发技能的一个好...

    Android 多线程断点下载源码demo

    在Android应用开发中,多线程断点下载是一项重要的...这个"Android多线程断点下载源码demo"应该包含以上所有组件和逻辑,通过分析和学习这个源码,开发者可以更好地理解和实现多线程断点下载功能,提高应用的用户体验。

    Java多线程断点下载Sample

    在这个WindowsApplicationTest中,我们可以推测它可能是一个包含Java Swing应用程序的测试项目,用于演示上述的多线程断点下载功能。通过运行这个程序,开发者可以直观地看到多线程下载的效率提升以及断点续传的便利...

Global site tag (gtag.js) - Google Analytics