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

Java文件上传与下载(javaweb文件上传下载)

bigegpt 2025-03-24 14:12 5 浏览


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

相关推荐

得物可观测平台架构升级:基于GreptimeDB的全新监控体系实践

一、摘要在前端可观测分析场景中,需要实时观测并处理多地、多环境的运行情况,以保障Web应用和移动端的可用性与性能。传统方案往往依赖代理Agent→消息队列→流计算引擎→OLAP存储...

warm-flow新春版:网关直连和流程图重构

本期主要解决了网关直连和流程图重构,可以自此之后可支持各种复杂的网关混合、多网关直连使用。-新增Ruoyi-Vue-Plus优秀开源集成案例更新日志[feat]导入、导出和保存等新增json格式支持...

扣子空间体验报告

在数字化时代,智能工具的应用正不断拓展到我们工作和生活的各个角落。从任务规划到项目执行,再到任务管理,作者深入探讨了这款工具在不同场景下的表现和潜力。通过具体的应用实例,文章展示了扣子空间如何帮助用户...

spider-flow:开源的可视化方式定义爬虫方案

spider-flow简介spider-flow是一个爬虫平台,以可视化推拽方式定义爬取流程,无需代码即可实现一个爬虫服务。spider-flow特性支持css选择器、正则提取支持JSON/XML格式...

solon-flow 你好世界!

solon-flow是一个基础级的流处理引擎(可用于业务规则、决策处理、计算编排、流程审批等......)。提供有“开放式”驱动定制支持,像jdbc有mysql或pgsql等驱动,可...

新一代开源爬虫平台:SpiderFlow

SpiderFlow:新一代爬虫平台,以图形化方式定义爬虫流程,不写代码即可完成爬虫。-精选真开源,释放新价值。概览Spider-Flow是一个开源的、面向所有用户的Web端爬虫构建平台,它使用Ja...

通过 SQL 训练机器学习模型的引擎

关注薪资待遇的同学应该知道,机器学习相关的岗位工资普遍偏高啊。同时随着各种通用机器学习框架的出现,机器学习的门槛也在逐渐降低,训练一个简单的机器学习模型变得不那么难。但是不得不承认对于一些数据相关的工...

鼠须管输入法rime for Mac

鼠须管输入法forMac是一款十分新颖的跨平台输入法软件,全名是中州韵输入法引擎,鼠须管输入法mac版不仅仅是一个输入法,而是一个输入法算法框架。Rime的基础架构十分精良,一套算法支持了拼音、...

Go语言 1.20 版本正式发布:新版详细介绍

Go1.20简介最新的Go版本1.20在Go1.19发布六个月后发布。它的大部分更改都在工具链、运行时和库的实现中。一如既往,该版本保持了Go1的兼容性承诺。我们期望几乎所...

iOS 10平台SpriteKit新特性之Tile Maps(上)

简介苹果公司在WWDC2016大会上向人们展示了一大批新的好东西。其中之一就是SpriteKitTileEditor。这款工具易于上手,而且看起来速度特别快。在本教程中,你将了解关于TileE...

程序员简历例句—范例Java、Python、C++模板

个人简介通用简介:有良好的代码风格,通过添加注释提高代码可读性,注重代码质量,研读过XXX,XXX等多个开源项目源码从而学习增强代码的健壮性与扩展性。具备良好的代码编程习惯及文档编写能力,参与多个高...

Telerik UI for iOS Q3 2015正式发布

近日,TelerikUIforiOS正式发布了Q32015。新版本新增对XCode7、Swift2.0和iOS9的支持,同时还新增了对数轴、不连续的日期时间轴等;改进TKDataPoin...

ios使用ijkplayer+nginx进行视频直播

上两节,我们讲到使用nginx和ngixn的rtmp模块搭建直播的服务器,接着我们讲解了在Android使用ijkplayer来作为我们的视频直播播放器,整个过程中,需要注意的就是ijlplayer编...

IOS技术分享|iOS快速生成开发文档(一)

前言对于开发人员而言,文档的作用不言而喻。文档不仅可以提高软件开发效率,还能便于以后的软件开发、使用和维护。本文主要讲述Objective-C快速生成开发文档工具appledoc。简介apple...

macOS下配置VS Code C++开发环境

本文介绍在苹果macOS操作系统下,配置VisualStudioCode的C/C++开发环境的过程,本环境使用Clang/LLVM编译器和调试器。一、前置条件本文默认前置条件是,您的开发设备已...