使用Spring MVC和Mybatis也做過幾個項目了,一直沒有整理過搭建過程,是以最近重新開始搞了一個最最基礎的example,隻有一個登陸,但是基礎配置還是相對完整的,可供新人參考思路。該項目使用maven進行管理,如果有不了解的地方可自行搜尋相關文檔學習一下,也可去http://blog.csdn.net/qq_19656425找從零開始系列部落格進行了解。
首先建立一個空的maven項目,在pom檔案中配置好相關依賴。
配置web.xml檔案
<!-- 該listener可視情況選擇是否配置 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring-config.xml</param-value>
</context-param>
<!-- html頁面和靜态資源交給容器(Tomcat、Jetty等)處理 -->
<servlet-mapping>
<!-- 容器預設的servlet名 -->
<servlet-name>default</servlet-name>
<url-pattern>*.html</url-pattern>
<url-pattern>/static/</url-pattern>
</servlet-mapping>
<!-- 該項目中使用rest servlet攔截restful風格的請求,統一傳回json字元串或json對象 -->
<servlet>
<servlet-name>rest</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:rest-servlet.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>rest</servlet-name>
<!-- 如果通路/rest/login,會截掉字首使用/login進行requestMapping的比對 -->
<url-pattern>/rest/*</url-pattern>
</servlet-mapping>
<!-- 該servlet攔截的請求傳回jsp頁面 -->
<servlet>
<servlet-name>pages</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:pages-servlet.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>pages</servlet-name>
<url-pattern>/pages/*</url-pattern>
</servlet-mapping>
web.xml的主要配置就這麼多,rest和pages兩個servlet可以根據需要一般隻配置一個統一的即可。
spring-config.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:tx="http://www.springframework.org/schema/tx"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mybatis="http://mybatis.org/schema/mybatis-spring"
xsi:schemaLocation="http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://mybatis.org/schema/mybatis-spring
http://mybatis.org/schema/mybatis-spring.xsd">
<!-- 開始Spring MVC功能 -->
<mvc:annotation-driven/>
<context:component-scan base-package="com.heuman.service"/>
<!-- 掃描base-package包下的接口并注冊成bean -->
<mybatis:scan base-package="com.heuman.mapper"/>
<bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<!-- 資料庫配置資源檔案 -->
<value>classpath:properties/db.properties</value>
</list>
</property>
</bean>
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="typeAliasesPackage" value="com.heuman.vo"/>
<property name="mapperLocations" value="classpath*:com/heuman/mapper/xml/**/*.xml"/>
<!--<property name="typeHandlersPackage" value="com.heuman.config.mybatis.typehandler"/>-->
<property name="configuration">
<bean class="org.apache.ibatis.session.Configuration">
<property name="mapUnderscoreToCamelCase" value="true"/>
</bean>
</property>
</bean>
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" destroy-method="close">
<!-- 資料庫基本資訊配置,可以使用${key}從上面配置的資源檔案中拿對應的value -->
<property name="url" value="${url}"/>
<property name="username" value="${username}"/>
<property name="password" value="${password}"/>
<property name="driverClassName" value="${driverClassName}"/>
<property name="maxActive" value="${maxActive}"/>
<property name="initialSize" value="${initialSize}"/>
<property name="maxWait" value="${maxWait}"/>
<property name="minIdle" value="${minIdle}"/>
<!-- 配置間隔多久才進行一次檢測,檢測需要關閉的空閑連接配接,機關是毫秒 -->
<property name="timeBetweenEvictionRunsMillis" value="${timeBetweenEvictionRunsMillis}"/>
<!-- 配置一個連接配接在池中最小生存的時間,機關是毫秒 -->
<property name="minEvictableIdleTimeMillis" value="${minEvictableIdleTimeMillis}"/>
<property name="validationQuery" value="${validationQuery}"/>
</bean>
<!-- 配置事務 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!-- 使用@Transactional開啟事務(注:方法修飾符必須為public) -->
<tx:annotation-driven transaction-manager="transactionManager"/>
</beans>
db.properties
url=jdbc:mysql://localhost:/test
driverClassName=com.mysql.jdbc.Driver
username=root
password=admin
maxActive=
initialSize=
maxWait=
minIdle=
timeBetweenEvictionRunsMillis=
minEvictableIdleTimeMillis=
validationQuery=SELECT
rest-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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd">
<mvc:annotation-driven>
<!-- 使用fastjson對json(content-type為application/json)進行解析(Spring MVC預設使用jackson) -->
<mvc:message-converters register-defaults="true">
<bean class="com.alibaba.fastjson.support.spring.FastJsonHttpMessageConverter4">
<property name="fastJsonConfig">
<bean class="com.alibaba.fastjson.support.config.FastJsonConfig">
<property name="dateFormat" value="yyyy-MM-dd"/>
</bean>
</property>
<property name="supportedMediaTypes">
<list>
<bean class="org.springframework.http.MediaType">
<constructor-arg type="java.lang.String" value="application"/>
<constructor-arg type="java.lang.String" value="json"/>
</bean>
</list>
</property>
</bean>
</mvc:message-converters>
</mvc:annotation-driven>
<context:component-scan base-package="com.heuman.controller.rest"/>
</beans>
pages-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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd">
<mvc:annotation-driven/>
<context:component-scan base-package="com.heuman.controller.page"/>
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
<property name="prefix" value="/WEB-INF/pages/"/>
<property name="suffix" value=".jsp"/>
</bean>
</beans>
LoginController.java
@Controller
public class LoginController {
private final transient Logger logger = LoggerFactory.getLogger(LoginController.class);
@Autowired
private UserService userService;
@RequestMapping(value = "/login", method = RequestMethod.POST, consumes = {"application/json"})
@ResponseBody
public Object login(@RequestBody User user, HttpSession session) {
if (logger.isDebugEnabled()) {
logger.debug(user.toJson());
}
// 已經登入
if (session.getAttribute("user") != null) {
return Result.newSuccessResult().toJson();
}
String errorMsg;
if (StringUtils.isBlank(user.getUsername()) || StringUtils.isBlank(user.getPassword())) {
errorMsg = "使用者名和密碼不能為空";
} else {
User u = userService.findByUsernamePassword(user);
if (u == null) {
errorMsg = "使用者名或密碼錯誤";
} else {
if (logger.isInfoEnabled()) {
logger.info("####### {} login #######", u.getUsername());
}
session.setAttribute("user", u);
return Result.newSuccessResult().toJson();
}
}
Result result = Result.newErrorResult();
result.setMsg(errorMsg);
return result.toJson();
}
@RequestMapping(value = "/logout")
public String logout(HttpSession session) {
session.removeAttribute("user");
return PathUtil.append("redirect:", Config.getPagesPath(), "/index");
}
}
注:toJson方法是自定義的對象轉json
UserService.java
@Service
public class UserService {
private final transient Logger logger = LoggerFactory.getLogger(UserService.class);
@Autowired
private UserMapper userMapper;
public User findByUsernamePassword(User user) {
return userMapper.findByUsernameAndPassword(user.getUsername(), user.getPassword());
}
}
UserMapper.java
@Mapper
public interface UserMapper {
//也可不用@Param,而使用#{param1},#{param2}...去擷取參數
@Select("select * from user where username=#{username} and password=#{password}")
User findByUsernameAndPassword(@Param("username") String username, @Param("password") String password);
}
HomeController.java
@Controller
public class HomeController {
private final transient Logger logger = LoggerFactory.getLogger(HomeController.class);
@RequestMapping("/index")
public String index(HttpSession session, Model model) {
if (session.getAttribute("user") != null) {
model.addAttribute("username", ((User)session.getAttribute("user")).getUsername());
return "index";
}
return "logout";
}
}
login.html頁面是登入頁面,當使用者登入時通過ajax通路/rest/login頁面,如果傳回結果表示登入成功,則轉到/pages/index,HomeController類中的index方法接收到該請求把username放到request中,接着轉到/WEB-INF/pages/index.jsp頁面。
index.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html lang="zh-cmn-Hans">
<head>
<title>首頁</title>
</head>
<body>
<div>
<h4><%=request.getAttribute("username")%>, HelloWorld welcome you!</h4>
</div>
</body>
</html>
代碼已上傳到github:https://github.com/heumanpub/ssm-example,細節解釋的不多,不了解的地方可以google。