springmv+mybatis整合以及遇到的问题

其实网上已经有不少文章描述sm整合,但个人感觉,它们或多或少有些细节没有交代清楚吧,本篇文章就是针对这些细节来进行详解~

首先是项目结构图:


springmv+mybatis整合以及遇到的问题_第1张图片
 

WebContent目录结构图:


springmv+mybatis整合以及遇到的问题_第2张图片
 

User类:


springmv+mybatis整合以及遇到的问题_第3张图片
 

 

UserMapper以及映射文件:


springmv+mybatis整合以及遇到的问题_第4张图片
 


springmv+mybatis整合以及遇到的问题_第5张图片
 

其中id需要与UserMapper的方法名一致,返回类型resultType以及参数类型parameterType无需多说,需要注意的是findUsers方法,在映射文件中声明了返回

类型是User,而UserMapper返回的是list<User>,实际上这是支持的。

接口IUserService以及它的实现类:


springmv+mybatis整合以及遇到的问题_第6张图片
 

package com.lee.sm.serviceImpl;

import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import com.lee.sm.mapper.UserMapper;
import com.lee.sm.pojo.User;

//默认将类中的所有函数纳入事务管理.   
//@Transactional("user")
@Service("iUserServiceImpl")
@Transactional
public class IUserServiceImpl implements IUserService {

 @Autowired
 private UserMapper userMapper;
 
 /**
  * 检查用户登录密码
  * */
 @Override
 public boolean authUser(User user) {
  String passWord = userMapper.showPassword(user);
  if(passWord != null && user.getPassword() != null && passWord.equals(user.getPassword())){
   return true;
  }
  return false;
 }

 @Override
 public String showUserList() {
   List<User> users = userMapper.findUsers();
   for(User user : users){
    System.out.println(user);
   }
   return "测试完毕";
 }

 @Override
 public String showStory(User user) {
  String userStory = "";
  userStory = userStory + "address=[" + userMapper.showAddress(user) + "]";
  return userStory;
 }

 @Override
 public String createUser(User user) throws RuntimeException{
  userMapper.insertUser(user);
  System.out.println("准备抛出异常");
  //在插入成功后,抛出一个异常,测试事物是否回滚
  try {
   int i = 1/0;
  } catch (Exception e) {
   throw new RuntimeException();  
  }
  return null;
 }

 public UserMapper getUserMapper() {
  return userMapper;
 }

 public void setUserMapper(UserMapper userMapper) {
  this.userMapper = userMapper;
 }
 
 
}

原本不是@Service("iUserServiceImpl")而是@Service这样注解的,结果在创建LoginController时报错说,自动注入IUserService时找不到这个bean,

话说@Service不是自动注册为首字母小写的bean吗?....求大神指点......

页面跳转的控制器LoginController:

package com.lee.sm.controller;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

import com.lee.sm.pojo.User;
import com.lee.sm.serviceImpl.IUserService;

@Controller
@RequestMapping("/mine")
public class LoginController {
	
	@Autowired
	@Qualifier("iUserServiceImpl")
	private IUserService iUserService;
	
	@RequestMapping
	public String mine(){
		System.out.println("in the mine()...");
		User user = new User();
		user.setName("003");
		String mineStory = iUserService.showStory(user);
		System.out.println(mineStory);
		return "main2";
	}
	
	@RequestMapping(value="/findFriends")
	public String findFriends() {
		System.out.println("in the findFriends()");
		System.out.println(iUserService.showUserList());
		User user = new User("阿诺", "40", "美国加州", 1, "123");
		iUserService.createUser(user);
		System.out.println("已插入新建用户");
		
		
		return "main2";
	}

	public IUserService getiUserService() {
		return iUserService;
	}

	public void setiUserService(IUserService iUserService) {
		this.iUserService = iUserService;
	}
	
	
	
}

 

web.xml:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
  <display-name>MySpringMVCMybatis</display-name>
  
  <context-param>
   <param-name>contextConfigLocation</param-name>
   <param-value>classpath:config/applicationContext.xml</param-value>
  </context-param>
  
  <listener>
   <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  
  <!-- spring提供转码 -->
 <filter>
        <filter-name>characterEncoding</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>characterEncoding</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
    
  <servlet>
        <servlet-name>woder</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>woder</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
  
  
</web-app>


 

