博客
关于我
SpringMVC之拦截器与文件上传下载
阅读量:609 次
发布时间:2019-03-13

本文共 13701 字,大约阅读时间需要 45 分钟。

SpringMVC之拦截器与文件上传下载

一、拦截器

SpringMVC的处理器拦截器类似于Servlet开发中的过滤器Filter,用于对处理器进行预处理和后处理。开发者可以自己定义一些拦截器来实现特定的功能。

过滤器与拦截器的区别:拦截器是AOP思想的具体应用。

过滤器,servlet规范中的一部分,任何java web工程都可以使用,在url-pattern中配置了/*之后,可以对所有要访问的资源进行拦截

拦截器 ,拦截器是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能使用,拦截器只会拦截访问的控制器方法, 如果访问的是jsp/html/css/image/js是不会进行拦截的

自定义拦截器

想要自定义拦截器,必须实现 HandlerInterceptor 接口。

步骤:

  1. 写一个拦截器类,实现HandlerInterceptor接口。

    preHandle:执行控制器Handler方法之前。

    postHandle:执行控制器Handler方法之后,视图层渲染前。
    afterCompletion:视图层渲染后。

  2. 在配置文件中配置:

public class MyInterceptor implements HandlerInterceptor {      //在请求处理的方法之前执行   //如果返回true执行下一个拦截器   //如果返回false就不执行下一个拦截器   public boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o) throws Exception {          System.out.println("------------处理前------------");       return true;  }   //在请求处理方法执行之后执行   public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {          System.out.println("------------处理后------------");  }   //在dispatcherServlet处理后执行,做清理工作.   public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {          System.out.println("------------清理------------");  }}
在springmvc的配置文件中配置拦截器

注意:如果使用自定义拦截器或者自定义过滤器都应该忽略静态资源

String url=request.getRequestURI();        if(url.indexOf(".css")>0||url.indexOf(".js")>0||url.indexOf(".png")>0||url.indexOf(".jpg")>0) {               return true;        }
//测试拦截器的控制器@Controllerpublic class InterceptorController {      @RequestMapping("/interceptor")   @ResponseBody   public String testFunction() {          System.out.println("控制器中的方法执行了");       return "hello";  }}

前端 index.jsp

拦截器测试

启动tomcat 测试一下!

例子二:

1、有一个登陆页面,需要写一个controller访问页面。

2、登陆页面有一提交表单的动作。需要在controller中处理。判断用户名密码是否正确。如果正确,向session中写入用户信息。返回登陆成功。

3、拦截用户请求,判断用户是否登陆。如果用户已经登陆。放行, 如果用户未登陆,跳转到登陆页面

1、编写一个登陆页面 login.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>   Title

登录页面


用户名:
密码:

2、编写一个Controller处理请求

@Controller@RequestMapping("/user")public class UserController {      //跳转到登陆页面   @RequestMapping("/jumplogin")   public String jumpLogin() throws Exception {          return "login";  }   //跳转到成功页面   @RequestMapping("/jumpSuccess")   public String jumpSuccess() throws Exception {          return "success";  }   //登陆提交   @RequestMapping("/login")   public String login(HttpSession session, String username, String pwd) throws Exception {          // 向session记录用户身份信息       System.out.println("接收前端==="+username);       session.setAttribute("user", username);       return "success";  }   //退出登陆   @RequestMapping("logout")   public String logout(HttpSession session) throws Exception {          // session 过期       session.invalidate();       return "login";  }}

3、编写一个登陆成功的页面 success.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>   Title

登录成功页面


${user}注销

4、在 index 页面上测试跳转!启动Tomcat 测试,未登录也可以进入主页!

<%@ page contentType="text/html;charset=UTF-8" language="java" %>    $Title$   

首页


<%--登录--%> 登录 成功页面

5、编写用户登录拦截器

public class LoginInterceptor implements HandlerInterceptor {      public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException {          // 如果是登陆页面则放行       System.out.println("uri: " + request.getRequestURI());       if (request.getRequestURI().contains("login")) {              return true;      }       HttpSession session = request.getSession();       // 如果用户已登陆也放行       if(session.getAttribute("user") != null) {              return true;      }       // 用户没有登陆跳转到登陆页面       request.getRequestDispatcher("/WEB-INF/jsp/login.jsp").forward(request, response);       return false;  }   public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {     }      public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {     }}

6、在Springmvc的配置文件中注册拦截器

7、再次重启Tomcat测试!

OK,测试登录拦截功能无误

二、文件上传和下载.

文件上传是项目开发中最常见的功能之一 ,springMVC 可以很好的支持文件上传,但是SpringMVC上下文中默认没有装配MultipartResolver,因此默认情况下其不能处理文件上传工作。如果想使用Spring的文件上传功能,则需要在上下文中配置MultipartResolver。

前端表单要求:为了能上传文件,必须将表单的method设置为POST,并将enctype设置为multipart/form-data。只有在这样的情况下,浏览器才会把用户选择的文件以二进制数据发送给服务器;

对表单中的 enctype 属性做个详细的说明:

  • application/x-www=form-urlencoded:默认方式,只处理表单域中的 value
    属性值,采用这种编码方式的表单会将表单域中的值处理成 URL 编码方式。
  • multipart/form-data:这种编码方式会以二进制流的方式来处理表单数据,这种编码方式会把文件域指定文件的内容也封装到请求参数中,不会对字符编码。
  • text/plain:除了把空格转换为 “+” 号外,其他字符都不做编码处理,这种方式适用直接通过表单发送邮件。

一旦设置了enctype为multipart/form-data,浏览器即会采用二进制流的方式来处理表单数据,而对于文件上传的处理则涉及在服务器端解析原始的HTTP响应。在2003年,Apache Software Foundation发布了开源的Commons FileUpload组件,其很快成为Servlet/JSP程序员上传文件的最佳选择。

Servlet3.0规范已经提供方法来处理文件上传,但这种上传需要在Servlet中完成。而Spring MVC则提供了更简单的封装。Spring MVC为文件上传提供了直接的支持,这种支持是用即插即用的MultipartResolver实现的。Spring MVC使用Apache Commons FileUpload技术实现了一个MultipartResolver实现类:CommonsMultipartResolver。因此,SpringMVC的文件上传还需要依赖Apache Commons FileUpload的组件。

文件上传

步骤:

  1. 视图层JSP页面表单必须设置:
enctype="multipart/form-data" method="post"

表单中设置文件上传组件:

  1. 在SpringMVC的配置文件springmvc-config.xml中设置文件上传的相关参数:
10485760
UTF-8
  1. 在控制器的控制方法参数中接收输入文件组件的值:

    必须在方法参数中提供一个类型为MultipartFile的输入参数。

@RequestParam("JSP表单输入文件组件的名") MultipartFile 参数名

文件上传在方法中的实现代码:

(1)获取并设置文件上传目录:

// 上传文件路径   String path = request.getServletContext()                       .getRealPath("/images/");

(2)获取上传文件的文件名:

// 上传文件名   String filename = file.getOriginalFilename();

(3)将上传的文件另存为:

// 将上传文件保存到一个目标文件当中   file.transferTo(new File(path+File.separator+ filename));

导入文件上传的jar包,commons-fileupload , Maven会自动帮我们导入他的依赖包 commons-io包;

commons-fileupload
commons-fileupload
1.3.3
javax.servlet
javax.servlet-api
4.0.1
provided

2、配置bean:multipartResolver

注意!!!这个bena的id必须为:multipartResolver , 否则上传文件会报400的错误!

CommonsMultipartFile 的 常用方法:

String getOriginalFilename():获取上传文件的原名InputStream getInputStream():获取文件流void transferTo(File dest):将上传文件保存到一个目录文件中

我们去实际测试一下

3、编写前端页面

4、Controller

第一种:

@Controllerpublic class FileController {      //@RequestParam("file") 将name=file控件得到的文件封装成CommonsMultipartFile 对象   //批量上传CommonsMultipartFile则为数组即可   @RequestMapping("/upload")   public String fileUpload(@RequestParam("file") CommonsMultipartFile file , HttpServletRequest request) throws IOException {          //获取文件名 : file.getOriginalFilename();       String uploadFileName = file.getOriginalFilename();       //如果文件名为空,直接回到首页!       if ("".equals(uploadFileName)){              return "redirect:/index.jsp";      }       System.out.println("上传文件名 : "+uploadFileName);       //上传路径保存设置       String path = request.getServletContext().getRealPath("/upload");       //如果路径不存在,创建一个       File realPath = new File(path);       if (!realPath.exists()){              realPath.mkdir();      }       System.out.println("上传文件保存地址:"+realPath);       InputStream is = file.getInputStream(); //文件输入流       OutputStream os = new FileOutputStream(new File(realPath,uploadFileName)); //文件输出流       //读取写出       int len=0;       byte[] buffer = new byte[1024];       while ((len=is.read(buffer))!=-1){              os.write(buffer,0,len);           os.flush();      }       os.close();       is.close();       return "redirect:/index.jsp";  }}

第二种:

// 上传文件会自动绑定到MultipartFile中	 @PostMapping(value="/upload")	 public String upload(HttpServletRequest request,			@RequestParam("description") String description,			@RequestParam("file") MultipartFile file) throws Exception{   		 		System.out.println(description);	    // 如果文件不为空,写入上传路径		if(!file.isEmpty()){   			// 上传文件路径			String path = request.getServletContext().getRealPath(	                "/images");			// 上传文件名			String filename = file.getOriginalFilename();		    File filepath = new File(path,filename);			// 判断路径是否存在,如果不存在就创建一个	        if (!filepath.getParentFile().exists()) {    	        	filepath.getParentFile().mkdirs();	        }	        // 将上传文件保存到一个目标文件当中			file.transferTo(new File(path+File.separator+ filename));			System.out.println("上传文件路径:" + (path+File.separator+ filename));			return "success";		}else{   			return "error";		}		 	 }	 	 @PostMapping(value="/register")	 public String register(HttpServletRequest request,			 @ModelAttribute User user,			 Model model)throws Exception{   		 System.out.println(user.getUsername());		// 如果文件不为空,写入上传路径		if(!user.getImage().isEmpty()){   			// 上传文件路径			String path = request.getServletContext().getRealPath(	                "/images");			// 上传文件名			String filename = user.getImage().getOriginalFilename();		    File filepath = new File(path,filename);			// 判断路径是否存在,如果不存在就创建一个	        if (!filepath.getParentFile().exists()) {    	        	filepath.getParentFile().mkdirs();	        }	        // 将上传文件保存到一个目标文件当中	        user.getImage().transferTo(new File(path+File.separator+ filename));	        // 将用户添加到model	        model.addAttribute("filename", user.getImage().getOriginalFilename());	        System.out.println("上传文件路径:" + (path+File.separator+ filename));	        return "userInfo";		}else{   			return "error";		}	}

5、测试上传文件,OK!

第三种:

采用file.Transto 来保存上传的文件

1、编写Controller

/** 采用file.Transto 来保存上传的文件*/@RequestMapping("/upload2")public String  fileUpload2(@RequestParam("file") CommonsMultipartFile file, HttpServletRequest request) throws IOException {      //上传路径保存设置   String path = request.getServletContext().getRealPath("/upload");   File realPath = new File(path);   if (!realPath.exists()){          realPath.mkdir();  }   //上传文件地址   System.out.println("上传文件保存地址:"+realPath);   //通过CommonsMultipartFile的方法直接写文件(注意这个时候)   file.transferTo(new File(realPath +"/"+ file.getOriginalFilename()));   return "redirect:/index.jsp";}

2、前端表单提交地址修改

3、访问提交测试,OK!

文件下载

文件下载步骤:

1、设置 response 响应头

2、读取文件 – InputStream

3、写出文件 – OutputStream

4、执行操作

5、关闭流 (先开后关)

代码实现:

方法一:

@RequestMapping(value="/download")public String downloads(HttpServletResponse response ,HttpServletRequest request) throws Exception{      //要下载的图片地址   String  path = request.getServletContext().getRealPath("/upload");   String  fileName = "xxx.jpg";   //1、设置response 响应头   response.reset(); //设置页面不缓存,清空buffer   response.setCharacterEncoding("UTF-8"); //字符编码   response.setContentType("multipart/form-data"); //二进制传输数据   //设置响应头   response.setHeader("Content-Disposition",           "attachment;fileName="+URLEncoder.encode(fileName, "UTF-8"));   File file = new File(path,fileName);   //2、 读取文件--输入流   InputStream input=new FileInputStream(file);   //3、 写出文件--输出流   OutputStream out = response.getOutputStream();   byte[] buff =new byte[1024];   int index=0;   //4、执行 写出操作   while((index= input.read(buff))!= -1){          out.write(buff, 0, index);       out.flush();  }   out.close();   input.close();   return null;}

前端

点击下载

测试,文件下载OK。

方法二:

@GetMapping(value="/download")	 public ResponseEntity
download(HttpServletRequest request, @RequestParam("filename") String filename, @RequestHeader("User-Agent") String userAgent )throws Exception{ // 下载文件路径 String path = request.getServletContext().getRealPath( "/round"); // 构建File File file = new File(path+File.separator+ filename); // ok表示Http协议中的状态 200 BodyBuilder builder = ResponseEntity.ok(); // 内容长度 builder.contentLength(file.length()); // application/octet-stream : 二进制流数据(最常见的文件下载)。 builder.contentType(MediaType.APPLICATION_OCTET_STREAM); // 使用URLDecoder.decode对文件名进行解码 filename = URLEncoder.encode(filename, "UTF-8"); // 设置实际的响应文件名,告诉浏览器文件要用于【下载】、【保存】attachment 以附件形式 // 不同的浏览器,处理方式不同,要根据浏览器版本进行区别判断 if (userAgent.indexOf("MSIE") > 0) { // 如果是IE,只需要用UTF-8字符集进行URL编码即可 builder.header("Content-Disposition", "attachment; filename=" + filename); } else { // 而FireFox、Chrome等浏览器,则需要说明编码的字符集 // 注意filename后面有个*号,在UTF-8后面有两个单引号! builder.header("Content-Disposition", "attachment; filename*=UTF-8''" + filename); } return builder.body(FileUtils.readFileToByteArray(file)); }
${requestScope.filename }

方法三:

@RequestMapping(value="/download")	 public ResponseEntity
download(HttpServletRequest request, @RequestParam("filename") String filename, Model model)throws Exception{ // 下载文件路径 String path = request.getServletContext().getRealPath( "/images/"); File file = new File(path+File.separator+ filename); HttpHeaders headers = new HttpHeaders(); // 下载显示的文件名,解决中文名称乱码问题 String downloadFielName = new String(filename.getBytes("UTF-8"),"iso-8859-1"); // 通知浏览器以attachment(下载方式)打开图片 headers.setContentDispositionFormData("attachment", downloadFielName); // application/octet-stream : 二进制流数据(最常见的文件下载)。 headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); // 201 HttpStatus.CREATED return new ResponseEntity
(FileUtils.readFileToByteArray(file), headers, HttpStatus.CREATED); }
${requestScope.user.image.originalFilename }

在这里插入图片描述

转载地址:http://drzaz.baihongyu.com/

你可能感兴趣的文章
MySQL InnoDB中意向锁的作用及原理探
查看>>
MySQL InnoDB事务隔离级别与锁机制深入解析
查看>>
Mysql InnoDB存储引擎 —— 数据页
查看>>
Mysql InnoDB存储引擎中的checkpoint技术
查看>>
Mysql InnoDB存储引擎中缓冲池Buffer Pool、Redo Log、Bin Log、Undo Log、Channge Buffer
查看>>
MySQL InnoDB引擎的锁机制详解
查看>>
Mysql INNODB引擎行锁的3种算法 Record Lock Next-Key Lock Grap Lock
查看>>
mysql InnoDB数据存储引擎 的B+树索引原理
查看>>
mysql innodb通过使用mvcc来实现可重复读
查看>>
mysql insert update 同时执行_MySQL进阶三板斧(三)看清“触发器 (Trigger)”的真实面目...
查看>>
mysql interval显示条件值_MySQL INTERVAL关键字可以使用哪些不同的单位值?
查看>>
Mysql join原理
查看>>
MySQL JOIN原理
查看>>
MySQL Join算法与调优白皮书(二)
查看>>
Mysql order by与limit混用陷阱
查看>>
Mysql order by与limit混用陷阱
查看>>
mysql order by多个字段排序
查看>>
MySQL Order By实现原理分析和Filesort优化
查看>>
mysql problems
查看>>
mysql replace first,MySQL中处理各种重复的一些方法
查看>>