springmvc和mybatis整合工程搭建
整合思路
在mybatis和spring整合的基础上 添加springmvc。
spring要管理springmvc编写的Handler(controller)、mybatis的SqlSessionFactory、mapper
第一步:整合dao,spring和mybatis整合
第二步:整合service,spring管理service接口,service中可以调用spring容器中dao(mapper)
第三步:整合controller,spring管理controller接口,在controller调用service
工程结构
配置文件
applicationContext-dao.xml---配置数据源、SqlSessionFactory、mapper扫描器
applicationContext-service.xml---配置service接口
applicationContext-transaction.xml--事务管理
springmvc.xml---springmvc的配置,配置处理器映射器、适配器、视图解析器
SqlMapConfig.xml---mybatis的配置文件,配置别名、settings、mapper
1.applicationContext-dao.xml
配置mybatis的数据源、sqlSessionFactory、mapper扫描器
<!-- 加载配置文件 -->
<context:property-placeholder location="classpath:db.properties"></context:property-placeholder>
<!-- 数据库连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}"></property>
<property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
<property name="password" value="${jdbc.password}"></property>
<property name="user" value="${jdbc.user}"></property>
</bean>
<!-- SqlsessionFactory -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"></property>
<property name="configLocation" value="classpath:mybatis/sqlMapConfig.xml"></property>
</bean>
<!--
MapperScannerConfigurer:mapper的扫描器,将包下边的mapper接口自动创建代理对象,
自动创建到spring容器中,bean的id是mapper的类名(首字母小写)-->
<bean id="mapperScannerConfigurer" class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<!-- 配置扫描包的路径
如果要扫描多个包,中间使用半角逗号分隔
要求mapper.xml和mapper.java同名且在同一个目录
-->
<property name="basePackage" value="com.yuhan.ssm.mapper"/>
<!-- 使用sqlSessionFactoryBeanName -->
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
</bean>
2.applicationContext-transation.xml
在此配置文件配置事务,声明式事务控制.
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!-- 通知 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="save*" propagation="REQUIRED"/>
<tx:method name="insert*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
<tx:method name="delete*" propagation="REQUIRED"/>
<tx:method name="find*" propagation="SUPPORTS" read-only="true"/>
<tx:method name="select*" propagation="SUPPORTS" read-only="true"/>
<tx:method name="get*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</tx:advice>
<!-- aop -->
<aop:config>
<aop:advisor advice-ref="txAdvice"
pointcut="execution(* com.yuhan.ssm.service.imp.*.*(..))"/>
</aop:config>
3.前端控制器配置(在web.xml中)
<!-- 前端控制器 -->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 加载springmvc配置 -->
<init-param>
<param-name>contextConfigLocation</param-name>
<!-- 配置文件的地址 如果不配置contextConfigLocation, 默认查找的配置文件名称classpath下的:servlet名称+"-serlvet.xml"即:springmvc-serlvet.xml -->
<param-value>classpath:spring/springmvc.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<!-- 可以配置/ ,此工程 所有请求全部由springmvc解析,此种方式可以实现 RESTful方式,需要特殊处理对静态文件的解析不能由springmvc解析
可以配置*.do或*.action,所有请求的url扩展名为.do或.action由springmvc解析,此种方法常用 不可以/*,如果配置/*,返回jsp也由springmvc解析,这是不对的。 -->
<url-pattern>*.action</url-pattern>
</servlet-mapping>
4.配置springmvc.xml
<!-- 使用spring组件扫描 -->
<context:component-scan base-package="com.yuhan.ssm.controller" />
<!-- 注解处理器映射器 -->
<bean
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping">
</bean>
<!-- 注解适配器 -->
<bean
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
</bean>
<!-- 配置视图解析器 要求将jstl的包加到classpath -->
<!-- ViewResolver -->
<bean
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
</bean>
5.在applicationContext-service.xml中配置service
<bean id="userService" class="com.yuhan.ssm.service.imp.UserServiceImp"></bean>
6.Controller
@Controller
public class MyController {
@Resource(name="userService")
private UserService userService;
//尽量保证名称和方法名一致,方便管理和维护
@RequestMapping("findByName")
public ModelAndView findByName(){
List<User> users = userService.findUserByName();
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("users",users);
modelAndView.setViewName("userList");
return modelAndView;
}
}
@Controller
注册自定义controller
@RequestMapping("name")
设置方法对应的url,一个方法对应一个url
窄化请求映射
好处:更新规范系统 的url,避免 url冲突。
@Controller
//定义url的根路径,访问时根路径+方法的url
@RequestMapping("/items")
public class ItemsController {
@RequestMapping("/findByName")
public ModelAndView findByName(){
...
}
...
}
url为/items/findByName
7.jsp
8.在web.xml配置spring监听器
<!-- 配置spring容器监听器 -->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/applicationContext-*.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
controller方法返回值
返回ModelAndView
@RequestMapping("findByName")
public ModelAndView findByName(){
List<User> users = userService.findUserByName();
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("users",users);
modelAndView.setViewName("userList");
return modelAndView;
}
返回字符串
//方法返回 字符串,字符串就是逻辑视图名,Model作用是将数据填充到request域,在页面展示
@RequestMapping(value="/editItems",method={RequestMethod.GET})
public String editItems(Model model,Integer id)throws Exception{
//调用 service查询商品信息
ItemsCustom itemsCustom = itemsService.findItemsById(id);
model.addAttribute("item", itemsCustom);
return "editItem";
}
redirect重定向
如果方法重定向到另一个url,方法返回值为“redirect:url路径”
使用redirect进行重定向,request数据无法共享,url地址栏会发生变化的.
forward转发
使用forward进行请求转发,request数据可以共享,url地址栏不会改变.
方法返回值为“forward:url路径”
返回void
//方法返回void
@RequestMapping(value="/editItems",method={RequestMethod.GET})
public void editItems(HttpServletRequest request, HttpServletResponse response, Integer id)
throws Exception {
//调用 service查询商品信息
ItemsCustom itemsCustom = itemsService.findItemsById(id);
request.setAttribute("item", itemsCustom);
//注意如果使用request转向页面,这里指定页面的完整路径
request.getRequestDispatcher("/WEB-INF/jsp/editItem.jsp").forward(request, response);
}
使用此方法,容易输出json、xml格式的数据:
通过response指定响应结果,例如响应json数据如下:
response.setCharacterEncoding("utf-8");
response.setContentType("application/json;charset=utf-8");
response.getWriter().write("json串");
参数绑定
参数绑定过程
Handler(Controller)方法默认支持的参数类型
处理器形参中添加如下类型的参数处理适配器会默认识别并进行赋值。
1.HttpServletRequest
通过request对象获取请求信息
2.HttpServletResponse
通过response处理响应信息
3.HttpSession
通过session对象得到session中存放的对象
4.Model
通过model向request域中存入数据
model.addAttribute("#",xxx);
5.@RequestParam
@RequestMapping("editItems")
public void editItems(
HttpServletRequest request,
HttpServletResponse response,
@RequestParam(value = "item_id", required = false, defaultValue = "1") Integer id
Integer id
){...}
如果request请求的参数名和controller方法的形参数名称一致,适配器自动进行参数绑定。如果不一致可以通过
@RequestParam 指定request请求的参数名绑定到哪个方法形参上.
对于必须要传的参数,通过@RequestParam中属性required设置为true,如果不传此参数则报错.
对于有些参数如果不传入,还需要设置默认值,使用@RequestParam中属性defaultvalue设置默认值.
6.可以绑定简单类型
可以绑定整型、 字符串、单精/双精度、日期、布尔型.
7.可以绑定简单pojo类型
简单pojo类型只包括简单类型的属性。
绑定过程:
request请求的参数名称和pojo的属性名一致,就可以绑定成功。
问题:
如果controller方法形参中有多个pojo且pojo中有重复的属性,使用简单pojo绑定无法有针对性的绑定,
比如:方法形参有items和User,pojo同时存在name属性,从http请求过程的name无法有针对性的绑定到items或user.
8.可以绑定包装的pojo
包装的pojo里边包括了pojo。
页面参数定义:
<tr>
<td>商品名称</td>
<td><input type="text" name="itemsCustom.name" value="${item.name }"/></td>
</tr>
<tr>
<td>商品价格</td>
<td><input type="text" name="itemsCustom.price" value="${item.price }"/></td>
</tr>
<tr>
<td>商品简介</td>
<td>
<textarea rows="3" cols="30" name="itemsCustom.detail">${item.detail }</textarea>
</td>
</tr>
包装类型的属性也是itemsCustom:
public class ItemsQueryVo {
//商品信息
private ItemsCustom itemsCustom;
...
}
即在绑定的简单属性名称前加上po的名称