springmvc&参数绑定&异常处理&Json数据交互&上传图片

springmvc的高级参数绑定

绑定数组
    <c:forEach items="${itemList }" var="item">
        <tr>
        <td><input type="checkbox" name="ids" value="${item.id}"/></td>
        </tr>
    </c:forEach>
Controller层接收
Controller方法中可以用String[]接收(变量名要和前端name属性一致),或者pojo的String[]属性(属性名要和前端的name属性值一致)接收。两种方式任选其一即可。
    public String queryItem(QueryVo queryVo, Integer[] ids) 
将表单的数据绑定到List
List中存放对象,并将定义的List放在包装类QueryVo中
name属性必须是list属性名+下标+元素属性。
pojo定义:
![pojo接收list参数.png][1]
jsp页面:
![接收list参数jsp页面.png][2]

<c:forEach items="${itemList }" var="item" varStatus="s">
    <tr>
    <td><input type="checkbox" name="ids" value="${item.id}"/></td>
    <td>
        <input type="hidden" name="itemList[${s.index}].id" value="${item.id }"/>
        <input type="text" name="itemList[${s.index}].name" value="${item.name }"/>
    </td>
    <td><input type="text" name="itemList[${s.index}].price" value="${item.price }"/></td>
    <td><input type="text" name="itemList[${s.index}].createtime" value="<fmt:formatDate value="${item.createtime}" pattern="yyyy-MM-dd HH:mm:ss"/>"/></td>
    <td><input type="text" name="itemList[${s.index}].detail" value="${item.detail }"/></td>
    
    <td><a href="${pageContext.request.contextPath }/itemEdit.action?id=${item.id}">修改</a></td>

    </tr>
