【Java】Java实现多线程下载


多线程下载原理及步骤:

    1、在本地创建一个大小跟服务器文件相同大小的临时文件。
    2、计算分配几个线程去下载服务器上的资源,知道每个线程下载文件的位置。

    步骤二的具体方法和操作:
    文件的长度/3(线程的个数)=每个线程下载文件的大小
    假设文件长度为10,则
    线程1:0-2
    线程2:3-5
    线程3:6-文件末尾

    每个线程下载的位置的计算方式:
    开始位置:
    (线程id - 1)* 每一块的大小
    结束位置:
    (线程id * 每一块大小)-1

    3、开启多(3)个线程,每一个线程下载对应位置的文件
    4、如果所有的线程都把自己的数据下载完毕了,服务器上的资源就被下载到本地了。


RandomAccessFile 随机文件访问类

      只有RandomAccessFile才有seek搜寻方法,而这个方法也只适用于文件。

      通过seek()方法指定位置,定位文件,即可以定       

      随机写文件的时候从哪个位置开始写。利用这个类才能实现文件的多线程下载。


 基本原理和相关介绍如上,就这些,现在我们看看代码:


package net.loonggg.test;

import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;

/***
 * 多线程下载
 * 
 * @author loonggg
 * 
 */
public class MutilDownloader {
	// 开启的线程的个数
	public static final int THREAD_COUNT = 3;

	public static void main(String[] args) throws Exception {
		String path = "http://down.360safe.com/yunpan/360wangpan_setup.exe";
		// 连接服务器,获取一个文件,获取文件的长度,在本地创建一个大小跟服务器文件大小一样的临时文件
		URL url = new URL(path);
		HttpURLConnection conn = (HttpURLConnection) url.openConnection();
		// 设置网络请求超时时间
		conn.setConnectTimeout(5000);
		// 设置请求方式
		conn.setRequestMethod("GET");
		int code = conn.getResponseCode();
		if (code == 200) {
			// 服务器返回的数据的长度,实际就是文件的长度
			int length = conn.getContentLength();
			System.out.println("----文件总长度----" + length);
			// 在客户端本地创建出来一个大小跟服务器端文件一样大小的临时文件
			RandomAccessFile raf = new RandomAccessFile("yunpan.exe", "rwd");
			// 指定创建的这个文件的长度
			raf.setLength(length);
			// 关闭raf
			raf.close();
			// 假设是3个线程去下载资源
			// 平均每一个线程下载的文件的大小
			int blockSize = length / THREAD_COUNT;
			for (int threadId = 1; threadId <= THREAD_COUNT; threadId++) {
				// 计算每个线程下载的开始位置和结束位置
				int startIndex = (threadId - 1) * blockSize;
				int endIndex = threadId * blockSize - 1;
				if (threadId == THREAD_COUNT) {
					endIndex = length;
				}
				System.out.println("----threadId---" + threadId
						+ "--startIndex--" + startIndex + "--endIndex--"
						+ endIndex);
				// 开启每一个线程
				new DownloadThread(path, threadId, startIndex, endIndex)
						.start();
			}
		}

	}

	/**
	 * 下载文件的子线程,每一个线程下载对应位置的文件
	 * 
	 * @author loonggg
	 * 
	 */
	public static class DownloadThread extends Thread {
		private int threadId;
		private int startIndex;
		private int endIndex;
		private String path;

		/**
		 * @param path
		 *            下载文件在服务器上的路径
		 * @param threadId
		 *            线程id
		 * @param startIndex
		 *            线程下载的开始位置
		 * @param endIndex
		 *            线程下载的结束位置
		 */
		public DownloadThread(String path, int threadId, int startIndex,
				int endIndex) {
			this.path = path;
			this.threadId = threadId;
			this.startIndex = startIndex;
			this.endIndex = endIndex;
		}

		@Override
		public void run() {
			try {
				URL url = new URL(path);
				HttpURLConnection conn = (HttpURLConnection) url
						.openConnection();
				conn.setRequestMethod("GET");
				// 重要:请求服务器下载部分的文件 指定文件的位置
				conn.setRequestProperty("Range", "bytes=" + startIndex + "-"
						+ endIndex);
				conn.setConnectTimeout(5000);
				// 从服务器请求全部资源的状态码200 ok 如果从服务器请求部分资源的状态码206 ok
				int code = conn.getResponseCode();
				System.out.println("---code---" + code);
				InputStream is = conn.getInputStream();// 已经设置了请求的位置,返回的是当前位置对应的文件的输入流
				RandomAccessFile raf = new RandomAccessFile("yunpan.exe", "rwd");
				// 随机写文件的时候从哪个位置开始写
				raf.seek(startIndex);// 定位文件
				int len = 0;
				byte[] buffer = new byte[1024];
				while ((len = is.read(buffer)) != -1) {
					raf.write(buffer, 0, len);
				}
				is.close();
				raf.close();
				System.out.println("线程" + threadId + ":下载完毕了!");
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
	}

}


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值