4、商品修改功能开发(springmvc笔记)

主要内容:

  • RequestMapping特性
  • Controller方法返回值
  • 参数绑定

一、需求分析

这里我们还是使用上次整合的工程。
操作流程:

  • (1)进入商品查询列表页面
  • (2)点击修改,进入商品修改页面,页面中显示了要修改的商品信息(从数据库中查询),要修改的商品从数据查询,根据商品id(主键)查询商品信息
  • (3)在商品修改页面,修改商品信息,修改后,点击提交。

二、开发 mapper

根据上面的需求我们可以知道,mapper中此处需要完成两个功能:

  • 根据id查询商品信息,返回给页面显示
  • 更新items表的数据,更新数据库
    但是这里的两个功能需求是相对简单的,在逆向工程中已经帮我们生成好了相关代码,我们只需要直接使用即可,不需要再次开发。

三、开发 service

根据持久层mapper的相关业务需求,这里我们就可以知道业务层的功能需求

  • 根据id查询商品信息
  • 修改商品信息

3.1 接口

ItemsServiceI.java

//根据id查询商品信息
public ItemsCustom findItemsById(Integer id) throws Exception;
    
//修改商品信息,这里id本来已经在itemsCustom存在,但是为了更好的开发,还是将id提取出来
//作为一个参数,表明此id必须传入
public void updateItems(Integer id, ItemsCustom itemsCustom) throws Exception;

3.2 实现

ItemsServiceImpl.java

package cn.itcast.ssm.service.impl;
import java.util.List;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import cn.itcast.ssm.mapper.ItemsMapper;
import cn.itcast.ssm.mapper.ItemsMapperCustom;
import cn.itcast.ssm.pojo.Items;
import cn.itcast.ssm.pojo.ItemsCustom;
import cn.itcast.ssm.pojo.ItemsQueryVo;
import cn.itcast.ssm.service.ItemsServiceI;

public class ItemsServiceImpl implements ItemsServiceI{
    
    @Autowired
    private ItemsMapperCustom itemsMapperCustom;
    @Autowired
    private ItemsMapper itemsMapper;
    
    @Override
    public List<ItemsCustom> findItemsList(ItemsQueryVo itemsQueryVo)
            throws Exception {
        //通过ItemsMapperCustom查询数据库,通过spring注入
        return itemsMapperCustom.findItemsList(itemsQueryVo);
    }

    @Override
    public ItemsCustom findItemsById(Integer id) throws Exception { 
        Items items = itemsMapper.selectByPrimaryKey(id);
        //查询出来的数据可能需要进行一些业务处理,最后要返回ItemsCustom
        ItemsCustom itemsCustom = new ItemsCustom();
        //将Items内容拷贝到ItemsCustom
        BeanUtils.copyProperties(items, itemsCustom);
        
        return itemsCustom;
    }

    @Override
    public void updateItems(Integer id, ItemsCustom itemsCustom) throws Exception {
        //添加一些业务校验,通常在service接口对关键的参数进行校验
        //校验id是否为空等等,如果为空则抛出异常
        
        
        //更新商品信息,使用updateByPrimaryKeyWithBLOBs可以根据id更新表中所有字段,
        //包括大文本字段,这里因为表中有个字段属性为text,所以使用此方法
        //要求必须传入id,即使类中已经存在
        itemsCustom.setId(id);
        itemsMapper.updateByPrimaryKeyWithBLOBs(itemsCustom);
    }
}

说明:这里我们的方法还不是很完善,比如这里就没有给出查询条件,在后面一步步完善。

四、开发Controller

ItemsController.java

package cn.itcast.ssm.controller;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.servlet.ModelAndView;
import cn.itcast.ssm.pojo.ItemsCustom;
import cn.itcast.ssm.service.ItemsServiceI;
//商品Controller

@Controller
public class ItemsController {
    
    @Autowired
    private ItemsServiceI itemsService;
    
