使用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虚拟机--类文件结构
本内容介绍了Java虚拟机与Class文件的关系及其内部结构。Class文件是一种与语言无关的二进制格式,包含JVM指令集、符号表等信息。无论使用何种语言,只要能生成符合规范的Class文件,即可在JVM上运行。文章详细解析了Class文件的组成,包括魔数、版本号、常量池、访问标志、类索引、字段表、方法表和属性表等,并说明其在Java编译与运行过程中的作用。
|
3月前
|
存储 人工智能 Java
java之通过Http下载文件
本文介绍了使用Java实现通过文件链接下载文件到本地的方法,主要涉及URL、HttpURLConnection及输入输出流的操作。
186 0
|
3月前
|
监控 Java API
Java语言按文件创建日期排序及获取最新文件的技术
这段代码实现了文件创建时间的读取、文件列表的获取与排序以及获取最新文件的需求。它具备良好的效率和可读性,对于绝大多数处理文件属性相关的需求来说足够健壮。在实际应用中,根据具体情况,可能还需要进一步处理如访问权限不足、文件系统不支持某些属性等边界情况。
193 14
|
4月前
|
存储 Java 数据安全/隐私保护
Java技术栈揭秘:Base64加密和解密文件的实战案例
以上就是我们今天关于Java实现Base64编码和解码的实战案例介绍。希望能对你有所帮助。还有更多知识等待你去探索和学习,让我们一同努力,继续前行!
327 5
|
4月前
|
网络协议 安全 Java
实现Java语言的文件断点续传功能的技术方案。
像这样,我们就完成了一项看似高科技、实则亲民的小工程。这样的技术实现不仅具备实用性,也能在面对网络不稳定的挑战时,稳稳地、不失乐趣地完成工作。
249 0
|
4月前
|
存储 安全 算法
Java 集合面试题 PDF 下载及高频考点解析
本文围绕Java集合面试题展开,详细解析了集合框架的基本概念、常见集合类的特点与应用场景。内容涵盖`ArrayList`与`LinkedList`的区别、`HashSet`与`TreeSet`的对比、`HashMap`与`ConcurrentHashMap`的线程安全性分析等。通过技术方案与应用实例,帮助读者深入理解集合类的特性和使用场景,提升解决实际开发问题的能力。文末附带资源链接,供进一步学习参考。
99 4
|
7月前
|
前端开发 Cloud Native Java
Java||Springboot读取本地目录的文件和文件结构,读取服务器文档目录数据供前端渲染的API实现
博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
Java||Springboot读取本地目录的文件和文件结构,读取服务器文档目录数据供前端渲染的API实现
|
8月前
|
Java API 数据处理
深潜数据海洋:Java文件读写全面解析与实战指南
通过本文的详细解析与实战示例,您可以系统地掌握Java中各种文件读写操作,从基本的读写到高效的NIO操作,再到文件复制、移动和删除。希望这些内容能够帮助您在实际项目中处理文件数据,提高开发效率和代码质量。
178 4
|
8月前
|
存储 网络协议 Linux
【Linux】进程IO|系统调用|open|write|文件描述符fd|封装|理解一切皆文件
本文详细介绍了Linux中的进程IO与系统调用,包括 `open`、`write`、`read`和 `close`函数及其用法,解释了文件描述符(fd)的概念,并深入探讨了Linux中的“一切皆文件”思想。这种设计极大地简化了系统编程,使得处理不同类型的IO设备变得更加一致和简单。通过本文的学习,您应该能够更好地理解和应用Linux中的进程IO操作,提高系统编程的效率和能力。
313 34
|
8月前
|
Linux 网络安全 Docker
尼恩一键开发环境: vagrant+java+springcloud+redis+zookeeper镜像下载(&制作详解)
尼恩提供了一系列文章,旨在帮助开发者轻松搭建一键开发环境,涵盖Java分布式、高并发场景下的多种技术组件安装与配置。内容包括但不限于Windows和CentOS虚拟机的安装与排坑指南、MySQL、Kafka、Redis、Zookeeper等关键组件在Linux环境下的部署教程,并附带详细的视频指导。此外,还特别介绍了Vagrant这一虚拟环境部署工具,
尼恩一键开发环境: vagrant+java+springcloud+redis+zookeeper镜像下载(&制作详解)

热门文章

最新文章