目录
一、概述
学习本章之前,我们应该对SpringMVC
, Spring
, Mybatis
框架有一个完整的了解,并熟练使用单独框架进行简单的代码开发,如果不能做到,请返回到相关课件页面进行学习,再进入当前页学习本章内容,谢谢。
此课件基于Maven
创建项目,如果对于Maven
工具不是很了解,请点击链接学习相关课件
本章框架集成所使用的软件版本要求如下:
软件 | 版本 |
---|---|
SpringMVC | 4.0.0.RELEASE |
Spring | 4.0.0.RELEASE |
Mybatis | 3.2.8 |
C3P0连接池 | 0.9.2 |
MYSQL驱动 | 5.1.37 |
二、创建Maven Web项目
在Spring Tool Suite
中创建Maven
项目,生成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" 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">
</web-app>
web.xml
是整个项目的核心配置文件,也可以理解为Web程序访问的入口,非常重要
在servlet3.0
及后续版本中,此配置文件可省略,采用注解方式替代,本课程暂不涉及。
在项目的pom.xml
文件中增加依赖关系:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" <dependencies> <!-- Spring整合MyBatis --> <!-- 控制日志输出:结合log4j --> <dependency> <!-- 石英调度 - 开始 --> <dependency> <dependency> <dependency> <dependency>
</project> |
三、集成Spring框架
Spring
框架是整个系统架构的核心,将前端请求数据的处理以及数据库的数据操作整合在一起,非常重要。
在web.xml
文件中增加配置信息集成Spring
框架
<web-app>
...
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:spring/spring-*.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
...
</web-app>
Spring
环境构建时需要读取web
应用的初始化参数contextConfigLocation
, 从classpath
中读取配置文件spring/spring-*.xml
在项目src/main/resources
目录中增加spring
文件夹,并在其中增加spring-context.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:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.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-4.0.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">
</beans>
Spring
框架的核心是构建对象,整合对象之间的关系(IOC
)及扩展对象功能(AOP
),所以需要在spring-context.xml
配置文件中增加业务对象扫描的相关配置。扫描后由Spring
框架进行管理和组合。
<beans>
...
<context:component-scan base-package="com.uwl.*" >
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
...
</beans>
扫描配置中为什么要排除Controller
注解
Controller
注解的的作用是声明控制器(处理器)类。
从数据流转的角度,这个类应该是由SpringMVC
框架进行管理和组织的,所以不需要由Spring
框架扫描。
四、集成SpringMVC框架
SpringMVC
框架用于处理系统中数据的流转及控制操作。
(从哪里来,到哪里去。多么有哲理的一句话。)
集成SpringMVC
框架,需要在web.xml
文件中增加配置信息
<web-app>
...
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/springmvc-context.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
...
</web-app>
SpringMVC
环境构建时需要读取servlet
初始化参数init-param
, 从classpath
中读取配置文件spring/springmvc-context.xml
在项目src/main/resources/spring
目录中,增加springmvc-context.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:context="http://www.springframework.org/schema/context"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">
</beans>
SpringMVC
框架的核心是处理数据的流转,所以需要在springmvc-context.xml
配置文件中增加控制器对象(Controller
)扫描的相关配置。扫描后由SpringMVC
框架进行管理和组合。
<beans>
...
<context:component-scan base-package="com.uwl.*" use-default-filters="false" >
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
...
</beans>
静态资源如何不被SpringMVC
框架进行拦截
在配置文件中增加<mvc:default-servlet-handler/>
, <mvc:annotation-driven />
即可
在实际的项目中静态资源不会和动态资源放在一起,也就意味着不会放置在服务器中,所以这些配置可以省略。
如果SpringMVC
框架数据处理为页面跳转,那么需要配置相应的视图解析器ViewResolver
。
<beans>
...
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" >
<property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/>
<property name="prefix" value="/WEB-INF/jsp/"/>
<property name="suffix" value=".jsp"/>
</bean>
...
</beans>
如果有多个视图解析器怎么办?
SpringMVC
框架中允许存在多个视图解析器,框架会按照配置声明顺序,依次进行解析。
SpringMVC
框架中配置多个视图解析器时,如果将InternalResourceViewResolver
解析器配置在前,那么即使找不到视图,框架也不会继续解析,直接发生404
错误,所以必须将InternalResourceViewResolver
解析器放置在最后。
如果SpringMVC
框架数据处理为响应JSON
字符串,那么为了浏览器方便对响应的字符串进行处理,需要明确字符串的类型及编码方式。
如果增加了<mvc:annotation-driven />
标签,下面的配置可省略。
<beans>
...
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" >
<property name="messageConverters" >
<list>
<bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter" >
<property name="supportedMediaTypes" >
<list>
<value>application/json;charset=UTF-8</value>
</list>
</property>
</bean>
</list>
</property>
</bean>
...
</beans>
如果项目中含有文件上传业务,还需要增加文件上传解析器MultipartResolver
<beans>
...
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver" p:defaultEncoding="UTF-8" >
<property name="maxUploadSize" value="2097152"/>
<property name="resolveLazily" value="true"/>
</bean>
...
</beans>
五、集成Mybatis框架
Mybatis
框架主要处理业务和数据库之间的数据交互,所以创建对象和管理对象生命周期的职责可以委托Spring
框架完成。如:创建Mybatis
核心对象。
<beans>
...
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean" >
<property name="configLocation" value="classpath:mybatis/config.xml" />
<property name="dataSource" ref="dataSource" />
<property name="mapperLocations" >
<list>
<value>classpath*:mybatis/mapper-*.xml</value>
</list>
</property>
</bean>
...
<bean id="mapperScannerConfigurer" class="org.mybatis.spring.mapper.MapperScannerConfigurer" >
<property name="basePackage" value="com.uwl.atcrowdfunding.**.dao" />
</bean>
...
</beans>
既然需要和数据库进行关联,那么Mybatis
核心对象就需要依赖于数据库连接池(C3P0
),所以在Spring
配置文件中增加相应的配置。
<beans>
...
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" >
<property name="driverClass" value="com.mysql.jdbc.Driver"/>
<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/atcrowdfunding?rewriteBatchedStatements=true&useUnicode=true&characterEncoding=utf8"/>
<property name="user" value="root"/>
<property name="password" value="root"/>
</bean>
...
</beans>
集成Mybatis
框架时同时还需要增加核心配置文件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>
...
</typeAliases>
</configuration>
及SQL
映射文件mybatis/mapper-*.xml
。
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="xxx.XXDao" >
...
</mapper>
为了保证数据操作的一致性,必须在程序中增加事务处理。Spring
框架采用声明式事务,通过AOP
的方式将事务增加到业务中。所以需要在Spring
配置文件中增加相关配置
<beans>
...
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager" >
<property name="dataSource" ref="dataSource"/>
</bean>
<tx:advice id="transactionAdvice" transaction-manager="transactionManager" >
<tx:attributes>
<tx:method name="*" propagation="REQUIRED" isolation="DEFAULT" rollback-for="java.lang.Exception" />
<tx:method name="query*" read-only="true" />
</tx:attributes>
</tx:advice>
<aop:config>
<aop:advisor advice-ref="transactionAdvice" pointcut="execution(* com.uwl..*Service.*(..))"/>
</aop:config>
...
</beans>
测试前,需要在数据库中增加atcrowdfunding
库及t_user
表。
CREATE DATABASE `atcrowdfunding`;
...
CREATE TABLE `t_user` (
`id` int(11) NOT NULL AUTO_INCREMENT
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8;
六、框架集成测试
在src/main/java/com/uwl/atcrowdfunding/controller
目录中增加UserController
package com.uwl.atcrowdfunding.controller;
import java.util.HashMap;
import java.util.Map;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping("/index")
public String index() {
return "user/index";
}
@ResponseBody
@RequestMapping("/json")
public Object json() {
Map map = new HashMap();
map.put("username", "张三");
return map;
}
}
将web
项目发布到服务器中,启动服务器,浏览器中分别输入路径http://127.0.0.1:8080/应用路径名称/user/index
,http://127.0.0.1:8080/应用路径名称/user/json
进行测试。
如果访问成功,说明项目中SpringMVC
框架集成成功。
在src/main/java/com/uwl/atcrowdfunding/service
目录中增加UserService
接口。
package com.uwl.atcrowdfunding.service;
public interface UserService {
}
在src/main/java/com/uwl/atcrowdfunding/service/impl
目录中增加UserServiceImpl
实现类。
package com.uwl.atcrowdfunding.service.impl;
import com.atguigu.atcrowdfunding.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class UserServiceImpl implements UserService {
@Autowired
private UserDao userDao;
}
在src/main/java/com/uwl/atcrowdfunding/dao
目录中增加UserDao
接口。
package com.uwl.atcrowdfunding.dao;
public interface UserDao {
}
修改UserController
类,增加对UserService
接口的引用。
package com.uwl.atcrowdfunding.controller;
import java.util.HashMap;
import java.util.Map;
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.ResponseBody;
@Controller
@RequestMapping("/user")
public class UserController {
@Autowired
private UserService userService;
@RequestMapping("/index")
public String index() {
return "user/index";
}
@ResponseBody
@RequestMapping("/json")
public Object json() {
Map map = new HashMap();
map.put("username", "张三");
return map;
}
}
重启服务器,重新通过浏览器中访问路径http://127.0.0.1:8080/应用路径名称/user/index
进行测试。
如果访问成功,说明项目中Spring
框架(IOC
功能)集成成功。
在src/main/java/com/uwl/corwdfunding/bean
目录中增加User
实体类。
package com.uwl.atcrowdfunding.bean;
public class User {
}
在UserService
接口中增加方法声明queryById
,并在UserServiceImpl
类中默认实现
package com.atguigu.atcrowdfunding.service;
import com.atguigu.atcrowdfunding.bean.User;
public interface UserService {
public User queryById();
}
package com.uwl.atcrowdfunding.service.impl;
import com.atguigu.atcrowdfunding.bean.User;
import com.atguigu.atcrowdfunding.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class UserServiceImpl implements UserService {
@Autowired
private UserDao userDao;
public User queryById() {
return userDao.queryById();
}
}
如果访问成功,说明项目中Spring
框架(AOP
功能)集成成功。
在UserDao
中增加查询语句,实现数据库查询
package com.uwl.atcrowdfunding.dao;
import com.uwl.atcrowdfunding.bean.User;
import org.apache.ibatis.annotations.Select;
public interface UserDao {
@Select("select * from t_user where id = 1")
public User queryById();
}
如果访问成功,说明项目中Mybatis
框架集成成功。
七、增加quarz
在web.xml中增加如下内容:
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:spring/spring-*.xml,classpath:quarz/quarz-config.xml</param-value>
</context-param>
创建src/main/resources/quarz/quarz-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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.1.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.1.xsd"
default-lazy-init="true">
<!-- 密码过期提醒任务调度 -->
<bean id="jobPasswordExpired" class="com.uwl.quartz.PasswordExpiredQuartz" />
<bean id="jobPasswordExpiredDetail"
class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<property name="targetObject" ref="jobPasswordExpired" />
<property name="targetMethod" value="doTask" />
<!-- 是否支持并发 true 支持; false不支持,等上一个调度任务执行完才会执行,如果间隔时间小于任务执行时间第二个调度任务启动但会推迟执行 -->
<property name="concurrent" value="false" />
</bean>
<bean id="jobPasswordExpiredTrigger"
class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
<property name="jobDetail" ref="jobPasswordExpiredDetail" />
<!-- 每隔30秒执行一次:0 */30 * * * ? -->
<property name="cronExpression" value="*/5 * * * * ?" />
</bean>
<!-- 总管理类如果将lazy-init='false'那么容器启动就会执行调度程序 -->
<bean id="startQuertz"
class="org.springframework.scheduling.quartz.SchedulerFactoryBean"
lazy-init="false">
<property name="triggers">
<!-- 作业调度器,list下可加入其他的调度器 -->
<list>
<ref bean="jobPasswordExpiredTrigger" />
</list>
</property>
</bean>
</beans>
创建/uwlssm/src/main/java/com/uwl/quartz/PasswordExpiredQuartz.java文件:
package com.uwl.quartz;
import org.apache.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import com.alibaba.fastjson.JSON;
import com.uwl.atcrowdfunding.bean.User;
import com.uwl.atcrowdfunding.service.UserService;
public class PasswordExpiredQuartz {
@Autowired
private UserService userService;
private static final Logger log = Logger.getLogger(PasswordExpiredQuartz.class);
public void doTask() {
log.info("密码过期提醒任务调度开始执行");
User user = userService.queryById();
log.info(JSON.toJSONString(user));
log.info("密码过期提醒任务调度执行结束");
}
}
增加/uwlssm/src/main/resources/log4j.properties文件:
#定义LOG输出级别
log4j.rootLogger=INFO,Console,File
#定义日志输出目的地为控制台
log4j.appender.Console=org.apache.log4j.ConsoleAppender
log4j.appender.Console.Target=System.out
#可以灵活地指定日志输出格式,下面一行是指定具体的格式
log4j.appender.Console.layout=org.apache.log4j.PatternLayout
log4j.appender.Console.layout.ConversionPattern=[%c] - %m%n
#日志输出级别
log4j.appender.Console.Threshold=ALL
#定义日志输出文件
#指定输出目录, #这个目录是在你tomcat下。实际路径:../tomcat/logs/ssm-base.log
log4j.appender.File.File=../logs/ssm-base.log
#定义文件最大大小
log4j.appender.File.MaxFileSize=10MB
#文件大小到达指定尺寸的时候产生一个新的文件
log4j.appender.File=org.apache.log4j.RollingFileAppender
#日志输出级别
log4j.appender.File.Threshold=ALL
#设置日志文件编码
log4j.appender.File.Encoding=utf-8
#日志信息布局
log4j.appender.File.layout=org.apache.log4j.PatternLayout
#日志格式
log4j.appender.File.layout.ConversionPattern=[%p] [%d{yyyy-MM-dd HH\:mm\:ss}][%c]%m%n
#打印mybatis sql语句
log4j.logger.com.uwl=DEBUG
八、项目完整截图