    //商品查询
    @RequestMapping("/queryItems")
    public ModelAndView queryItems() throws Exception{
        List<ItemsCustom> itemsList = itemsService.findItemsList(null);//这里暂时还没有查询条件
        
        //返回ModelAndView
        ModelAndView modelAndView = new ModelAndView();
        //相当于request的setAttribute方法,在jsp页面中就可以通过items取数据了
        modelAndView.addObject("itemsList",itemsList);
        
        //指定视图
        modelAndView.setViewName("items/itemsList");
        
        return modelAndView;
    }
    //商品信息修改页面展示
    @RequestMapping("/editItems")
    public ModelAndView editItems() throws Exception{
        //调用service根据id查询商品信息
        ItemsCustom itemsCustom = itemsService.findItemsById(1);
        
        ModelAndView modelAndView = new ModelAndView();
        //将商品信息放入Model
        modelAndView.addObject("itemsCustom", itemsCustom);
        
        //返回商品修改页面
        modelAndView.setViewName("items/editItems");
        
        return modelAndView;
    }
    
    
    //商品修改提交
    @RequestMapping("/editItemsSubmit")
    public ModelAndView editItemsSubmit() throws Exception{
        //调用service更新商品信息,页面需要将商品信息传到此方法
        //............
        
        ModelAndView modelAndView = new ModelAndView();
        //先返回一个成功页面
        
        //返回商品修改页面
        modelAndView.setViewName("success");
        
        return modelAndView;
    }
}

说明:对于商品列表展示页面在整合工程中已经给出,这里我们给出修改页面:
WEB-INF/jsp/items/editItems.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"
    isELIgnored="false"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>修改商品信息</title>
</head>
<body>
    <form id="itemForm" action="${pageContext.request.contextPath }/editItemsSubmit.action"
        method="post">
        <input type="hidden" name="id" value="${itemsCustom.id }" /> 修改商品信息:
        <table width="100%" border=1>
            <tr>
                <td>商品名称</td>
                <td><input type="text" name="name" value="${itemsCustom.name }" /></td>
            </tr>
            <tr>
                <td>商品价格</td>
                <td><input type="text" name="price" value="${itemsCustom.price }" /></td>
            </tr>
            <tr>
                <td>商品简介</td>
                <td><textarea rows="3" cols="30" name="detail">${itemsCustom.detail }</textarea></td>
            </tr>
            <tr>
                <td colspan="2" align="center"><input type="submit" value="提交" /></td>
            </tr>
        </table>
    </form>
</body>
</html>

下面我们就可以部署工程,使用地址http://localhost:8080/springmvc-mybatis01/queryItems.action进行访问,此时点击页面中的修改页面,就会找到editItems.action,我们可以看到修改页面中回显了我们选择要修改商品的信息。当然这里还有一个成功页面WEB-INF/jsp/items/success.jsp

五、RequestMapping特性

5.1 普通 url 地址映射

普通url地址映射在之前讲过,比如:

@RequestMapping("/queryItems")

5.2 窄化请求映射

所谓窄化请求映射就是当控制器中方法很多,映射很多的时候,我们为了便于管理,一般会将这些url地址进行分类管理。这里我们对控制器进行改造,窄化请求映射。

1.png

这里我们使用:

@RequestMapping("/items")

进行窄化请求映射,于是最终的url就是跟路径+子路径,比如/items/queryItems.action。我们在此处改动之后需要对jsp页面进行改进,在itemsList.jsp

<td><a href="${pageContext.request.contextPath }/items/editItems.action?id=${item.id}">修改</a></td>
<%-- <td><a href="${pageContext.request.contextPath }/editItems.action?id=${item.id}">修改</a></td> --%>

editItems.jsp

<form id="itemForm" action="${pageContext.request.contextPath }/items/editItemsSubmit.action" method="post">
<%-- <form id="itemForm" action="${pageContext.request.contextPath }/editItemsSubmit.action" method="post"> --%>

其中被注释掉的是之前的方式。

5.3 限制 http 请求方法

一般常用请求方式有GETPOST两种,但是有时候我们需要规定必须使用哪种方式,在ItemsController.java:

