百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

Java文件上传与下载(java文件上传下载io流)

wptr33 2025-03-19 17:22 19 浏览


1 文件上传

1.1 文件上传入门

1.1.1 实现文件上传条件

1)表单的提交方式必须是POST方式。(才有content-type属性)

2)有文件上传表单,表单中有的选择文件的标签

3)把表单设置为enctype="multipart/form-data",提交的数据不再是key-value对,而是字节数据

<form action="${pageContext.request.contextPath }/UploadDemo1" method="post" enctype="multipart/form-data">

请选择文件: <input type="file" name="img"/><br/>

<input type="submit" value="上传" />

</form>

1.1.2 手动解析上传文件

/**

* 手动处理上传文件的逻辑

* @author APPle

*/

public class UploadDemo1 extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

//得到实体内容数据

InputStream in = request.getInputStream();

BufferedReader br = new BufferedReader(new InputStreamReader(in));


//读取文件的开始符

String startTag = br.readLine();


//读取文件名: Content-Disposition: form-data; name="img"; filename="news.txt"

String line = br.readLine();

String fileName = line.substring(line.lastIndexOf("filename=\"")+10, line.lastIndexOf("\"") );

System.out.println("文件名:"+fileName);


//跳过2行

br.readLine();

br.readLine();


//读取文件的实际内容

String str = null;

BufferedWriter bw = new BufferedWriter(new FileWriter("E:/files/"+fileName));

while((str=br.readLine())!=null){

//读到文件结束符时退出循环

if((startTag+"--").equals(str)){

break;

}


//把内容写出文件中

bw.write(str);

bw.newLine();

bw.flush();

}

//关闭

bw.close();

br.close();

}


public void doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

doGet(request, response);

}

}

1.2 工具实现文件上传

1.2.1 commons-fileupload组件概述

commons-fileupload是Apache组织旗下的开源的文件上传的组件。使用非常简单易用。

1.2.2 核心的API

DiskFileItemFactory类: 用于创建上传对象,设置文件缓存区大小,设置文件缓存目录。

ServletFileUpload类: 用于在Servlet程序中实现文件上传

List list = parseRequest(request): 用于解析请求数据,提取和封装文件信息。

FileItem类: 封装一个文件的所有相关的信息javabean。包含文件名称,文件大小,文件类型, 文件数据内容。

1.2.3 fileUpload开发步骤

1)导入commoms-fileuload的jar包


commons-fileupload-1.2.2.jar 核心包

commons-io-2.1.jar 辅助包

2)编写程序

1.3 实现单文件上传

//1.创建DiskFileItemFactory类

/**

* 参数一: 表示文件缓存区的大小。如果上传的文件没有超过缓存区大小,则文件不缓存;否则缓存文件,缓存到临时目录。(byte)

* 参数二: 表示缓存区的临时目录。

*/

DiskFileItemFactory factory = new DiskFileItemFactory(10*1024,new File("e:/temp/"));


//2.创建ServletFileUpload类

ServletFileUpload upload = new ServletFileUpload(factory);


/**

* 设置文件名的编码

*/

upload.setHeaderEncoding("utf-8");


//3.解析request数据(把每一个文件封装到FileItem对象中,FileItem放入List中)

try {

List list = upload.parseRequest(request);


//取出第一个上传的文件

FileItem file = list.get(0);

//得到文件名(getName())

String fileName = file.getName();

//得到文件大小

long fileSize = file.getSize();

//得到内容类型

String contentType = file.getContentType();

//得到文件数据内容

InputStream in = file.getInputStream();


/**

* 4.把文件数据内容存储到服务器端的硬盘中

*/

FileUtils.copyInputStreamToFile(in, new File("e:/files/"+fileName));


/**

* 5.文件上传完毕,手动清理缓存文件

*/

file.delete();


System.out.println("文件名:"+fileName);

System.out.println("文件大小:"+fileSize);

System.out.println("文件类型:"+contentType);

System.out.println("文件数据内容:"+in);


} catch (FileUploadException e) {

e.printStackTrace();

}

1.4 实现多文件上传

//1.创建DiskFileItemFactory对象

DiskFileItemFactory factory = new DiskFileItemFactory(10*1024, new File("e:/temp/"));

//2.创建ServletFileUpload对象

ServletFileUpload upload = new ServletFileUpload(factory);

//3.设置文件编码

upload.setHeaderEncoding("utf-8");

//4.开始解析文件

