目录
1、正常创建一个mavenweb项目
2、导入要用的包,并且add as library
由于是web项目,要写servlet和jsp的代码,也要有servlet的包,通过maven依赖就可以导入
3、index.jsp的代码
4、上传成功后的跳转到的页面
5、servlet代码
package com.zhou.servlet; import org.apache.commons.fileupload.FileItem; import org.apache.commons.fileupload.FileUploadException; import org.apache.commons.fileupload.ProgressListener; import org.apache.commons.fileupload.disk.DiskFileItemFactory; import org.apache.commons.fileupload.servlet.ServletFileUpload; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.*; import java.util.List; import java.util.UUID; public class FileServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 判断上传的文件是普通表单还是带文件的表单 if (!ServletFileUpload.isMultipartContent(req)) { return; //终止方法运行,说明这是一个普通的表单,直接返回 } // 如果通过了这个if说明表单是上传文件的 // 创建上传文件的保存路径,建议在web-inf路径下,安全,用户无法直接访问上传的文件 String uploadPath = this.getServletContext().getRealPath("/WEB-INF/upload"); // 存放的路径 File uploadFile = new File(uploadPath); // new一个文件夹 if (!uploadFile.exists()) { // 如果没有这个目录 uploadFile.mkdir(); // 创建这个目录 } // 缓存,零时文件 // 零时路径,加入文件超过了与其大小,我们就把他放到一个零时文件中,过几天自动删除,或者提醒用户转存为永久 String tmpPath = this.getServletContext().getRealPath("/WEB-INF/tmp"); // 存放的路径 File file = new File(tmpPath); if (!file.exists()) { // 如果没有这个目录 file.mkdir(); // 创建这个零时目录 } // 处理上传的文件,一般都需要通过流来获取,我们可以使用request。getInputStream(),原生态的文件上传流获取,十分麻烦 // 建议使用Apache的文件上传组件来实现,common-fileupload,需要依赖于common-io组件 try { // 1.创建DiskFileItemFactory对象,处理文件上传路径或者大小限制,临时文件 DiskFileItemFactory factory = getDiDiskFileItemFactory(file); // 2. 获取ServletFileUpload ServletFileUpload upload = getServletFileUpload(factory); // 3.处理上传文件 String msg = uploadParseRequest(upload, req, uploadPath); // servlet请求转发消息 req.setAttribute("msg", msg); req.getRequestDispatcher("info.jsp").forward(req, resp); } catch (FileUploadException e) { throw new RuntimeException(e); } } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doGet(req, resp); } public static DiskFileItemFactory getDiDiskFileItemFactory(File file){ DiskFileItemFactory factory = new DiskFileItemFactory(); // 通过这个工厂设置一个缓冲区,当上传的文件大于这个缓冲区的时候,将他放到临时文件中 factory.setSizeThreshold(1024*1024); // 缓冲区大小为1M 可以不写,有默认值 factory.setRepository(file); // 零时目录的保存目录,需要一个file 可以不写,有默认值 return factory; } public static ServletFileUpload getServletFileUpload(DiskFileItemFactory factory){ ServletFileUpload upload = new ServletFileUpload(factory); // 监听文件上传进度 可以不用做 upload.setProgressListener(new ProgressListener() { @Override // pBytesRead :已读取到的文件大小 // pContentLength : 文件大小 public void update(long pBytesRead, long pContentLength, int pItems) { System.out.println("总大小:" + pContentLength + "已上传:" + pBytesRead); } }); // 处理乱码问题 upload.setHeaderEncoding("UTF-8"); // 设置单个文件的最大值 可以不写,有默认值 upload.setFileSizeMax(1024 * 1024 * 10); // 设置总共能够上传文件的大小 可以不写,有默认值 // 1024=1kb*1024=1M*10=10M upload.setSizeMax(1024 * 1024 * 10); return upload; } public static String uploadParseRequest(ServletFileUpload upload, HttpServletRequest req, String uploadPath) throws FileUploadException, IOException { String msg = ""; // 把前端请求解析,封装成一个FileItem对象,需要从ServletFileUpload对象中获取 List<FileItem> fileItems = upload.parseRequest(req); // fileItem 每一个表单对象 for (FileItem fileItem : fileItems) { // 判断上传的文件是普通的表单还是带文件的 if (fileItem.isFormField()) { // getFieldName指的是前端表单控件的name String name = fileItem.getFieldName(); String value = fileItem.getString("UTF-8");//处理乱码 System.out.println(name + ":" + value); } else { // 文件 //=========================处理文件==============================// // 拿到文件名字 String uploadFileName = fileItem.getName(); System.out.println("上传的文件名:" + uploadFileName); // 可能存在文件名不合法的情况 if (uploadFileName.trim().equals("") || uploadFileName == null) { continue; } // 获得上传的文件名 /images/girls/paojie.png String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1); // 获得文件名的后缀 String fileExtName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1); /* * 如果文件后缀名fileExtName不是我们需要的,就直接return,不处理,告诉用户文件类型不对 * * */ // 网络传输中的东西,都需要序列化 // pojo,实体类,如果想要在多个电脑上运行, 传输====>需要把对象都序列化了 // implements Serializable : 标记接口,JVM ---> 本地方法栈 native --- > C++ // 可以使用UUID(唯一识别的通用码),保证文件名唯一 // UUID.randomUUID(),随机生成一个唯一识别的通用码 String uuidPath = UUID.randomUUID().toString(); //=========================存放地址==============================// // 存到哪 uploadPath // 文件真实存在的路径realPath String realPath = uploadPath + "/" + uuidPath; // 给每个文件创建一个对应的文件夹 File realPathFile = new File(realPath); if (!realPathFile.exists()) { realPathFile.mkdir(); } //=========================文件传输==============================// // 获得文件上传的流 InputStream inputStream = fileItem.getInputStream(); // 创建一个文件输出流 // realPath = 真实的文件夹 // 差了一个文件;加上输出文件的名字 +"/"+uuidFileName FileOutputStream fos = new FileOutputStream(realPath + "/" + fileName); // 创建一个缓冲区 byte[] buffer = new byte[1024 * 1024]; // 判断是否读取文笔 int len = 0; // 如果大于0说明还存在数据 while ((len = inputStream.read(buffer)) > 0) { fos.write(buffer, 0, len); } // 关闭流 fos.close(); inputStream.close(); msg = "文件上传成功!"; fileItem.delete(); // 上传成功,清楚临时文件 } } return msg; } } |
6、记得在web.xml中配置servlet
7、遇到的问题:
在配置的Artifacts中找不到创建的upload和tmp文件夹以及上传的文件。
Artifacts这一块的ideal配置不太懂啊,既然web项目能正常运行并且在tomcat里找到了本地映射后的文件地址目前就不深究了,待到后来知识和能力够了再研究研究。
花了几个小时去搜解决方案,后来想起来之前的web项目运行之后都会在tomcat的webapps里面产生一个文件夹,然后进去看了一下,果然在。
文件夹的名字是在ideal里面配置tomcat的时候,设置的路径名字
为什么会这样,我感觉可能是装tomcat的时候创建webapps文件夹时好像是配置了一下web项目发布的文件夹都是存在webapps里面的。具体的话去看一下B站狂神的tomcat安装配置的视频就好了。因为这个不是很重要就没有太过重视
8、问题解决
是在ideal中配置tomcat时出了错误。这样配置好之后,web项目运行就不会映射到tomcat的webapps里面了