@RequestMapping(value="/editItems", method={RequestMethod.POST, RequestMethod.GET})
public ModelAndView editItems() throws Exception{

这里我们限制请求方式必须是GET或者POST,当然如果这里我们限制为POST,那么默认的GET提交方式就提交不成功。

六、Controller方法的返回值

6.1 返回 ModelAndView

这个在之前我们已经讲过,这里不再说明。

6.2 返回 String

  • 1.返回String
    如果Controller方法返回的是String,表示返回逻辑视图名。而真正视图(jsp)=前缀+逻辑视图名+后缀
@RequestMapping(value="/editItems", method={RequestMethod.POST, RequestMethod.GET})
public String editItems(Model model) throws Exception{
    //调用service根据id查询商品信息
    ItemsCustom itemsCustom = itemsService.findItemsById(1);
    //通过形参中的model将model数据传递到页面
    //相当于modelAndView.addObject("itemsCustom", itemsCustom);
    model.addAttribute("itemsCustom", itemsCustom);
    return "items/editItems";
}

说明:可以看到我们首先将数据存入到Model中,然后返回逻辑视图地址。而真正的视图(jsp)还需要加上前缀和后缀。

  • 2.redirect重定向
    需求:商品修改成功之后重定向到商品查询列表。修改提交的request数据无法传递到重定向的地址。因为重定向后重新进行request,也就是重定向不能共享request
@RequestMapping("/editItemsSubmit")
public String editItemsSubmit() throws Exception{
    //调用service更新商品信息,页面需要将商品信息传到此方法
    //............
    
    //ModelAndView modelAndView = new ModelAndView();
    //先返回一个成功页面
    
    //返回商品修改页面
    //modelAndView.setViewName("success");
    
    //重定向
    return "redirect:queryItems.action";//根路径不需要加
    //return "success";
}

说明:一定注意在Controller方法中重定向时不需要加根路径。

  • 3.forward请求转发
@RequestMapping("/editItemsSubmit")
public String editItemsSubmit() throws Exception{
    return "forward:queryItems.action";
}

说明:通过此种方式进行页面转发,url地址栏不变,request可以共享。当然我们说request可以共享,那么我们可以给方法传递一个参数HttpServletRequest

@RequestMapping("/editItemsSubmit")
public String editItemsSubmit(HttpServletRequest request) throws Exception{
    return "forward:queryItems.action";
}

于是我们在其跳转的方法中可以接收request

//商品查询
@RequestMapping("/queryItems")
public ModelAndView queryItems(HttpServletRequest request) throws Exception{
        //测试request共享
        System.out.println(request.getParameter("id"));
......
}

这样就实现了request的共享。我们可以在此方法中进行测试。

6.3 返回 void

Controller方法行参上可以定义requestresponse,使用request或者response指定响应结果:

  • 1)、使用request转向页面,如下:
request.getRequestDispatcher("").forward(request, response);
  • 2)、也可以通过response页面重定向
response.sendRedirect("url");
  • 3)、也可以通过response指定相应结果,例如响应json数据如下:
response.setCharacterEncoding("utf-8");
response.setContentType("application/json;charset=utf-8");
response.getWriter().write("json串");

七、参数绑定(工程springmvc-mybatis02

7.1 参数绑定过程

  • 从客户端请求key/value数据,经过参数绑定,将key/value数据绑定到Controller方法的形参上,接收页面提交的数据是通过方法的形参,而不是在Controller类定义成员变量。

  • 处理器适配器调用springmvc提供参数绑定组件将key/value数据转成Controller方法的形参。

  • 参数绑定组件:在springmvc早起版本使用PropertyEditor(只能将字符串转成java对象),后期使用converter(进行任意参数类型的转换)。springmvc提供 了很多converter(转换器)。在特殊情况下需要自定义converter(比如对日期数据的绑定需要自定义)。然后调用Controller

7.2 参数绑定默认支持的类型

直接在Controller方法的形参上定义下面类型的对象,就可以使用这些对象。在参数绑定过程中,如果遇到下边的类型直接进行绑定(自动进行的)。

  • 1)、HttpServletRequest
    通过request对象获取请求信息

  • 2)HttpServletResponse
    通过response处理响应信息

  • 3)HttpSession
    通过session对象得到session中存放的对象。

  • 4)Model/ModelMap
    Model是一个接口,ModelMap是接口实现。在Controller方法形参中我们可以定义成前者或者后者都行。作用:将模型数据填充到request域。

7.3 绑定简单类型

public String editItems(Model model, @RequestParam(value="id", required=true, defaultValue="") Integer items_id)

说明:这里我们绑定了一个简单类型(Integer),如果这个参数名称为id,也就是和request传入参数名称一致,那么我们不需要使用注解。但是这里我们参数名称是items_id,也就是不一致,那么我们需要使用上面的注解进行绑定。@RequestParamvalue指定request传入参数名称和形参绑定,required=true指定此参数是否必须被传入,defaultValue=""设置默认值。可以绑定的简单类型还有String、float、double、boolean

7.4 绑定 pojo 对象