try {

List list = upload.parseRequest(request);

if(list!=null){

List ufList = new ArrayList();

//遍历多个文件

for(FileItem file: list){

//取出文件相关信息

String fileName = file.getName();

long fileSize = file.getSize();

String contentType = file.getContentType();

//封装到javabean

UploadFile uf = new UploadFile();

uf.setFileName(fileName);

uf.setFileSize(fileSize);

uf.setFileType(contentType);

//放入list中

ufList.add(uf);


//把文件保存到服务器端的硬盘

FileUtils.copyInputStreamToFile(file.getInputStream(), new File("e:/files/"+fileName));

//删除缓存文件

file.delete();

}

request.setAttribute("ufList", ufList);

request.getRequestDispatcher("/success.jsp").forward(request, response);

}

} catch (FileUploadException e) {

e.printStackTrace();

}

1.5 动态选择多文件上传

<html>

<head>

<title>使用组件实现动态多文件上传</title>

</head>

<body>

<form action="${pageContext.request.contextPath }/UploadDemo3" method="post" enctype="multipart/form-data" name="uploadForm">

<table border="1" width="400px">

<tbody>

<tr id="1">

<td>

请选择文件:

</td>

<td>

<input type="file" name="file"/><input type="button" value="删除" onclick="delItem(1)"/>

</td>

</tr>

</tbody>

<tr>

<td colspan="2"><input type="button" value="添加" onclick="addIten()"/></td>

</tr>

<tr>

<td colspan="2"><input type="button" value="上传" onclick="checkSunbit()"/></td>

</tr>

</table>

</form>


<script type="text/javascript">

var id = 2;

//添加一行

function addIten(){

var trNode = document.createElement("tr");

trNode.setAttribute("id", id);


var tdNode1 = document.createElement("td");

tdNode1.innerHTML = "请选择文件:";


var tdNode2 = document.createElement("td");

var input1 = document.createElement("input");

input1.setAttribute("type", "file");

input1.setAttribute("name", "file");

var input2 = document.createElement("input");

input2.setAttribute("type", "button");

input2.setAttribute("value", "删除");

input2.setAttribute("onclick", "delItem("+id+")");

tdNode2.appendChild(input1);

tdNode2.appendChild(input2);


trNode.appendChild(tdNode1);

trNode.appendChild(tdNode2);


var tbodyNode = document.getElementsByTagName("tbody")[0];

tbodyNode.appendChild(trNode);


id++;

}


//删除一行(根据tr的id值删除)

function delItem(id){

if(id>1){

var trNode = document.getElementById(id);

var tbodyNode = document.getElementsByTagName("tbody")[0];

tbodyNode.removeChild(trNode);

id--;

}

}


//提交并且检查file属性

function checkSunbit(){

//检查file属性是否全部填上

var fileList = document.getElementsByName("file");

for(var i=0;i<fileList.length;i++){

//如果为选择file,则其value值为空

if(fileList[i].value==null || fileList[i].value==""){

alert("请选择第"+(i+1)+"个文件");

return;

}

}

//提交表单

var form = document.forms['uploadForm'];

form.submit();

}

</script>

</body>

</html>

1.6 文件上传的细节

1.6.1 限制文件类型

//得到文件类型

String contentType = file.getContentType();

System.out.println(contentType);

//如果是图片,才可以上传(image/bmpjepgjpggif)

if(!contentType.toLowerCase().matches("image/[a-z]*")){

throw new FileTypeErrorException("文件类型不符合条件!");

}

1.6.2 限制上传文件大小

ServletFileUpload类:

setFileSizeMax:设置单个文件的最大容量。

setSizeMax : 设置所有文件的最大容量。

//4.解析请求

try {

List list = upload.parseRequest(request);

if(list!=null){

for(FileItem file : list){

/**

* 限制文件类型

*/

//得到文件类型

String contentType = file.getContentType();

System.out.println(contentType);

//如果是图片,才可以上传(image/bmpjepgjpggif)

if(!contentType.toLowerCase().matches("image/[a-z]*")){

throw new FileTypeErrorException("文件类型不符合条件!");

}


//保存文件

FileUtils.copyInputStreamToFile(file.getInputStream(), new File("e:/files/"+file.getName()));

//删除缓存文件

file.delete();

}

}

} catch (FileTypeErrorException e) {

//e.printStackTrace();

//处理文件类型错误的异常

request.setAttribute("message", e.getMessage());

request.getRequestDispatcher("/05.upload.jsp").forward(request, response);

return;

} catch (FileSizeLimitExceededException e) {

//e.printStackTrace();

//处理文件超过限制的异常

request.setAttribute("message", "单个文件不能超过1M");

request.getRequestDispatcher("/05.upload.jsp").forward(request, response);

return;

} catch (SizeLimitExceededException e) {

//e.printStackTrace();

//处理文件超过限制的异常

request.setAttribute("message", "所有文件不能超过5M");

request.getRequestDispatcher("/05.upload.jsp").forward(request, response);

return;

} catch (FileUploadException e) {

e.printStackTrace();

}

