使用HttpURLConnection下载文件时出现 java.io.FileNotFoundException彻底解决办法

简介:

使用HttpURLConnection下载文件时经常会出现 java.io.FileNotFoundException文件找不到异常,下面介绍下解决办法

首先设置tomcat对get数据的编码:conf/server.xml

    <Connector port="8080" protocol="HTTP/1.1" 
               connectionTimeout="20000" 
               redirectPort="8443"
			   URIEncoding="UTF-8" />

其次对请求的文件名进行编码:

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLEncoder;

/**
 * 多线程下载
 * @author bing 
 * 
 */
public class OmbDownloadOfThreadsUtil {

	private String urlPath ; // 资源网络路径
	private String targetFilePath ; // 所下载文件的保存路径
	private int threadNum ; // 启用多少条线程进行下载
	// 用于下载线程对象集合
	private DownloadThread[] downloadThreads ;
	// 要下载文件的大小
	private int fileSize ;
	
	
	public OmbDownloadOfThreadsUtil(String urlPath, String targetFilePath,
			int threadNum) {
		this.urlPath = urlPath;
		this.targetFilePath = targetFilePath;
		this.threadNum = threadNum;
		downloadThreads = new DownloadThread[threadNum] ;
	}

	public void downloadFile() throws Exception{
		URL url = new URL(urlPath) ;
		HttpURLConnection conn = (HttpURLConnection) url.openConnection() ;
		conn.setConnectTimeout(4*1000) ;
		conn.setRequestMethod("GET") ;
		conn.setRequestProperty(
				"Accept",
				"image/gif, image/jpeg, image/pjpeg, image/pjpeg, " +
				"application/x-shockwave-flash, application/xaml+xml, " +
				"application/vnd.ms-xpsdocument, application/x-ms-xbap, " +
				"application/x-ms-application, application/vnd.ms-excel, " +
				"application/vnd.ms-powerpoint, application/msword, */*");
		conn.setRequestProperty("Accept-Language", "zh-CN");
		conn.setRequestProperty("Charset", "UTF-8");
		//设置浏览器类型和版本、操作系统,使用语言等信息
		conn.setRequestProperty(
				"User-Agent",
				"Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.2; Trident/4.0; " +
				".NET CLR 1.1.4322; .NET CLR 2.0.50727; .NET CLR 3.0.04506.30; " +
				".NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");
		//设置为长连接
		conn.setRequestProperty("Connection", "Keep-Alive");
		//得到要下载文件的大小
		fileSize = conn.getContentLength() ;
		
		System.out.println("fileSize:"+fileSize);
		
		//断开连接
		conn.disconnect() ;
		//计算每条线程需要下载的大小
		int preThreadDownloadSize = fileSize/threadNum+1 ;
		System.out.println("preThreadDownloadSize:"+preThreadDownloadSize);
		
		
		RandomAccessFile file = new RandomAccessFile(targetFilePath, "rw") ;
		file.setLength(fileSize) ;
		file.close() ;
		for (int i = 0; i < threadNum; i++) {
			// 计算每条线程下载的起始位置
			int startPos = i*preThreadDownloadSize+1 ;
			RandomAccessFile currentPart = new RandomAccessFile(targetFilePath, "rw") ;
			currentPart.seek(startPos) ;
			downloadThreads[i] = new DownloadThread(startPos,preThreadDownloadSize,currentPart) ;
			new Thread(downloadThreads[i]).start() ;
		}
	}
	
	/**
	 * 获取下载的完成百分比
	 * @return 完成的百分比
	 */
	public double getCompleteRate() {
		// 统计多条线程已经下载的总大小
		int sumSize = 0;
		for (int i = 0; i < threadNum; i++) {
			sumSize += downloadThreads[i].hasReadLength;
		}
		// 返回已经完成的百分比
		return sumSize * 1.0 / fileSize;
	}
	/**
	 * 用于下载的线程
	 * @author bing
	 *
	 */
	private final class DownloadThread implements Runnable{

		private int startPos ;
		private int preThreadDownloadSize ;
		private RandomAccessFile currentPart ;
		//已下载长度
		private int hasReadLength ;
		
		public DownloadThread(int startPos, int preThreadDownloadSize,
				RandomAccessFile currentPart) {
			this.startPos = startPos;
			this.preThreadDownloadSize = preThreadDownloadSize;
			this.currentPart = currentPart;
		}

