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

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

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


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();

相关推荐

SQL轻松入门(5):窗口函数(sql语录中加窗口函数的执行)

01前言标题中有2个字让我在初次接触窗口函数时,真真切切明白了何谓”高级”?说来也是一番辛酸史!话说,我见识了窗口函数的强大后,便磨拳擦掌的要试验一番,结果在查询中输入语句,返回的结果却是报错,Wh...

28个SQL常用的DeepSeek提示词指令,码住直接套用

自从DeepSeek出现后,极大地提升了大家平时的工作效率,特别是对于一些想从事数据行业的小白,只需要掌握DeepSeek的提问技巧,SQL相关的问题也不再是个门槛。...

从零开始学SQL进阶,数据分析师必备SQL取数技巧,建议收藏

上一节给大家讲到SQL取数的一些基本内容,包含SQL简单查询与高级查询,需要复习相关知识的同学可以跳转至上一节,本节给大家讲解SQL的进阶应用,在实际过程中用途比较多的子查询与窗口函数,下面一起学习。...

SQL_OVER语法(sql语句over什么含义)

OVER的定义OVER用于为行定义一个窗口,它对一组值进行操作,不需要使用GROUPBY子句对数据进行分组,能够在同一行中同时返回基础行的列和聚合列。...

SQL窗口函数知多少?(sql窗口怎么执行)

我们在日常工作中是否经常会遇到需要排名的情况,比如:每个部门按业绩来排名,每人按绩效排名,对部门销售业绩前N名的进行奖励等。面对这类需求,我们就需要使用sql的高级功能——窗口函数。...

如何学习并掌握 SQL 数据库基础:从零散查表到高效数据提取

无论是职场数据分析、产品运营,还是做副业项目,掌握SQL(StructuredQueryLanguage)意味着你能直接从数据库中提取、分析、整合数据,而不再依赖他人拉数,节省大量沟通成本,让你...

SQL窗口函数(sql窗口函数执行顺序)

背景在数据分析中,经常会遇到按某某条件来排名、并找出排名的前几名,用日常SQL的GROUPBY,ORDERBY来实现特别的麻烦,有时甚至实现不了,这个时候SQL窗口函数就能发挥巨大作用了,窗...

sqlserver删除重复数据只保留一条,使用ROW_NUMER()与Partition By

1.使用场景:公司的小程序需要实现一个功能:在原有小程序上,有一个优惠券活动表。存储着活动产品数据,但因为之前没有做约束,导致数据的不唯一,这会使打开产品详情页时,可能会出现随机显示任意活动问题。...

SQL面试经典问题(一)(sql经典面试题及答案)

以下是三个精心挑选的经典SQL面试问题及其详细解决方案,涵盖了数据分析、排序限制和数据清理等常见场景。这些问题旨在考察SQL的核心技能,适用于初学者到高级开发者的面试准备。每个问题均包含清晰的...

SQL:求连续N天的登陆人员之通用解答

前几天发了一个微头条:...

SQL四大排序函数神技(sql中的排序是什么语句)

在日常SQL开发中,排序操作无处不在。当大家需要排序时,是否只会想到ORDERBY?今天,我们就来揭秘SQL中四个强大却常被忽略的排序函数:ROW_NUMBER()、RANK()、DENSE_RAN...

四、mysql窗口函数之row_number()函数的使用

1、窗口函数之row_number()使用背景窗口函数中,排序函数rank(),dense_rank()虽说都是排序函数,但是各有用处,假如像上章节说的“同组同分”两条数据,我们不想“班级名次”出现“...

ROW_NUMBER()函数(rownumber函数与rank区别)

ROW_NUMBER()是SQL中的一个窗口函数(WindowFunction)...

Dify「模板转换」节点终极指南:动态文本生成进阶技巧(附代码)Jinja2引擎解析

这篇文章是关于Dify「模板转换」节点的终极指南,解析了基于Jinja2模板引擎的动态文本生成技巧,涵盖多源文本整合、知识检索结构化、动态API构建及个性化内容生成等六大应用场景,助力开发者高效利用模...

Python 最常用的语句、函数有哪些?

1.#coding=utf-8①代码中有中文字符,最好在代码前面加#coding=utf-8②pycharm不加可能不会报错,但是代码最终是会放到服务器上,放到服务器上的时候运行可能会报错。③...