applicationContext.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns:cxt="http://www.springframework.org/schema/context" 
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans 
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd 
http://www.springframework.org/schema/context 
http://www.springframework.org/schema/context/spring-context-2.5.xsd 
http://www.springframework.org/schema/tx 
http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">

 <!-- 注释此项,因为后面已使用cxt:component-scan -->
    <!-- <cxt:annotation-config /> -->
     
    <cxt:component-scan base-package="com.lee.sm.serviceImpl">           
  <!-- <cxt:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> -->
 </cxt:component-scan>
 
    
    <bean id="propertyConfig" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">  
    <property name="locations">  
             <list>  
                 <value>classpath:config/jdbc.properties</value>  
             </list>  
    </property>  
    </bean>
    
    
 <bean id="myDataSource" class="org.apache.commons.dbcp.BasicDataSource">
        <property name="driverClassName" value="${jdbc.driverClassName}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
        <property name="maxActive" value="${jdbc.maxActive}"/>
        <property name="maxIdle" value="${jdbc.maxIdle}"/>
        <property name="maxWait" value="${jdbc.maxWait}"/>
        <property name="defaultAutoCommit" value="${jdbc.defaultAutoCommit}"/>
    </bean>

    
     <!-- define the SqlSessionFactory -->
    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="myDataSource" />
         <property name="configLocation" value="classpath:config/mybatis-config.xml"/> 
    </bean>
    
 
 <!-- 开启事务注解驱动 -->  
 <tx:annotation-driven />
 
 
  <!-- Transaction Manager -->
  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  <property name="dataSource" ref="myDataSource" />
  <!-- 如果是user表,用@Transactional("user"),如果是smis用 @Transactional("smis")管理事务-->
  </bean>
    
    
    <!-- scan for mappers and let them be autowired -->
    <!-- 使用自动扫描包的方式来注册各种mapper -->
    <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
        <property name="basePackage" value="com.lee.sm.mapper" />
    </bean>
    
</beans>

 

 

woder-servlet.xml,相当于另一些文章说的sprigmvc-servlet.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns:mvc="http://www.springframework.org/schema/mvc" 
xmlns:cxt="http://www.springframework.org/schema/context" 
xsi:schemaLocation="http://www.springframework.org/schema/beans 
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
http://www.springframework.org/schema/mvc 
http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd 
http://www.springframework.org/schema/context 
http://www.springframework.org/schema/context/spring-context-3.0.xsd ">
	
	<!-- 启动包扫描功能,以便注册带有@Controller、@Service、@repository、@Component等注解的类成为spring的bean -->
	<cxt:component-scan base-package="com.lee.sm.controller">
		<!-- <cxt:exclude-filter type="annotation" expression="org.springframework.stereotype.Service"/> -->
	</cxt:component-scan>
	
	<!--  主要作用于@Controller,激活该模式,下面是一种简写形式,完全可以手动配置替代这种简写形式,它会自动注册DefaultAnnotationHandlerMapping与AnnotationMethodHandlerAdapter,是spring MVC为@Controllers分发请求所必须的   -->
	<mvc:annotation-driven/>
	
	
	
	<!-- 对模型视图名称的解析,在请求时模型视图名称添加前后缀 -->
	<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
		<!-- <property name="cache" value="true"/> -->
		
		<!-- viewClass属性可以用来指定前台在解析数据时,所允许采用的手段。实际上其默认值就是JstlView --> 
		<!-- 将来有需要的话,就可以在这里把JstlView改成其它的,如FreeMarkerView,VelocityView,TilesView --> 
		<!-- <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/> -->
		
		<property name="prefix" value="/WEB-INF/view/"/>
		<property name="suffix" value=".jsp"/>
		<property name="contentType" value="text/html;charset=UTF-8"/>
	</bean>
	
	<!-- <mvc:view-controller path="/" view-name="redirect:view/main.jsp"/> -->
	<mvc:view-controller path="/" view-name="main"/> 
	<!--  <mvc:view-controller path="*.jsp"/>-->
</beans>

 

这里涉及到了mybatis事务无法回滚的问题

      原本我是在woder-servlet.xml中使用<cxt:component-scan base-package="com.lee.sm.**">来加载位于该目录下(包括子目录下的所有)所有标注了注解的类的(比如@Controller/@Service),结果mybatis无法回滚

      在试过了把@Transactional改成@Transactional(rollbackFor=Exception.class)、查看mysql引擎是否为innodb之后,仍然无效。

      最后是把service和controller分别在applicationContext.xml及woder-servlet.xml中加载,才得以解决——这里不得不吐槽某篇文章,因为实际上applicationContext.xml是先于springmvc-servlet.xml加载的,所以service应该在applicationContext.xml中加载,然后springmvc-servlet.xml再加载controller——

      那篇文章所叙述的刚好相反...我是在好几次构建controller报错找不到对应的bean注入(service这个bean还没创建)我才醒悟的TnT...

mybatis-config.xml:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
  PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
 
 <typeAliases>
  <!-- <package name="com.chenliang.mvc.entity"/> -->
  <typeAlias alias="BaseMapper" type="com.lee.sm.baseMapper.BaseMapper"/>
  <typeAlias alias="UserMapper" type="com.lee.sm.mapper.UserMapper"/>
  <typeAlias type="com.lee.sm.pojo.User" alias="User"/>
 </typeAliases>

  
  <mappers>
    <!-- <mapper class="com.lee.sm.mapper.*"/> -->
    <mapper resource="com/lee/sm/mapper/UserMapper.xml"/>
  </mappers>
  
</configuration>


jdbc.properties:

jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/SPRINGMVC?useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull
jdbc.username=root
jdbc.password=root
jdbc.maxActive=100
jdbc.maxIdle=30
jdbc.maxWait=500
jdbc.defaultAutoCommit=true


这个项目使用的jar包有

mybatis-3.0.6.jar

mybatis-spring-1.0.1-sources.jar

spring-core-3.0.5.RELEASE.jar

spring-webmvc-3.1.1.RELEASE.jar

 

 

 



 

 

你可能感兴趣的:(spring,mybatis,spring+mybatis)