热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Java实现多线程断点下载

这篇文章主要为大家详细介绍了Java实现多线程断点下载的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

JAVA多线程断点下载原理如图:


代码如下:

import java.io.BufferedReader; 
import java.io.File; 
import java.io.FileInputStream; 
import java.io.InputStream; 
import java.io.InputStreamReader; 
import java.io.RandomAccessFile; 
import java.net.HttpURLConnection; 
import java.net.URL; 
 
public class MutileThreadDownload { 
  /** 
   * 线程的数量 
   */ 
  private static int threadCount = 3; 
 
  /** 
   * 每个下载区块的大小 
   */ 
  private static long blocksize; 
 
  /** 
   * 正在运行的线程的数量 
   */ 
  private static int runningThreadCount; 
 
  /** 
   * @param args 
   * @throws Exception 
   */ 
  public static void main(String[] args) throws Exception { 
    // 服务器文件的路径 
    String path = "http://192.168.1.100:8080/ff.exe"; 
    URL url = new URL(path); 
    HttpURLConnection cOnn= (HttpURLConnection) url.openConnection(); 
    conn.setRequestMethod("GET"); 
    conn.setConnectTimeout(5000); 
    int code = conn.getResponseCode(); 
    if (code == 200) { 
      long size = conn.getContentLength();// 得到服务端返回的文件的大小 
      System.out.println("服务器文件的大小:" + size); 
      blocksize = size / threadCount; 
      // 1.首先在本地创建一个大小跟服务器一模一样的空白文件。 
      File file = new File("temp.exe"); 
      RandomAccessFile raf = new RandomAccessFile(file, "rw"); 
      raf.setLength(size); 
      // 2.开启若干个子线程分别去下载对应的资源。 
      runningThreadCount = threadCount; 
      for (int i = 1; i <= threadCount; i++) { 
        long startIndex = (i - 1) * blocksize; 
        long endIndex = i * blocksize - 1; 
        if (i == threadCount) { 
          // 最后一个线程 
          endIndex = size - 1; 
        } 
        System.out.println("开启线程:" + i + "下载的位置:" + startIndex + "~" 
            + endIndex); 
        new DownloadThread(path, i, startIndex, endIndex).start(); 
      } 
    } 
    conn.disconnect(); 
  } 
 
  private static class DownloadThread extends Thread { 
    private int threadId; 
    private long startIndex; 
    private long endIndex; 
    private String path; 
 
    public DownloadThread(String path, int threadId, long startIndex, 
        long endIndex) { 
      this.path = path; 
      this.threadId = threadId; 
      this.startIndex = startIndex; 
      this.endIndex = endIndex; 
    } 
 
