主要内容:
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 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 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
isELIgnored="false"%>
修改商品信息method="post">
修改商品信息:
商品名称 | |
商品价格 | |
商品简介 | ${itemsCustom.detail } |
下面我们就可以部署工程,使用地址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中
修改修改 --%>
在editItems.jsp
--%>
其中被注释掉的是之前的方式。
5.3 限制 http 请求方法
一般常用请求方式有GET和POST两种,但是有时候我们需要规定必须使用哪种方式,在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方法行参上可以定义request和response,使用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,也就是不一致,那么我们需要使用上面的注解进行绑定。@RequestParam中value指定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的名称和Controller的pojo形参中的属性名称一致,可以自动将页面中的数据绑定到pojo中
问题一:乱码问题
POST乱码
在web.xml中添加post乱码过滤器:
CharacterEncodingFilter
org.springframework.web.filter.CharacterEncodingFilter
encoding
utf-8
CharacterEncodingFilter
/*
GET乱码
对于get请求中文参数中出现乱码解决方法有两个:
1.修改tomcat配置文件(server.xml)添加编码与工程一致,如下:
1
2.另一种方法对参数进行重新编码:
String username = new String(request.getParameter("username").getBytes("iso8859-1"),"utf-8");
问题二:日期类型的绑定问题
我们将editItems.jsp中这段代码的注释去掉:
商品生产日期"/>然后我们再次访问时发生错误,这里的问题是日期类型不能自动绑定,需要我们手动编写转换器之后绑定。对于Controller形参中pojo对象,如果属性中有日期类型,需要自定义参数绑定。将请求的日期数据串转换成日期类型,要转换的日期类型和pojo中日期属性的类型保持一致。需要像处理器适配器中注入自定义的参数绑定组件。在springmvc.xml中:
说明:我们使用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{
@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是我们最终需要的类型。
最后:上面我们是将适配器和映射器配置在一起,可能不太容易理解,如果单独配置,那么我们的转换器配置应该如下:
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">
class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
当然这种方式较为繁琐,这里只是了解。
八、springmvc 和 struts2的区别
1、springmvc基于方法开发的,struts2基于类开发的。springmvc映射时将url和Controller方法进行映射。映射成功之后springmvc会生成一个handler对象,对象中只包括了一个method,方法执行结束,形参数据就销毁了。springmvc的Controller开发类似service开发。
2、springmvc可以进行单例开发,并且也建议单例开发。Struts2通过类的成员变量接收的参数,所以无法使用单例,只能使用多例。
3、struts2速度慢在于使用struts2标签。建议,如果使用struts2,直接使用jstl,不要使用struts2的标签。
最后:我们看到这里我们使用这样一个例子将springmvc开发中所需要用到的一些基本内容说明了,对于其他功能开发基本就类似了。