</c:forEach>
${current}    当前这次迭代的(集合中的)项
${status.first}    判断当前项是否为集合中的第一项,返回值为true或false
${status.last}    判断当前项是否为集合中的最
varStatus属性常用参数总结下:
${status.index}    输出行号,从0开始。
${status.count}    输出行号,从1开始。
${status.后一项,返回值为true或false
begin、end、step分别表示:起始序号,结束序号,跳跃步伐。

RequestMapping

通过@RequestMapping注解可以定义不同的处理器映射规则。
@RequestMapping(value = { "itemList", "itemListAll" })
添加在类上面
在class上添加@RequestMapping(url)指定通用请求前缀, 限制此类下的所有方法请求url必须以请求前缀开头

请求方法限定

除了可以对url进行设置,还可以限定请求进来的方法
限定GET方法
@RequestMapping(method = RequestMethod.GET)
如果通过POST访问则报错:
HTTP Status 405 - Request method 'POST' not supported
限定POST方法
@RequestMapping(method = RequestMethod.POST)
如果通过GET访问则报错:
@RequestMapping(method = RequestMethod.POST)

Controller方法返回值

 - 返回ModelAndView(不推荐使用,耦合太高)
controller方法中定义ModelAndView对象并返回,对象中可添加model数据、指定view。

 - 返回void(ajax请求使用)
    在Controller方法形参上可以定义request和response,使用request或response指定响应结果:
1、使用request转发页面,如下:
request.getRequestDispatcher("页面路径").forward(request, response);
request.getRequestDispatcher("/WEB-INF/jsp/success.jsp").forward(request, response);
2、可以通过response页面重定向:
response.sendRedirect("url")
response.sendRedirect("/springmvc-web2/itemEdit.action");
3、可以通过response指定响应结果,例如响应json数据如下:
response.getWriter().print("{\"abc\":123}");
 - 返回字符串
5.3.1.逻辑视图名
controller方法返回字符串可以指定逻辑视图名,通过视图解析器解析为物理视图地址。
//指定逻辑视图名,经过视图解析器(配置了)解析为jsp物理路径:/WEB-INF/jsp/itemList.jsp
return "itemList";

Redirect重定向和forward转发

Redirect重定向:
Contrller方法返回字符串可以重定向到一个url地址
如下商品修改提交后重定向到商品编辑页面
    public String updateItemById(Item item) {
        // 更新商品
        this.itemService.updateItemById(item);

        // 修改商品成功后,重定向到商品编辑页面
        // 重定向后浏览器地址栏变更为重定向的地址,
        // 重定向相当于执行了新的request和response,所以之前的请求参数都会丢失
        // 如果要指定请求参数,需要在重定向的url后面添加 ?itemId=1 这样的请求参数
        return "redirect:/itemEdit.action?itemId=" + item.getId();
    }
forward转发:
Controller方法执行后继续执行另一个Controller方法
如下商品修改提交后转向到商品修改页面,修改商品的id参数可以带到商品修改方法中。
    @RequestMapping("updateItem")
    public String updateItemById(Item item) {
        // 更新商品
        this.itemService.updateItemById(item);

        // 修改商品成功后,重定向到商品编辑页面
        // 重定向后浏览器地址栏变更为重定向的地址,
        // 重定向相当于执行了新的request和response,所以之前的请求参数都会丢失
        // 如果要指定请求参数,需要在重定向的url后面添加 ?itemId=1 这样的请求参数
        // return "redirect:/itemEdit.action?itemId=" + item.getId();

        // 修改商品成功后,继续执行另一个方法
        // 使用转发的方式实现。转发后浏览器地址栏还是原来的请求地址,
        // 转发并没有执行新的request和response,所以之前的请求参数都存在
        return "forward:/itemEdit.action";
    }

异常处理器

springmvc在处理请求过程中出现异常信息交由异常处理器进行处理,自定义异常处理器可以实现一个系统的异常处理逻辑
异常处理思路:
    系统中异常包括两类:预期异常和运行时异常RuntimeException,前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发、测试通过手段减少运行时异常的发生。
    系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理
![异常处理器.png][3]

自定义异常类:
public class MyException extends Exception {
    // 异常信息
    private String message;

    public MyException() {
        super();
    }

    public MyException(String message) {
        super();
        this.message = message;
    }

    public String getMessage() {
        return message;
    }

    public void setMessage(String message) {
        this.message = message;
    }
}
自定义异常处理器:
public class CustomHandleException implements HandlerExceptionResolver {

    @Override
    public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler,
            Exception exception) {
        // 定义异常信息
        String msg;

        // 判断异常类型
        if (exception instanceof MyException) {
            // 如果是自定义异常,读取异常信息
            msg = exception.getMessage();
        } else {
            // 如果是运行时异常,则取错误堆栈,从堆栈中获取异常信息
            Writer out = new StringWriter();
            PrintWriter s = new PrintWriter(out);
            exception.printStackTrace(s);
            msg = out.toString();

        }

        // 把错误信息发给相关人员,邮件,短信等方式
        // TODO

        // 返回错误页面,给用户友好页面显示错误信息
        ModelAndView modelAndView = new ModelAndView();
        modelAndView.addObject("msg", msg);
        modelAndView.setViewName("error");

        return modelAndView;
    }
}
异常处理器配置:
在springmvc.xml中添加:
<!-- 配置全局异常处理器 -->
    <bean id="customHandleException"     class="cn.itcast.ssm.exception.CustomHandleException"/>

上传图片

配置虚拟目录
在tomcat上配置图片虚拟目录,在tomcat下conf/server.xml中添加:
    <Context docBase="D:\develop\upload\temp" path="/pic" reloadable="false"/>
    访问http://localhost:8080/pic即可访问D:\develop\upload\temp下的图片。

1.加入jar包
![上传图片所需要的jar包.png][4]
2.配置上传解析器
在springmvc.xml中配置文件上传解析器
<!-- 文件上传,id必须设置为multipartResolver -->
    <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!-- 设置文件上传大小 -->
        <property name="maxUploadSize" value="5000000" />
    </bean>
3.jsp页面修改
前端图片上传功能
![上传图片jsp页面1.png][5]
设置表单可以进行文件上传
![上传图片jsp页面2.png][6]

4.图片上传代码:
    @RequestMapping("updateItem")
    public String updateItemById(Item item, MultipartFile pictureFile) throws Exception {
        // 图片上传
        // 设置图片名称,不能重复,可以使用uuid
        String picName = UUID.randomUUID().toString();

        // 获取文件名
        String oriName = pictureFile.getOriginalFilename();
        // 获取图片后缀
        String extName = oriName.substring(oriName.lastIndexOf("."));

        // 开始上传
        pictureFile.transferTo(new File("C:/upload/image/" + picName + extName));

        // 设置图片名到商品中
        item.setPic(picName + extName);

        // 更新商品
        this.itemService.updateItemById(item);

        return "forward:/itemEdit.action";
    }

json数据交互

@RequestBody
作用:
    @RequestBody注解用于读取http请求的内容(字符串),通过springmvc提供的HttpMessageConverter接口将读到的内容(json数据)转换为java对象并绑定到Controller方法的参数上。

传统的请求参数:
    itemEdit.action?id=1&name=zhangsan&age=12
现在的请求参数:
    使用POST请求,在请求体里面加入json数据
    {
    "id": 1,
    "name": "测试商品",
    "price": 99.9,
    "detail": "测试商品描述",
    "pic": "123456.jpg"
    }
@RequestBody注解实现接收http请求的json数据,将json数据转换为java对象进行绑定

@ResponseBody
作用:
@ResponseBody注解用于将Controller的方法返回的对象,通过springmvc提供的HttpMessageConverter接口转换为指定格式的数据如:json,xml等,通过Response响应给客户端

@ResponseBody注解实现将Controller方法返回java对象转换为json响应给客户端。

1.加入jar包
如果需要springMVC支持json,必须加入json的处理jar
![json数据交互jar.png][7]
2.ItemController编写
    /**
     * 测试json的交互
     * @param item
     * @return
     */
    @RequestMapping("testJson")
    public @ResponseBody Item testJson(@RequestBody Item item) {
        return item;
    }
配置json转换器(建议使用)
如果不使用注解驱动<mvc:annotation-driven />,就需要给处理器适配器配置json转换器,参考之前学习的自定义参数绑定。

在springmvc.xml配置文件中,给处理器适配器加入json转换器
<!--处理器适配器 -->
    <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
        <property name="messageConverters">
        <list>
        <bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean>
        </list>
        </property>
    </bean>

拦截器

Spring Web MVC 的处理器拦截器类似于Servlet 开发中的过滤器Filter,用于对处理器进行预处理和后处理。

拦截器定义:
实现HandlerInterceptor接口,代码如下:
public class HandlerInterceptor1 implements HandlerInterceptor {
    // controller执行后且视图返回后调用此方法
    // 这里可得到执行controller时的异常信息
    // 这里可记录操作日志
    @Override
    public void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3)
            throws Exception {
        System.out.println("HandlerInterceptor1....afterCompletion");
    }

    // controller执行后但未返回视图前调用此方法
    // 这里可在返回用户前对模型数据进行加工处理,比如这里加入公用信息以便页面显示
    @Override
    public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3)
            throws Exception {
        System.out.println("HandlerInterceptor1....postHandle");
    }
@Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object arg2) throws Exception {
    // 从request中获取session
    HttpSession session = request.getSession();
    // 从session中获取username
    Object username = session.getAttribute("username");
    // 判断username是否为null
    if (username != null) {
        // 如果不为空则放行
        return true;
    } else {
        // 如果为空则跳转到登录页面
        response.sendRedirect(request.getContextPath() + "/user/toLogin.action");
    }

    return false;
    }
}
拦截器配置:
在springmvc.xml中配置拦截器
<!-- 配置拦截器 -->
    <mvc:interceptors>
        <mvc:interceptor>
            <!-- 所有的请求都进入拦截器 -->
            <mvc:mapping path="/**" />
            <!-- 配置具体的拦截器 -->
            <bean class="cn.itcast.ssm.interceptor.HandlerInterceptor1" />
        </mvc:interceptor>
    </mvc:interceptors>
如果配置两个拦截器的执行流程:
HandlerInterceptor1和HandlerInterceptor2是两个拦截器执行流程
    HandlerInterceptor1..preHandle..
    HandlerInterceptor2..preHandle..

    HandlerInterceptor2..postHandle..
    HandlerInterceptor1..postHandle..

    HandlerInterceptor2..afterCompletion..
    HandlerInterceptor1..afterCompletion..

猜你喜欢

转载自blog.csdn.net/qq_40325734/article/details/80793792
今日推荐