1.6.3 查看文件上传的进度

ServletFileUpload类:

setProgressListener(进度监听器): 设置文件上传的监听器

1.6.4 处理文件名重复问题

/**

* 解决文件名重复问题:

* 1)日期_时间_随机数.jpg

* 2)使用UUID算法(在一台PC都是唯一的)

*/

String fileName = file.getName();

//得到源文件的后缀名

String supfix = fileName.substring(fileName.lastIndexOf(".")); //.jpg

//使用UUID算法生成随机名称

fileName = UUID.randomUUID().toString()+supfix;

1.6.5 把同一个目录的文件打散到不同的目录下

假如所有文件都放在同一个目录下,不方便管理。

1.6.6 得到普通文本控件内容

//判断该FileItem是否是文件还是普通文本空间

if(file.isFormField()){

//普通文本空间(text/password/checkbox/radio/select/texearea

//得到控件内容


//处理一个普通文本

/*String info = file.getString("utf-8");

System.out.println("描述:"+info);*/


//处理多个普通文本

String fieldName = file.getFieldName();

if("info1".equals(fieldName)){

String info1 = file.getString("utf-8");

System.out.println("描述1:"+info1);

}

if("info2".equals(fieldName)){

String info2 = file.getString("utf-8");

System.out.println("描述2:"+info2);

}

}

2 文件下载

2.1 普通文件下载

使用超链接。缺点:1)暴露文件的路径 2)扩展型和安全性不好

2.2 使用servlet程序下载(推荐)

//得到需要下载的文件

String path = this.getServletContext().getRealPath("/upload/9/1/图片1.png");


File file = new File(path);

//读取服务器本地的文件

FileInputStream in = new FileInputStream(file);


/**

* 处理URL编码问题

*/

String fileName = file.getName();

//对文件名进行URl编码

fileName = URLEncoder.encode(fileName, "utf-8");


//判断不同浏览器

String userAgent = request.getHeader("user-agent");

String filefix = null;

if(userAgent.contains("Trident")){

//IE

filefix = "filename="+fileName;

}else if(userAgent.contains("Firefox")){

//Firefox

filefix = "filename*="+fileName;

}else{

filefix = "filename="+fileName;

}


//告诉浏览器以下载方式打开资源

response.setHeader("Content-Disposition", "attachment;"+filefix);


//把本地文件发送给浏览器

byte[] buf = new byte[1024];

int len = 0;

while( (len=in.read(buf))!=-1 ){

response.getOutputStream().write(buf, 0, len);

}

//关闭

in.close();

相关推荐

oracle数据导入导出_oracle数据导入导出工具

关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...

继续学习Python中的while true/break语句

上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个else解...

python continue和break的区别_python中break语句和continue语句的区别

python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...

简单学Python——关键字6——break和continue

Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...

2-1,0基础学Python之 break退出循环、 continue继续循环 多重循

用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...

Python 中 break 和 continue 傻傻分不清

大家好啊,我是大田。今天分享一下break和continue在代码中的执行效果是什么,进一步区分出二者的区别。一、continue例1:当小明3岁时不打印年龄,其余年龄正常循环打印。可以看...

python中的流程控制语句:continue、break 和 return使用方法

Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...

L017:continue和break - 教程文案

continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...

作为前端开发者,你都经历过怎样的面试?

已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...

面试被问 const 是否不可变?这样回答才显功底

作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...

2023金九银十必看前端面试题!2w字精品!

导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。答案:CSS的盒模型是用于布局和定位元素的概念。它由内容区域...

前端面试总结_前端面试题整理

记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...

由浅入深,66条JavaScript面试知识点(七)

作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录由浅入深,66条JavaScript面试知识点(一)由浅入深,66...

2024前端面试真题之—VUE篇_前端面试题vue2020及答案

添加图片注释,不超过140字(可选)1.vue的生命周期有哪些及每个生命周期做了什么?beforeCreate是newVue()之后触发的第一个钩子,在当前阶段data、methods、com...

今年最常见的前端面试题,你会做几道?

在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...