@RequestMapping("/editItemsSubmit")
public String editItemsSubmit(HttpServletRequest request, Integer id, ItemsCustom itemsCustom) throws Exception{
    
    itemsService.updateItems(id, itemsCustom);
    
    return "forward:queryItems.action";
}

说明:这里我们绑定pojo类有个前提,就是页面中input的名称和Controllerpojo形参中的属性名称一致,可以自动将页面中的数据绑定到pojo

  • 问题一:乱码问题
    • POST乱码
      web.xml中添加post乱码过滤器:
<!-- post乱码过虑器 -->
<filter>
    <filter-name>CharacterEncodingFilter</filter-name>
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <init-param>
        <param-name>encoding</param-name>
        <param-value>utf-8</param-value>
    </init-param>
</filter>
<filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
  • GET乱码
    对于get请求中文参数中出现乱码解决方法有两个:
    1.修改tomcat配置文件(server.xml)添加编码与工程一致,如下:
    1

    2.另一种方法对参数进行重新编码:
String username = new String(request.getParameter("username").getBytes("iso8859-1"),"utf-8");
  • 问题二:日期类型的绑定问题
    我们将editItems.jsp中这段代码的注释去掉:
<tr>
    <td>商品生产日期</td>
    <td><input type="text" name="createtime" value="<fmt:formatDate value="${itemsCustom.createtime}" pattern="yyyy-MM-dd HH:mm:ss"/>"/></td>
</tr>

然后我们再次访问时发生错误,这里的问题是日期类型不能自动绑定,需要我们手动编写转换器之后绑定。对于Controller形参中pojo对象,如果属性中有日期类型,需要自定义参数绑定。将请求的日期数据串转换成日期类型,要转换的日期类型和pojo中日期属性的类型保持一致。需要像处理器适配器中注入自定义的参数绑定组件。在springmvc.xml中:

<mvc:annotation-driven conversion-service="conversionService"></mvc:annotation-driven>

<!-- 自定义参数绑定 -->
<bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
    <!-- 转换器 -->
    <property name="converters">
        <list>
            <!-- 日期类型的转换 -->
            <bean class="cn.itcast.ssm.controller.convert.CustomDateConverter"></bean>
            <!-- 其他类型的转换 -->
        </list>
    </property>
</bean>

说明:我们使用conversion-service属性给适配器中添加自定义转换器。
自定义转换器CustomDateConverter.java

package cn.itcast.ssm.controller.convert;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import org.springframework.core.convert.converter.Converter;

public class CustomDateConverter implements Converter<String, Date>{

    @Override
    public Date convert(String source) {
        //实现将日期字符串转换成日期类型(格式是yyyy-MM-dd HH:mm:ss)
        SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        try {
            //如果转换成功
            return format.parse(source);
        } catch (ParseException e) {
            e.printStackTrace();
        }
        //如果绑定失败
        return null;
    }
}

说明:注意自定义转换器需要实现Converter接口。其中String是我们需要转换的类型,Date是我们最终需要的类型。

  • 最后:上面我们是将适配器和映射器配置在一起,可能不太容易理解,如果单独配置,那么我们的转换器配置应该如下:
<!--注解适配器 -->
    <bean
        class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
         <property name="webBindingInitializer" ref="customBinder"></property> 
    </bean>
    
    <!-- 自定义webBinder -->
    <bean id="customBinder"
        class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">
        <property name="conversionService" ref="conversionService" />
    </bean>
    <!-- conversionService -->
    <bean id="conversionService"
        class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
        <!-- 转换器 -->
        <property name="converters">
            <list>
                <bean class="cn.itcast.ssm.controller.converter.CustomDateConverter"/>
            </list>
        </property>
    </bean>

当然这种方式较为繁琐,这里只是了解。

八、springmvc 和 struts2的区别

  • 1、springmvc基于方法开发的,struts2基于类开发的。springmvc映射时将urlController方法进行映射。映射成功之后springmvc会生成一个handler对象,对象中只包括了一个method,方法执行结束,形参数据就销毁了。springmvcController开发类似service开发。

  • 2、springmvc可以进行单例开发,并且也建议单例开发。Struts2通过类的成员变量接收的参数,所以无法使用单例,只能使用多例。

  • 3、struts2速度慢在于使用struts2标签。建议,如果使用struts2,直接使用jstl,不要使用struts2的标签。

最后:我们看到这里我们使用这样一个例子将springmvc开发中所需要用到的一些基本内容说明了,对于其他功能开发基本就类似了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容