1. 文件上传
SpringMVC 的文件上传非常简便,首先导入文件上传依赖的 jar:
<!-- 文件上传所依赖的 jar 包 -->
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency>
在 springMVC-servlet.xml 配置文件中配置文件解析器:
<!--1*1024*1024 即 1M resolveLazily 属性启用是为了推迟文件解析,以便捕获文件大小异常
-->
<!--文件上传解析器-->
<bean id="multipartResolver"
class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<property name="maxUploadSize" value="1048576"/>
<property name="defaultEncoding" value="UTF-8"/>
<property name="resolveLazily" value="true"/>
</bean>
注意解析器的 id 必须等于 multipartResolver,否则上传会出现异常:
1.1 单个文件上传
@Controller
public class FileController {
/**
* 上传单个文件操作
* MultipartFile file 就是上传的文件
* @return
*/
@RequestMapping(value = "/upload1.html")
public String fileUpload1(@RequestParam("file") MultipartFile file) {
try {
//将上传的文件存在 E:/upload/下
FileUtils.copyInputStreamToFile(file.getInputStream(),
new File("E:/upload/",
file.getOriginalFilename()));
} catch (Exception e) {
e.printStackTrace();
}
//上传成功返回原来页面
return "/file.jsp";
}
}上传文件时,Controller 的方法中参数类型是 MultipartFile 即可将文件映射到参数上。
页面file.jsp:
<form method="post" action="/upload1.html" enctype="multipart/form-data">
<input type="file" name="file"/>
<button type="submit" >提交</button>
</form>
另外上传的文件还可以映射成 CommonsMultipartFile,它是 MultipartFile 的子类:
/**
* CommonsMultipartFile 是 MultipartFile 的子类
* @return
*/
@RequestMapping("/upload2.html")
public String fileUpload2(@RequestParam("file") CommonsMultipartFile file) {
try {
System. out .println("fileName:" + file.getOriginalFilename());
String path = "E:/upload/" + file.getOriginalFilename();
File newFile = new File(path);
//通过 CommonsMultipartFile 的方法直接写文件
file.transferTo(newFile);
} catch (Exception e) {
e.printStackTrace();
}
return "/file.jsp";
}
页面:
<form method="post" action="/upload2.html" enctype="multipart/form-data">
<input type="file" name="file"/>
<button type="submit" >提交</button>
</form>
1.2 批量上传
/**
* 批量上传的时候参数是数组
* @return
*/
@RequestMapping("/upload3.html")
public String fileUpload3(@RequestParam("file") CommonsMultipartFile[] file) {
try {
//批量上传时遍历文件数组
for (CommonsMultipartFile f : file) {
System. out .println("fileName:" + f.getOriginalFilename());
String path = "E:/upload/" + f.getOriginalFilename();
File newFile = new File(path);
f.transferTo(newFile);
}
} catch (Exception e) {
e.printStackTrace();
}
return "/file.jsp";
}
页面:
<form method="post" action="/upload3.html" enctype="multipart/form-data">
<!--注意 name 都是 file,与 Controller 中方法的参数名对应-->
<input type="file" name="file"/>
<input type="file" name="file"/>
<input type="file" name="file"/>
<button type="submit" >提交</button>
</form>
2. 使用拦截器
模拟登陆拦截器:
public class LoginInterceptor extends HandlerInterceptorAdapter { {
@Override
public bo olean preHandle(HttpServletRequest request, HttpServletResponse response,
Object handler) throws Exception { {
System. out .println(" " 开始拦截" ");
// 其他情况判断 n session 中是否有 key ,有的话继续用户的操作
if ( request. getSession(). getAttribute( "user") != null) { {
return true;
} }
// 最后的情况就是进入登录页面
response. sendRedirect( request. getContextPath() + "/login.jsp");
return false;
} }
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response,
Object handler, ModelAndView modelAndView) throws Exception { {
System. out .println(" " 视图解析前 postHandle");
} }
@Override
public void afterConcurrentHandlingStarted(HttpServletRequest request,
HttpServletResponse response, Object handler) throws Exception { {
System. out .println(" " 处理异步请求" ");
} }
} }
配置拦截器:
<mvc:interceptors>
<mvc:interceptor><!--配置局部拦截器,需要满足下列路径条件-->
<mvc:mapping path="/**"/>
<mvc:exclude-mapping path="/login.html" />
<bean class="interceptor.LoginInterceptor"/><!--自定义拦截器注册-->
</mvc:interceptor>
</mvc:interceptors>
登陆的 Controller
@Controller
public class LoginController { {
@RequestMapping( "login.html")
public String login( String username, HttpServletRequest request){ {
// 模拟登陆
request. getSession(). setAttribute( "user", username);
return "redirect:index.html";
} }
} }
登陆页面:
<form action="/login.html">
<input name="username"/><input type="submit">
</form>
3. 异常处理
3.1 集成异常处理
对 于 一 些 通用 的 ,不 需 要特 殊 处 理的 异 常,可 以 使 用统 一 的 异常 处 理器 , 在springMVC-servlet.xml 中加入配置:
<!--集成异常处理-->
<bean
class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
<!-- 定义默认的异常处理页面-->
<property name="defaultErrorView" value="error.jsp"></property>
<!-- 定义异常处理页面用来获取异常信息的变量名,默认名为 exception -->
<property name="exceptionAttribute" value="ex"></property>
<!-- 定义需要特殊处理的异常,用类名或完全路径名作为 key,异常也页名作为值 -->
<property name="exceptionMappings">
<props>
<prop key="java.lang.NullPointerException">error.jsp</prop>
<prop key="java.lang.ClassCastException">error.jsp</prop>
<prop key="java.lang.IndexOutOfBoundsException">error.jsp</prop>
<!-- 这里还可以继续扩展对不同异常类型的处理 -->
</props>
</property>
</bean>
我们写一个 controller 专门抛出异常,用来模拟程序中可能出现的异常信息:
@Controller
public class ExceptionController {
@RequestMapping("/ex.html")
public String exceptionTest(Integer type) throws Exception {
//手动抛出几个异常,模拟程序中可能出现的异常
switch (type) {
case 1:
throw new NullPointerException("测试空指针异常");
case 2:
throw new ClassCastException("测试类型转换异常");
case 3:
throw new IndexOutOfBoundsException("测试越界异常");
}
return "index.jsp";
}
}
测试页面:
<body>
<a href="/ex.html?type=1">空指针</a><br/>
<a href="/ex.html?type=2">类型转换</a><br/>
<a href="/ex.html?type=3">越界</a><br/>
${ex}<!--就是 spring 异常处理器中配置的 exceptionAttribute-->
</body>
运行结果:
点击不同的链接可以看大不同的提示信息。
3.2 自定义异常处理
对于需要特殊处理的异常,可以自定义异常处理器自定义异常处理的类需要继承 HandlerExceptionResolver
public class MyExceptionHandler implements HandlerExceptionResolver { {
@Override
public ModelAndView resolveException(HttpServletRequest httpServletRequest,
HttpServletResponse httpServletResponse, Object o o, Exception e e) { {
String msg = e e.getMessage();
httpServletRequest. setAttribute( "ex", msg);
return new ModelAndView( "/error.jsp");
} }
} }
在 spring 配置文件中定义 bean:
<!--自定义异常处理-->
<bean id="exceptionHandler" class="exception.MyExceptionHandler"/>
注:自定义异常处理和前面的集成异常处理不能一起使用,需要将 前面例子的 bean 注释掉。如果需要处理 ajax 发送的请求,出现异常时返回 json 数据,可以增加处理 json 的代码。
修改 MyExceptionHandler 中的代码:
public class MyExceptionHandler implements HandlerExceptionResolver {
@Override
public ModelAndView resolveException(HttpServletRequest httpServletRequest,
HttpServletResponse httpServletResponse, Object o, Exception e) {
String msg = e.getMessage();
if (httpServletResponse.isCommitted()) {
return null;
}
//如果是 ajax 请求就返回 json 数据
if (isAjax(httpServletRequest)) {
Map<String, String> result = new HashMap<>();
result.put("ex", msg);
MappingJackson2JsonView view = new MappingJackson2JsonView();
return new ModelAndView(view, "result", result);
} else {//不是 ajax 就返回错误页面
httpServletRequest.setAttribute("ex", msg);
return new ModelAndView("/error.jsp");
}
}
public static boolean isAjax(HttpServletRequest request) {
return
"XMLHttpRequest".equalsIgnoreCase(request.getHeader("X-Requested-With")) ||
request.getParameter("ajax") != null;
}
}
测试页面:
<script type="text/javascript" src="jquery-1.10.1.min.js"></script>
<input type="button" onclick=" testEx ()" value="测试自定义异常"/>
<script type="text/javascript">
function testEx (){
$.ajax({
type: "POST",
url: "/ex.html?type=1",
success: function(data){
alert(data.result.ex);//读取 json 数据
}
});
}
</script>
测试结果: