1、文件的上传介绍
1、要有一个 form 标签,method=post 请求
2、form 标签的 encType 属性值 必须为 multipart/form-data 值
3、在 form标签中使用 input type=file 添加上传的文件
4、编写服务器代码(Servlet 程序)接收,处理上传的数据。
1.1、文件上传时发送的HTTP协议内容
代码实现
upload.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<form action="http://localhost:8080/09_EL_JSTL/uploadServlet" method="post" enctype="multipart/form-data">
用户名: <input type="text" name="username" /><br/>
头像: <input type="file" name="photo" /><br/>
<input type="submit" value="上传">
</form>
</body>
</html>
UploadServlet
public class UploadServlet extends HttpServlet {
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// System.out.println("文件传过来了");
ServletInputStream inputStream = req.getInputStream();
byte[] buffer = new byte[1024000];
int read = inputStream.read(buffer);
System.out.println(new String(buffer,0,read));
}
}
运行结果
------WebKitFormBoundaryD5hVdfZ6PYzGvXJo
Content-Disposition: form-data; name="username"
qwe123
------WebKitFormBoundaryD5hVdfZ6PYzGvXJo
Content-Disposition: form-data; name="photo"; filename="3.png"
Content-Type: image/png
塒NG
?6猟;鋮然z)?1岳獊?d[r╩}?!栀﹑?9[`??诡;?y锘拭
<0靥陛B?;)?%j玖
?+n朗gy?-?薳K?/&謐 魕纂$? 5饜k@咕?U走
憌?!?5?0j鄋`q凍oQ<埧攍?
勅鲮V?
勅? 鱊玪?)虽`A ?i?2{b葇?%! ?2q圃?@僷朮i徏肓?"K?%?s熆YB沤$\鷤$t梄 橒膄7?7'f? gvvv??勁?%1>>U/I?
_5?鍝愞tI?4@Z_>?孠育'嵥0螢2:?.栾镞穙^0L艫|硗诼??B鋧牷'(渓R▕? m?偳_2_紡Ix?'?4?>?
炔u藘2M]??%ζ`詤`4茏铔?2﹩x&$(??$x靖O@胅CL/秽铩C?>LZ{洏娻?S!?>橗bR蓌d?2Wl,#峺?68B>繟>伈?'-b逪.[顠Cq什衅匚"譡/5?&?1A隷?兀 擩# A玶2K8?&x? C巇?V蠐?)2*瞤?8河o"髚|ーJ(AKqi/\葛痏簍墲=Up%6E$昛g?9馪?P?*+z?*H跧@@&A|?!(囏?.軋 ?$? i梞?"'捫#<?黴醒僉悽?,?B鋧0!?)C?;&L詓vr紓]萪?“l[e嫞?+b$佂衰%-髒8`嶡棨 `窩ArH?馄-鋭 nAC>a=xa 笁?M? b傭?&搯?n镐??%`%淋s缥伖g螠羚厼[?"?2狿?4侖p?bJ$ 4銹喪Qu饎硋0&礜ET
(? 嶇CBAl沇瑻?蓏彖眩?=鲐?#G?F?鋥﹟?0? h D??6朅( [盜櫡 梶徏2fJ啜刡4??8B?z賆鋣H個#鋹?.臕(鼇BSP坦嫶扨9j*i倶(?8x)b??4噟?%X食
&(%耞紉袂?㏒б榇2屸?s?BcA4J倆?*絆p?"蘒?h? [?&碂?C鋈噿品DO?B焿贗3G>蛉#?<筇?>?1|a1趑?8L|?戺U?礳KB?^?;鄴L,g?vUL箶Mz?%Gq"q?镄艘!=9?1?拴&bj噣釓$\" b@啯1涿&硱缟嘵Rb耛?鑅(?.9?$櫵02y觩St?4エ偦*頲 厉碂?貪湝$裳?堁-j?,1亗釯L3
?'S伌洹i4禒
事VH任jAuzW?鍼蛭1i?K3!缾嗻YMIB?)M珸?R)?0!G0::诚~gX*j??< 撙x :彷"葠埍L?,匒捗?\\毶6W?奶Cv▅b?nQ?剦^EI4橆噜 蒯?? 琿蹳+j鏡;?惼禷.n8?,?9旹 enA@??U{ )H鄴侫?濺b+n亯 #粶?6剚4T獋膜侢??鳫$骶'J@嚜b駠咢-?0D`?狇膜筀訍&H-饑-乑繷姡芟北1$鋁.剏鑇攨??
{ (忉&剞<丱 挙?8鼔1f蘙棤 媧桍喁?L?咢s╖|(H? ?=虊?郞,蓧?葮 ?Z禗?:電wIu??縹逬聗.礖?H?H"鎎K飓? h?: .佲.卐?憯/>\澍?"襊??>82Llb獱R"y耐_翵軘 朖uIZAAN>鉴鈰'N湢?I/!?
??60?│愴漢燼1?:?L`⒇ ?2{硊?Z埛_\9氦^.}Z(&a?%1乕?蹅脊姪s稥fr @$/?
`?.茝#嘖櫤7~oc{?!S丩贈l?8(
8騤2b\謸樇eF靷?\)W惺曞G*1q dw譣毡+鉟婻S?8Y=?2蝷GC幵b,[c鮆琕岶礑椗q倴倓?+弪押孎?,汙.?!魡D傕m?H藺層婂鳾,?R?&覽棫??奵?(烡0鳵dB?/I? H! 眃&FTO)Od甗-9|->}J懷旧鸾飣镫_?:os? g?" 8僗1 ?尨8忝?饠~!毱%hH纻燛*蕑?(鳀掘暞痿彊OL腬1?0岚-浛 /E妊L j'嚑~!嘆A浓1?擟&魀僗?!(韈(冫
4L7Tb慾,`土嘑胝R绰}?晩慭伋Z紲P捂
?蔭镊??J*f?%艉TA贅2!P?C罗d~$ S雦Y%埍L,焁7s槾餛諍?K?
?-J+M裢??^??緶?;hU?跔V砼鼋 2舂R?
1.2、commons-fileupload.jar 常用 API 介绍说明
commons-fileupload.jar 需要依赖 common-io.jar 这个包,所以两个包都要引入
commons-fileupload.jar 和 common-io.jar 包中,常用的类:
ServletFileUpload 类:用于解析上传的数据。
boolean ServletFileUpload.isMultipartContent(HttpServletRequest request):判断当前上传的数据格式是否是多段的格式
public List parseRequest(HttpServletRequest request):解析上传的数据
boolean FileItem.isFormField():判断当前这个表单项,是否是普通的表单项。还是上传的文件类型
- true 表示普通类型的表单项
- false 表示上传的文件类型
String FileItem.getFieldName():获取表单项的 name 属性值
String FileItem.getString():获取当前表单项值。
String FileItem.getName();:获取上传的文件名。
void FileItem.write(file);:将上传的文件 写到 参数 file 所指向的硬盘位置。
代码实现:
UploadServlet
package com.aiguigu.servlet;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
import javax.servlet.ServletException;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.util.List;
public class UploadServlet extends HttpServlet {
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1.判断上传的数据是否多段数据(只有多端的数据,才是文件上传的)
if (ServletFileUpload.isMultipartContent(req)){
//创建FileItemFatory 工厂实现类
FileItemFactory fileItemFactory = new DiskFileItemFactory();
//创建用于解析上传数据的工具类ServletFileUpload类
ServletFileUpload servletFileUpload = new ServletFileUpload(fileItemFactory);
//解析上传的数据,得到每一个表单项FileItem
try {
List<FileItem> list = servletFileUpload.parseRequest(req);
//循环判断,每一个表单项,是普通类型,还是上传的文件
for(FileItem fileItem : list){
if (fileItem.isFormField()){
//普通表单项
System.out.println("表单项的name属性值:" + fileItem.getFieldName());
//参数UTF-8,解决乱码问题
System.out.println("表单项的value属性值:" + fileItem.getString("UTF-8"));
}else {
//上传文件
System.out.println("表单项的name属性值:" + fileItem.getFieldName());
System.out.println("上传的文件名:" + fileItem.getName());
fileItem.write(new File("d:\\" + fileItem.getName()));
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
upload,jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<form action="http://localhost:8080/09_EL_JSTL/uploadServlet" method="post" enctype="multipart/form-data">
用户名: <input type="text" name="username" /><br/>
头像: <input type="file" name="photo" /><br/>
<input type="submit" value="上传">
</form>
</body>
</html>
运行结果:
点击上传
表单项的name属性值:username
表单项的value属性值:qwe123
表单项的name属性值:photo
上传的文件名:3.png
2、文件的下载
下载的常用 API 说明
response.getOutputStream();
servletContext.getResourceAsStream();
servletContext.getMimeType();
response.setContentType();
response.setHeader(“Content-Disposition”,“attachment;fileName=1.jpg”);
这个响应头告诉浏览器。这是需要下载的。而 attachment 表示附件,也就是下载的一个文件。fileName=后面,表示下载的文件名。
思路:
代码实现
Download
package com.aiguigu.servlet;
import org.apache.commons.io.IOUtils;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URLEncoder;
public class Download extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1.获取要下载的文件名
String downloadFileName = "3.png";
//2.读取要下载的文件内容(通过 ServletContext)
ServletContext servletContext = getServletContext();
//获取要下载的文件类型
String mimeType = servletContext.getMimeType("/file/" + downloadFileName);
System.out.println("下载的文件类型:" + mimeType);
//4.在回传前,通过响应头告诉客户端返回的数据类型
resp.setContentType(mimeType);
//5.告诉客户端收到的数据是用于下载使用(还是使用响应头)
/** Content-Disposition 响应头,表示收到的数据这么处理
attachment 表示附件,表示下载使用
filename= 表示指定下载的文件名
url 编码是把汉字转换成为%xx%xx的格式
*/
resp.setHeader("Content-Disposition","attachment;filename="+ URLEncoder.encode("中国.png","UTF-8") );
/**
* /斜杠被服务器解析表示地址为 http:ip:port/工程名/ 映射到代码的 web目录
*/
InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" + downloadFileName);
//获取响应的输出流
OutputStream outputStream = resp.getOutputStream();
//3.把下载的文件内容回传给客户端
//读取输入流中全部的数据,复制给输出流,输出给客户端
IOUtils.copy(resourceAsStream,outputStream);
}
}
运行结果
完成上面的两个步骤,下载文件是没问题。但是下载的文件是中文名,你会发现,下载无法正确显示出正确的中文名。
原因是在响应头中,不能包含有中文字符,只能包含 ASCII
解决:
resp.setHeader("Content-Disposition","attachment;filename="+ URLEncoder.encode("中国.png","UTF-8") );
url 编码是把汉字转换成为%xx%xx的格式
2.1、使用Base64 解决火狐浏览器附件中文乱码问题
代码解决:
resp.setHeader("Content-Disposition","attachment;filename==?UTF-8?B?"+ Base64.getEncoder().encodeToString("中国.png".getBytes("UTF-8")) + "?=");
2.2、使用 User-Agent 请求头 判断,动态切换不同的方案解决所以浏览器附件中文乱码问题
if (req.getHeader("User-Agent").contains("Firefox")){
//如果是火狐浏览器, 使用Base64编码
resp.setHeader("Content-Disposition","attachment;filename==?UTF-8?B?"+ Base64.getEncoder().encodeToString("中国.png".getBytes("UTF-8")) + "?=");
}else {
//如果不是火狐,是IE或谷歌,使用URL编码操作
resp.setHeader("Content-Disposition","attachment;filename="+ URLEncoder.encode("中国.png","UTF-8") );
}
3、Base64 编码解码的操作
package com.aiguigu.base64;
import java.io.UnsupportedEncodingException;
import java.util.Base64;
public class Base64Test {
public static void main(String[] args) throws UnsupportedEncodingException {
String content = "这是需要base64编码的内容";
//创建一个Base64编码器
Base64.Encoder encoder = Base64.getEncoder();
String result = encoder.encodeToString(content.getBytes("UTF-8"));
System.out.println(result);
Base64.Decoder decoder = Base64.getDecoder();
byte[] bytes = decoder.decode(result);
String str = new String(bytes, "UTF-8");
System.out.println(str);
}
}
运行结果:
6L+Z5piv6ZyA6KaBYmFzZTY057yW56CB55qE5YaF5a65
这是需要base64编码的内容
看尚硅谷JavaWeb视频写的笔记
链接:https://www.bilibili.com/video/BV1Y7411K7zz?p=221