    @Override 
    public void run() { 
      try { 
        // 当前线程下载的总大小 
        int total = 0; 
        File positiOnFile= new File(threadId + ".txt"); 
        URL url = new URL(path); 
        HttpURLConnection cOnn= (HttpURLConnection) url 
            .openConnection(); 
        conn.setRequestMethod("GET"); 
        // 接着从上一次的位置继续下载数据 
        if (positionFile.exists() && positionFile.length() > 0) {// 判断是否有记录 
          FileInputStream fis = new FileInputStream(positionFile); 
          BufferedReader br = new BufferedReader( 
              new InputStreamReader(fis)); 
          // 获取当前线程上次下载的总大小是多少 
          String lasttotalstr = br.readLine(); 
          int lastTotal = Integer.valueOf(lasttotalstr); 
          System.out.println("上次线程" + threadId + "下载的总大小:" 
              + lastTotal); 
          startIndex += lastTotal; 
          total += lastTotal;// 加上上次下载的总大小。 
          fis.close(); 
        } 
 
        conn.setRequestProperty("Range", "bytes=" + startIndex + "-" 
            + endIndex); 
        conn.setConnectTimeout(5000); 
        int code = conn.getResponseCode(); 
        System.out.println("code=" + code); 
        InputStream is = conn.getInputStream(); 
        File file = new File("temp.exe"); 
        RandomAccessFile raf = new RandomAccessFile(file, "rw"); 
        // 指定文件开始写的位置。 
        raf.seek(startIndex); 
        System.out.println("第" + threadId + "个线程:写文件的开始位置:" 
            + String.valueOf(startIndex)); 
        int len = 0; 
        byte[] buffer = new byte[512]; 
        while ((len = is.read(buffer)) != -1) { 
          RandomAccessFile rf = new RandomAccessFile(positionFile, 
              "rwd"); 
          raf.write(buffer, 0, len); 
          total += len; 
          rf.write(String.valueOf(total).getBytes()); 
          rf.close(); 
        } 
        is.close(); 
        raf.close(); 
 
      } catch (Exception e) { 
        e.printStackTrace(); 
      } finally { 
        // 只有所有的线程都下载完毕后 才可以删除记录文件。 
        synchronized (MutileThreadDownload.class) { 
          System.out.println("线程" + threadId + "下载完毕了"); 
          runningThreadCount--; 
          if (runningThreadCount <1) { 
            System.out.println("所有的线程都工作完毕了。删除临时记录的文件"); 
            for (int i = 1; i <= threadCount; i++) { 
              File f = new File(i + ".txt"); 
              System.out.println(f.delete()); 
            } 
          } 
        } 
 
      } 
    } 
  } 
} 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 基于PgpoolII的PostgreSQL集群安装与配置教程
    本文介绍了基于PgpoolII的PostgreSQL集群的安装与配置教程。Pgpool-II是一个位于PostgreSQL服务器和PostgreSQL数据库客户端之间的中间件,提供了连接池、复制、负载均衡、缓存、看门狗、限制链接等功能,可以用于搭建高可用的PostgreSQL集群。文章详细介绍了通过yum安装Pgpool-II的步骤,并提供了相关的官方参考地址。 ... [详细]
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • 本文介绍了使用AJAX的POST请求实现数据修改功能的方法。通过ajax-post技术,可以实现在输入某个id后,通过ajax技术调用post.jsp修改具有该id记录的姓名的值。文章还提到了AJAX的概念和作用,以及使用async参数和open()方法的注意事项。同时强调了不推荐使用async=false的情况,并解释了JavaScript等待服务器响应的机制。 ... [详细]
  • PHP设置MySQL字符集的方法及使用mysqli_set_charset函数
    本文介绍了PHP设置MySQL字符集的方法,详细介绍了使用mysqli_set_charset函数来规定与数据库服务器进行数据传送时要使用的字符集。通过示例代码演示了如何设置默认客户端字符集。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • Centos7.6安装Gitlab教程及注意事项
    本文介绍了在Centos7.6系统下安装Gitlab的详细教程,并提供了一些注意事项。教程包括查看系统版本、安装必要的软件包、配置防火墙等步骤。同时,还强调了使用阿里云服务器时的特殊配置需求,以及建议至少4GB的可用RAM来运行GitLab。 ... [详细]
  • 本文介绍了在Hibernate配置lazy=false时无法加载数据的问题,通过采用OpenSessionInView模式和修改数据库服务器版本解决了该问题。详细描述了问题的出现和解决过程,包括运行环境和数据库的配置信息。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • 本文介绍了如何找到并终止在8080端口上运行的进程的方法,通过使用终端命令lsof -i :8080可以获取在该端口上运行的所有进程的输出,并使用kill命令终止指定进程的运行。 ... [详细]
  • 禁止程序接收鼠标事件的工具_VNC Viewer for Mac(远程桌面工具)免费版
    VNCViewerforMac是一款运行在Mac平台上的远程桌面工具,vncviewermac版可以帮助您使用Mac的键盘和鼠标来控制远程计算机,操作简 ... [详细]
  • 本文详细介绍了云服务器API接口的概念和作用,以及如何使用API接口管理云上资源和开发应用程序。通过创建实例API、调整实例配置API、关闭实例API和退还实例API等功能,可以实现云服务器的创建、配置修改和销毁等操作。对于想要学习云服务器API接口的人来说,本文提供了详细的入门指南和使用方法。如果想进一步了解相关知识或阅读更多相关文章,请关注编程笔记行业资讯频道。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • [译]技术公司十年经验的职场生涯回顾
    本文是一位在技术公司工作十年的职场人士对自己职业生涯的总结回顾。她的职业规划与众不同,令人深思又有趣。其中涉及到的内容有机器学习、创新创业以及引用了女性主义者在TED演讲中的部分讲义。文章表达了对职业生涯的愿望和希望,认为人类有能力不断改善自己。 ... [详细]
  • 如何基于ggplot2构建相关系数矩阵热图以及一个友情故事
    本文介绍了如何在rstudio中安装ggplot2,并使用ggplot2构建相关系数矩阵热图。同时,通过一个友情故事,讲述了真爱难觅的故事背后的数据量化和皮尔逊相关系数的概念。故事中的小伙伴们在本科时参加各种考试,其中有些沉迷网络游戏,有些热爱体育,通过他们的故事,展示了不同兴趣和特长对学习和成绩的影响。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
author-avatar
Black
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有