		@Override
		public void run() {
			InputStream inputStream = null ;
			try{
				URL url = new URL(urlPath) ;
				HttpURLConnection conn = (HttpURLConnection) url.openConnection() ;
				conn.setConnectTimeout(4*1000) ;
				conn.setRequestMethod("GET") ;
				conn.setRequestProperty(
						"Accept",
						"image/gif, image/jpeg, image/pjpeg, image/pjpeg, " +
						"application/x-shockwave-flash, application/xaml+xml, " +
						"application/vnd.ms-xpsdocument, application/x-ms-xbap, " +
						"application/x-ms-application, application/vnd.ms-excel, " +
						"application/vnd.ms-powerpoint, application/msword, */*");
				conn.setRequestProperty("Accept-Language", "zh-CN");
				conn.setRequestProperty("Charset", "UTF-8");
				inputStream = conn.getInputStream() ;
				inputStream.skip(startPos) ;//定位到开始位置
				byte[] buffer = new byte[1024] ;
				int temp = 0 ;
				while(hasReadLength<preThreadDownloadSize
						&&(temp=inputStream.read(buffer))!=-1){
					currentPart.write(buffer,0,temp) ;
					hasReadLength += temp ;
				}
				
			}catch(Exception e){
				e.printStackTrace() ;
			}finally{
				try {
					currentPart.close() ;
				} catch (Exception e) {
					e.printStackTrace();
				}
				try {
					inputStream.close() ;
				} catch (Exception e) {
					e.printStackTrace();
				}
			}

		}
		
	}
	
	public static void main(String[] args) throws Exception {
		String songName = "许嵩 - 半城烟沙.mp3" ;
		songName = URLEncoder.encode(songName,"UTF-8") ;
		String urlPath = "http://172.16.2.50:8080/mp3/"+songName ;
		String targetDir = "E:"+File.separator+songName ;
		OmbDownloadOfThreadsUtil odtu = new OmbDownloadOfThreadsUtil(urlPath,targetDir, 6) ;
		odtu.downloadFile() ;
	}
}
经过以上三步基本上问题已经解决,但如果的文件名含有空格的话还需一步:

URLs是不能包含空格的。URL encoding一般会使用“+”号去替换空格,但后台服务器(我的是Tomcat6.0)又不能把“+”还原为空格,所以导致文件找不到,解决办法:只需把“+”替换为“%20”

	public static void main(String[] args) throws Exception {
		String songName = "许嵩 - 半城烟沙.mp3" ;
		songName = URLEncoder.encode(songName,"UTF-8").replace("+", "%20") ;
		
		String urlPath = "http://172.16.2.50:8080/mp3/"+songName ;
		String targetDir = "E:"+File.separator+songName ;
		
		OmbDownloadOfThreadsUtil odtu = new OmbDownloadOfThreadsUtil(urlPath,targetDir, 6) ;
		odtu.downloadFile() ;
	}




相关文章
|
3天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
14 2
|
6天前
|
存储 缓存 安全
在 Java 编程中,创建临时文件用于存储临时数据或进行临时操作非常常见
在 Java 编程中,创建临时文件用于存储临时数据或进行临时操作非常常见。本文介绍了使用 `File.createTempFile` 方法和自定义创建临时文件的两种方式,详细探讨了它们的使用场景和注意事项,包括数据缓存、文件上传下载和日志记录等。强调了清理临时文件、确保文件名唯一性和合理设置文件权限的重要性。
18 2
|
18天前
|
安全 Java API
深入探索Java网络编程中的HttpURLConnection:从基础到进阶
本文介绍了Java网络编程中HttpURLConnection的高级特性,包括灵活使用不同HTTP方法、处理重定向、管理Cookie、优化安全性以及处理大文件上传和下载。通过解答五个常见问题,帮助开发者提升网络编程的效率和安全性。
|
15天前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
|
17天前
|
存储 Java API
Java实现导出多个excel表打包到zip文件中,供客户端另存为窗口下载
Java实现导出多个excel表打包到zip文件中,供客户端另存为窗口下载
24 4
|
19天前
|
Java 数据格式 索引
使用 Java 字节码工具检查类文件完整性的原理是什么
Java字节码工具通过解析和分析类文件的字节码,检查其结构和内容是否符合Java虚拟机规范,确保类文件的完整性和合法性,防止恶意代码或损坏的类文件影响程序运行。
|
19天前
|
Java API Maven
如何使用 Java 字节码工具检查类文件的完整性
本文介绍如何利用Java字节码工具来检测类文件的完整性和有效性,确保类文件未被篡改或损坏,适用于开发和维护阶段的代码质量控制。
|
22天前
|
Web App开发 Java
使用java操作浏览器的工具selenium-java和webdriver下载地址
【10月更文挑战第12天】Selenium-java依赖包用于自动化Web测试,版本为3.141.59。ChromeDriver和EdgeDriver分别用于控制Chrome和Edge浏览器,需确保版本与浏览器匹配。示例代码展示了如何使用Selenium-java模拟登录CSDN,包括设置驱动路径、添加Cookies和获取页面源码。
|
29天前
|
Java Apache Maven
Java将word文档转换成pdf文件的方法?
【10月更文挑战第13天】Java将word文档转换成pdf文件的方法?
135 1
|
29天前
|
监控 Java
Java定时扫码一个文件夹下的文件,如何保证文件写入完成后才进行处理?
【10月更文挑战第13天】Java定时扫码一个文件夹下的文件,如何保证文件写入完成后才进行处理?
90 1