Spring注解开发
- Spring是轻代码而重配置的框架,配置比较繁重,影响开发效率,所以注解开发是一种趋势,它能够代替xml配置文件,可以简化配置,提高开发效率
- Spring注解根据出现时间分类
- Spring原始注解:主要代替
<bean>
的配置 - Spring新注解:Spring配置文件中有些配置是Spring原始注解无法替代的,此时就需要新注解,比如:
- 非自定义的Bean的配置,比如Druid、Cp30的bean
- 加载properties配置文件的配置:
<context:property-placeholder location="classpath:jdbc.properties"/>
- 组件扫描的配置:
<context:component-scan base-package="at.guigu"></context:component-scan>
- 引入其它配置文件的配置:
<import resource="applicationContext-xxx.xml"/>
- Spring原始注解:主要代替
- 注意
- 使用注解开发时,需要在Spring配置文件中配置组件扫描
- 作用:指定哪个包及其子包下的bean需要进行扫描,以便可以识别使用注解配置的类、字段和方法
- 使用注解开发时,需要在Spring配置文件中配置组件扫描
Spring原始注解快速入门
-
不使用注解开发时的代码截图如下(以
setter
方法注入为例) -
使用注解开发的步骤
-
Step1: 在pom.xml文件中导入相应的坐标及插件:Spring坐标、Annotation坐标、Tomcat插件。pom.xml文件完整代码如下
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.example</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>SpringAnno</artifactId> <packaging>war</packaging> <name>SpringAnno Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> </dependencies> <build> <finalName>SpringAnno</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
-
Step2: 给对应的类添加注解
-
UserDaoImpl
类代码如下package at.guigu.dao.impl; import at.guigu.dao.UserDao; import org.springframework.stereotype.Repository; // <bean id="userDaoImpl" class="at.guigu.dao.impl.UserDaoImpl"></bean> // 等同于@Component("userDaoImpl") @Repository("userDaoImpl") public class UserDaoImpl implements UserDao { public void save() { System.out.println("UserDao save running..."); } }
-
BookServiceImpl
类代码如下package at.guigu.service.impl; import at.guigu.dao.UserDao; import at.guigu.service.BookService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.stereotype.Service; //<bean id="bookServiceImpl" class="at.guigu.service.impl.BookServiceImpl"></bean> // 等同于@Component("bookServiceImpl") @Service("bookServiceImpl") public class BookServiceImpl implements BookService { // <property name="userDao" ref="userDaoImpl"/> @Autowired @Qualifier("userDaoImpl") private UserDao userDao; /*public void setUserDao(UserDao userDao) { this.userDao = userDao; }*/ @Override public void save() { System.out.println("BookService save..."); userDao.save(); } }
-
-
Step3: 在Spring配置文件中配置注解的组件扫描,Spring配置文件代码如下
- 配置组件扫描需要在Spring配置文件中引入
context
命名空间和context
约束路径然后使用context
命名空间加载 properties 文件。Spring配置文件代码如下- 命名空间:
xmlns:context="http://www.springframework.org/schema/context"
- 约束路径:
http://www.springframework.org/schema/context
、http://www.springframework.org/schema/context/spring-context.xsd
- 命名空间:
- 配置组件扫描代码:
<context:component-scan base-package="at.guigu"></context:component-scan>
base-package
:给定一个包,然后会自动扫描该包下的所有内容,以便可以识别使用注解配置的类、字段和方法
<?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.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!--配置注解的组件扫描--> <context:component-scan base-package="at.guigu"></context:component-scan> </beans>
- 配置组件扫描需要在Spring配置文件中引入
-
Step4: 测试类TestOne代码如下
package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TestOne { @Test public void test1() { //1 获取IOC容器 ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); //2 从IOC容器中获取bean对应的对象 BookServiceImpl bookService = (BookServiceImpl) app.getBean("bookServiceImpl"); bookService.save(); } }
-
-
使用注解开发的与使用配置文件进行依赖注入的对比图例如下
Spring原始注解
原始注解 | 解释 |
---|---|
@Component | 使用在类上,用于实例化bean |
@Controller | 使用在Web 层的类上,用于实例化bean |
@Service | 使用在Service 层的类上,用于实例化bean |
@Repository | 使用在Dao/Mapper 层的类上,用于实例化bean |
@Autowired | 使用在字段上,用于根据类型进行自动依赖注入 |
@Qualifier("beanId") | 必须与@Autowired 一起使用,用于根据名称进行依赖注入,从而避免按类型自动装配时的歧义 |
@Resource(name = "beanId") | 相当于@Autowired +@Resource 。它既可以按名称注入,也可以按类型注入。它的默认行为是按名称注入,如果找不到匹配的名称,再按类型注入。 |
@Value | 注入普通属性 |
@Scope | 使用在类上,标注bean的作用范围。默认为@Scope("singleton") :单例; @Scope("prototype") :多例 |
@PostConstruct | 使用在方法上,标注该方法是bean的初始化方法 |
@PreDestroy | 使用在方法上,标注该方法是bean的销毁方法 |
-
注意
-
@Repository
、@Service
、@Controller
以上三个注解能用@Component来代替- 因为这三个注解本质上都是
@Component
注解的扩展
- 因为这三个注解本质上都是
-
使用
@Autowired
、@Qualifier("beanId")
、@Resource("beanId")
这三个注解进行依赖注入时,可以省略不写setter
方法 -
@Value
用于替代value
,即只能用于基本数据类型与字符串类型 -
@Scope
不在进行测试,可自行测试
-
-
@Autowired
、@Qualifier
注解详解@Autowired
是按照数据类型从Spring容器中进行自动匹配的。它会自动匹配对应引用类型的实现类的bean@Qualifier
注解必须与@Autowired
一起配合使用。它是按照对应bean的id值从Spring容器中进行匹配的- 若只使用
@Autowired
,而不与@Qualifier
配合使用时:使用@Autowired
注解的引用数据类型对应的bean在Spring容器中只能有一个;反之则必须与@Qualifier
联合使用
-
@Resource
注解详解@Resource
相当于@Autowired
+@Resource
。@Resource(name = "beanId")
:如果beanId
属性被指定,首先按名称查找;如果没有指定beanId
,则按字段或属性名进行查找;如果没有找到匹配的名称,再按类型查找。@Resource
注解在代码中必须写为@Resource(name = "beanId")
的形式,若不加上name=
,则会报错
-
只使用
@Autowired
的示例-
使用
@Autowired
注解的属性只存在一个对应实现类的bean -
使用
@Autowired
注解的属性存在多个对应实现类的bean- 错误示例
- 正确示例:此时必须与
@Qualifier
联合使用
@Autowired
、@Qualifier("beanId")
联合使用@Resource
代替@Autowired+@Resource
-
@Value注入普通属性
-
简单形式的代码示例图如下
-
与properties配置文件结合给属性赋值的步骤
-
Step1: 在pom.xml文件中导入Spring、Annotation、数据源(连接池)的坐标和数据库mysql驱动坐标。pom.xml文件完整代码如下
- 博主测试C3P0、Druid两个数据库连接池,所以它俩的坐标均会导入
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.example</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>SpringAnno</artifactId> <packaging>war</packaging> <name>SpringAnno Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> </dependencies> <build> <finalName>SpringAnno</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
-
Step2: 右键源代码配置文件目录(即资源文件
resources
)→New
→File
,创建properties配置文件,博主文件名为jdbc.properties
,该配置文件代码如下- 注意: properties配置文件中配置的各个属性前必须添加个
id.
(即id.属性
,比如:属性url
就设置为id.url
,博主设置的为jdbc.url
),以供Spring配置文件可以使用属性占位符${}
语法引用这些属性
#driverClassName代表数据库驱动,后跟驱动全类名(在MySQL驱动jar包下的META-INF下的services文件夹下的java.sql.Driver文件内) jdbc.driverClassName=com.mysql.cj.jdbc.Driver # 数据库连接URL jdbc.url=jdbc:mysql://localhost:3306/test02?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=Asia/Shanghai # 数据库用户名 jdbc.username=root # 数据库密码 jdbc.password=123456 # 初始化连接数量---即容器中初始的数据库连接数量 jdbc.initialSize=5 # 最大活跃连接数量---容器中初始为5个,但若5个用完了,此时可以在申请5个数据库连接数量 #也就是说容器中最多存放10个数据库连接 jdbc.maxActive=10 # 获取连接时的最大等待时间,单位:毫秒。---与数据库进行连接时若超过3s仍未连接成功,则会报错 jdbc.maxWait=3000 #最小空闲连接数量---minIdle=5 # 配置检测连接是否有效的SQL,可以是一个查询语句,如果不指定则默认为"SELECT 1"---validationQuery=SELECT 1 # 是否开启自动提交事务---defaultAutoCommit=true
- 注意: properties配置文件中配置的各个属性前必须添加个
-
Step3: 右键源代码配置文件目录(即资源文件
resources
)→New
→XML Configuration File
→Spring Config
,文件名为applicationContext.xml
,然后在该Spring配置文件中引入context
命名空间、context
约束路径、配置注解的组件扫描,并引入properties配置文件后配置数据源(连接池)的bean。Spring配置文件完整代码如下<?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.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!--配置注解的组件扫描--> <context:component-scan base-package="at.guigu"></context:component-scan> <!--使用`context`命名空间加载 `properties` 文件--> <context:property-placeholder location="classpath:jdbc.properties"/> <!--Cp30对应的bean--> <bean id="dataSourceCp30" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <!--使用属性占位符`${}`语法引用properties文件中的属性--> <property name="driverClass" value="${jdbc.driverClassName}"/> <property name="jdbcUrl" value="${jdbc.url}"/> <property name="user" value="${jdbc.username}"/> <property name="password" value="${jdbc.password}"/> </bean> <!--Druid对应的bean--> <bean id="dataSourceDruid" class="com.alibaba.druid.pool.DruidDataSource"> <!--使用属性占位符`${}`语法引用properties文件中的属性--> <property name="driverClassName" value="${jdbc.driverClassName}"/> <property name="url" value="${jdbc.url}"/> <property name="username" value="${jdbc.username}"/> <property name="password" value="${jdbc.password}"/> </bean> </beans>
-
Step4:
BookServiceImpl
类属性配置代码如下package at.guigu.service.impl; import at.guigu.dao.UserDao; import at.guigu.service.BookService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; import javax.annotation.Resource; //<bean id="bookServiceImpl" class="at.guigu.service.impl.BookServiceImpl"></bean> @Service("bookServiceImpl") public class BookServiceImpl implements BookService { // <property name="userDao" ref="userDaoImpl"/> @Resource(name = "userDaoImpl") private UserDao userDao; @Value("${jdbc.driverClassName}") private String driverClass; @Value("${jdbc.username}") private String name; @Value("15") private int age; @Override public void save() { System.out.println(driverClass); System.out.println(name + "===" + age); userDao.save(); } }
-
Step5: 测试代码如下
package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TestOne { @Test public void test1() { //1 获取IOC容器 ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); //2 从IOC容器中获取bean对应的对象 BookServiceImpl bookService = (BookServiceImpl) app.getBean("bookServiceImpl"); bookService.save(); } }
-
@PostConstruct
、@PreDestroy
:初始、销毁方法
-
繁琐配置步骤均省略,可详见之前的步骤,此处只说明重点步骤
-
BookServiceImpl类代码如下
package at.guigu.service.impl; import at.guigu.dao.UserDao; import at.guigu.service.BookService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Scope; import org.springframework.stereotype.Service; import javax.annotation.PostConstruct; import javax.annotation.PreDestroy; import javax.annotation.Resource; //<bean id="bookServiceImpl" scope = "singleton" class="at.guigu.service.impl.BookServiceImpl"></bean> @Service("bookServiceImpl") @Scope("singleton") public class BookServiceImpl implements BookService { // <property name="userDao" ref="userDaoImpl"/> @Resource(name = "userDaoImpl") private UserDao userDao; @PostConstruct public void init() { System.out.println("Service Init..."); } @PreDestroy public void destroy() { System.out.println("Service Destroy..."); } @Override public void save() { System.out.println("BookService running..."); userDao.save(); } }
-
测试代码如下
package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TestOne { @Test public void test2() { //1 获取IOC容器 ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); //2 从IOC容器中获取bean对应的对象 BookServiceImpl bookService = (BookServiceImpl) app.getBean("bookServiceImpl"); bookService.save(); } }
-
-
注意:在上述测试代码运行截图中并未运行销毁方法,原因及解决方式请详见本章中的Spring配置文件标签详解一中的初始销毁知识点
Spring新注解
-
作用:配合Spring原始注解完全代替Spring配置文件
-
可替代的配置如下
- 非自定义的Bean的配置,比如Druid、Cp30的bean
- 加载properties配置文件的配置:
<context:property-placeholder location="classpath:jdbc.properties"/>
- 组件扫描的配置(若有多个包则中间用逗号或空格间隔):
<context:component-scan base-package="at.guigu"></context:component-scan>
<context:component-scan base-package="at.guigu.dao at.guigu.service"></context:component-scan>
<context:component-scan base-package="at.guigu.dao, at.guigu.service"></context:component-scan>
- 引入其它配置文件的配置:
<import resource="applicationContext-xxx.xml"/>
新注解 | 解释 |
---|---|
@Configuration | 使用在类上,用于指定当前类是Spring的核心配置类。当创建容器时会从该类上加载注解 |
@ComponentScan | 用于指定Spring在初始化容器时要扫描的包。即相当于组件扫描的配置。若扫描一个包,则为:@ComponentScan("包1") ;若扫描多个包,则为:@ComponentScan({"包1", "包2", ...}) |
@Bean | 使用在方法上,标注将该方法的返回值存储到容器中 |
@PropertySource | 用于加载properties 文件的配置 |
@Import | 用于导入其它文件的配置类。若只导入一个文件配置类,则为@Import(Aaa.class) ;若导入多个,则为@Import({Aaa.class, Bbb.class}) |
@ComponentScan 注解可能用到的属性 | 解释 |
value/basePackages | 指定要扫描的包 |
includeFilters | 用于指定一个或多个包含过滤器,限制只有符合条件的类才会被扫描和注册为 Spring Bean。过滤条件可以是注解、正则表达式等。 |
excludeFilters | 用于指定一个或多个排除过滤器,排除掉不需要扫描的类。它和 includeFilters 是互斥的 |
@ComponentScan 注解的内部注解 | 解释 |
@ComponentScan.Filter | 用于定义在组件扫描时的过滤规则。主要用于 includeFilters 和 excludeFilters 属性,用来实现扫描范围的精确控制。 |
@ComponentScan.Filter 注解用到的属性 | 解释 |
type | 指定过滤的类型。共有四种即FilterType.ANNOTATION :根据注解类型过滤类;FilterType.REGEX :使用正则表达式过滤类的全限定名;FilterType.ASPECTJ :使用 AspectJ 表达式进行过滤;FilterType.ANY :匹配所有类,不常用。 |
value/classes | 根据过滤类型不同,指定具体的过滤条件。如果 type 是 FilterType.ANNOTATION ,则 value 是要过滤的注解类型;如果 type 是 FilterType.ASPECTJ ,则 value 是 AspectJ 表达式。 |
pattern | 根据过滤类型不同,指定具体的过滤条件。如果 type 是 FilterType.REGEX ,则用pattern 要匹配的正则表达式 |
注意:
@ComponentScan
注解中的属性及内部注解示例可详见SpringMVC注解形式的快速入门示例
Spring新注解代码实现
-
Spring新注解快速入门的
UserDaoImpl
类、BookServiceImpl
类代码省略,详见Spring原始注解快速入门 -
在Spring原始注解中,利用
@Value()
注解注入普通属性时有一个与properties配置文件结合给属性赋值的步骤-
该步骤的Step3中的Spring配置文件代码如下
<?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.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!--配置注解的组件扫描--> <context:component-scan base-package="at.guigu"></context:component-scan> <!--使用`context`命名空间加载 `properties` 文件--> <context:property-placeholder location="classpath:jdbc.properties"/> <!--Cp30对应的bean--> <bean id="dataSourceCp30" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <!--使用属性占位符`${}`语法引用properties文件中的属性--> <property name="driverClass" value="${jdbc.driverClassName}"/> <property name="jdbcUrl" value="${jdbc.url}"/> <property name="user" value="${jdbc.username}"/> <property name="password" value="${jdbc.password}"/> </bean> <!--Druid对应的bean--> <bean id="dataSourceDruid" class="com.alibaba.druid.pool.DruidDataSource"> <!--使用属性占位符`${}`语法引用properties文件中的属性--> <property name="driverClassName" value="${jdbc.driverClassName}"/> <property name="url" value="${jdbc.url}"/> <property name="username" value="${jdbc.username}"/> <property name="password" value="${jdbc.password}"/> </bean> </beans>
-
-
利用新注解进行配置的步骤如下
-
Step1: 在pom.xml文件中导入Spring、Annotation、数据源(连接池)的坐标和数据库mysql驱动坐标。pom.xml文件完整代码如下
- 博主测试C3P0、Druid两个数据库连接池,所以它俩的坐标均会导入
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.example</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>SpringAnno</artifactId> <packaging>war</packaging> <name>SpringAnno Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> </dependencies> <build> <finalName>SpringAnno</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
-
Step2: 右键源代码配置文件目录(即资源文件
resources
)→New
→File
,创建properties配置文件,博主文件名为jdbc.properties
,该配置文件代码如下- 注意: properties配置文件中配置的各个属性前必须添加个
id.
(即id.属性
,比如:属性url
就设置为id.url
,博主设置的为jdbc.url
),以供Spring配置文件可以使用属性占位符${}
语法引用这些属性
#driverClassName代表数据库驱动,后跟驱动全类名(在MySQL驱动jar包下的META-INF下的services文件夹下的java.sql.Driver文件内) jdbc.driverClassName=com.mysql.cj.jdbc.Driver # 数据库连接URL jdbc.url=jdbc:mysql://localhost:3306/test02?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=Asia/Shanghai # 数据库用户名 jdbc.username=root # 数据库密码 jdbc.password=123456 # 初始化连接数量---即容器中初始的数据库连接数量 jdbc.initialSize=5 # 最大活跃连接数量---容器中初始为5个,但若5个用完了,此时可以在申请5个数据库连接数量 #也就是说容器中最多存放10个数据库连接 jdbc.maxActive=10 # 获取连接时的最大等待时间,单位:毫秒。---与数据库进行连接时若超过3s仍未连接成功,则会报错 jdbc.maxWait=3000 #最小空闲连接数量---minIdle=5 # 配置检测连接是否有效的SQL,可以是一个查询语句,如果不指定则默认为"SELECT 1"---validationQuery=SELECT 1 # 是否开启自动提交事务---defaultAutoCommit=true
- 注意: properties配置文件中配置的各个属性前必须添加个
-
Step4: 在三层架构包下创建一个
config
包,并在该包下创建一个**代替Spring配置文件的类**SpringConfiguration
,完整代码如下package at.guigu.config; import com.alibaba.druid.pool.DruidDataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource; import javax.sql.DataSource; // 该注解代表该类是Spring的核心配置类 @Configuration // 配置注解的组件扫描<context:component-scan base-package="at.guigu"></context:component-scan> @ComponentScan("at.guigu") // 加载properties配置文件<context:property-placeholder location="classpath:jdbc.properties"/> @PropertySource("classpath:jdbc.properties") public class SpringConfiguration { @Value("${jdbc.driverClassName}") private String driverClassName; @Value("${jdbc.url}") private String url; @Value("${jdbc.username}") private String username; @Value("${jdbc.password}") private String password; /** * Cp30对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceCp30") public DataSource getCp30DataSource() throws Exception{ // 创建数据源对象 ComboPooledDataSource dataSource = new ComboPooledDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClass(driverClassName); dataSource.setJdbcUrl(url); dataSource.setUser(username); dataSource.setPassword(password); return dataSource; } /** * Druid对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceDruid") public DataSource getDruidDataSource() throws Exception{ // 创建数据源对象 DruidDataSource dataSource = new DruidDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClassName(driverClassName); dataSource.setUrl(url); dataSource.setUsername(username); dataSource.setPassword(password); return dataSource; } }
-
测试代码如下
package at.guigu.web; import at.guigu.config.SpringConfiguration; import at.guigu.service.impl.BookServiceImpl; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.annotation.AnnotationConfigApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TestOne { @Test public void test3() { //1 获取IOC容器 // 通过Spring配置文件获取:ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); // 通过Spring类获取 ApplicationContext app = new AnnotationConfigApplicationContext(SpringConfiguration.class); //2 从IOC容器中获取bean对应的对象 BookServiceImpl bookService = (BookServiceImpl) app.getBean("bookServiceImpl"); bookService.save(); } @Test public void test4() { //1 获取IOC容器 // 通过Spring配置文件获取:ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); // 通过Spring类获取 ApplicationContext app = new AnnotationConfigApplicationContext(SpringConfiguration.class); //2 从IOC容器中获取bean对应的对象 ComboPooledDataSource dataSourceCp30 = (ComboPooledDataSource) app.getBean("dataSourceCp30"); System.out.println(dataSourceCp30); } }
-
-
在实际开发中代替Spring配置文件的
SpringConfiguration
类中的代码会很多,我们可以将其根据不同种类拆分开,然后在主Spring配置文件对应的类中引入拆分后的类,代码如下-
Step1: 在
config
包下创建拆分配置文件对应的拆分类DataSourceConfiguration
,代码如下package at.guigu.config; import com.alibaba.druid.pool.DruidDataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.PropertySource; import javax.sql.DataSource; // 分配置文件对应的类不用配置@Configuration以及@ComponentScan注解 // 加载properties配置文件<context:property-placeholder location="classpath:jdbc.properties"/> @PropertySource("classpath:jdbc.properties") public class DataSourceConfiguration { @Value("${jdbc.driverClassName}") private String driverClassName; @Value("${jdbc.url}") private String url; @Value("${jdbc.username}") private String username; @Value("${jdbc.password}") private String password; /** * Cp30对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceCp30") public DataSource getCp30DataSource() throws Exception{ // 创建数据源对象 ComboPooledDataSource dataSource = new ComboPooledDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClass(driverClassName); dataSource.setJdbcUrl(url); dataSource.setUser(username); dataSource.setPassword(password); return dataSource; } /** * Druid对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceDruid") public DataSource getDruidDataSource() throws Exception{ // 创建数据源对象 DruidDataSource dataSource = new DruidDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClassName(driverClassName); dataSource.setUrl(url); dataSource.setUsername(username); dataSource.setPassword(password); return dataSource; } }
-
Step2: 在Spring主配置文件对应的主类
SpringConfiguration
中利用@Import
注解引入分配置文件对应的拆分类DataSourceConfiguration
,代码如下:package at.guigu.config; import com.alibaba.druid.pool.DruidDataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.*; import javax.sql.DataSource; // 该注解代表该类是Spring的核心配置类 @Configuration // 配置注解的组件扫描<context:component-scan base-package="at.guigu"></context:component-scan> @ComponentScan("at.guigu") // 引入拆分配置文件<import resource="applicationContext-xxx.xml"/> @Import(DataSourceConfiguration.class) public class SpringConfiguration { }
-
为非自定义(即第三方)的bean注入资源
-
以数据源为例,假设现在容器中的数据源需要使用
UserDao
实现类的对象(即引用类型),此时直接将其作为形式参数传递给数据源方法中即可,因为Spring在检测到该数据源bean为第三方bean时,会自动为其提供IOC容器中参数所对应引用类型的bean,代码如下package at.guigu.config; import com.alibaba.druid.pool.DruidDataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.PropertySource; import javax.sql.DataSource; // 分配置文件对应的类不用配置@Configuration以及@ComponentScan注解 // 加载properties配置文件<context:property-placeholder location="classpath:jdbc.properties"/> @PropertySource("classpath:jdbc.properties") public class DataSourceConfiguration { @Value("${jdbc.driverClassName}") private String driverClassName; @Value("${jdbc.url}") private String url; @Value("${jdbc.username}") private String username; @Value("${jdbc.password}") private String password; /** * Cp30对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceCp30") public DataSource getCp30DataSource(BookService bookService) throws Exception{ System.out.println(bookService); // 创建数据源对象 ComboPooledDataSource dataSource = new ComboPooledDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClass(driverClassName); dataSource.setJdbcUrl(url); dataSource.setUser(username); dataSource.setPassword(password); return dataSource; } /** * Druid对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceDruid") public DataSource getDruidDataSource(BookService bookService) throws Exception{ System.out.println(bookService); // 创建数据源对象 DruidDataSource dataSource = new DruidDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClassName(driverClassName); dataSource.setUrl(url); dataSource.setUsername(username); dataSource.setPassword(password); return dataSource; } }
此处不在进行运行示例
Spring集成Junit
-
原始Junit测试Spring的问题
-
在测试类中均需要以下两行代码,这两行代码若不写的话就会造成空指针异常
//1 获取IOC容器 // 通过Spring配置文件获取:ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); // 通过Spring类获取 ApplicationContext app = new AnnotationConfigApplicationContext(SpringConfiguration.class); //2 从IOC容器中获取bean对应的对象 ComboPooledDataSource dataSourceCp30 = (ComboPooledDataSource) app.getBean("dataSourceCp30");
-
-
解决原始Junit测试Spring问题的方案
- 让SpringJunit负责创建Spring容器,但需要将配置文件或配置类告知,然后将需要进行测试的bean直接在测试类中进行注入
Spring集成Junit代码实现
-
Spring集成Junit步骤
-
导入相关坐标
- 导入Spring基础坐标:spring-context
- 导入Spring提供的监听器
ContextLoaderListener
的相关坐标:spring-web - 导入Spring集成Web环境相关坐标:servlet、jsp
- 导入Spring注解相关坐标:Annotation
- 导入数据库相关坐标:mysql、数据源坐标(druid、cp30)
- 导入Spring集成JUnit相关坐标:junit、spring-test
- 导入Spring集成MyBatis相关坐标:mybatis、spring-jdbc、mybatis-spring
-
使用
@Runwith
注解替换原来的运行器,并设置新的类运行器- 其属性为
SpringRunner.class
或SpringJUnit4ClassRunner.class
:用于集成 Spring 测试框架
- 其属性为
-
使用
@ContextConfiguration
指定Spring配置文件或Spring配置类- 指定Spring配置文件:
@ContextConfiguration("classpath:applicationContext.xml")
- 指定单个Spring配置类:
@ContextConfiguration(classes = SpringConfiguration.class)
- 指定多个Spring配置类:
@ContextConfiguration(classes = {SpringConfiguration.class,...})
- 指定Spring配置文件:
-
在该类中使用
@Autowired
注入需要测试的对象 -
在该类中创建对应测试方法测试对应方法
-
-
步骤如下
-
该步骤有两种形式
- Spring配置文件的形式:Spring配置文件及与之对应的
UserDaoImpl
、BookServiceImpl
代码均省略 - Spring配置类的形式:Spring配置类及与之对应的
UserDaoImpl
、BookServiceImpl
代码均省略,可详见Spring注解开发部分内容
- Spring配置文件的形式:Spring配置文件及与之对应的
-
Step1: 导入Spring集成Junit的坐标,即spring-test坐标
- 除了Junit坐标外还有spring、Annotation、数据源(连接池)的坐标和数据库mysql驱动坐标。pom.xml文件完整代码如下
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.example</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>SpringAnno</artifactId> <packaging>war</packaging> <name>SpringAnno Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> <!--spring-test坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>6.1.6</version> <scope>test</scope> </dependency> </dependencies> <build> <finalName>SpringAnno</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
-
Step2: 创建测试类
TestTwo
。代码如下- Spring配置文件的形式
package at.guigu.web; import at.guigu.service.BookService; import at.guigu.service.impl.BookServiceImpl; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; // 1 集成 Spring 测试框架 @RunWith(SpringJUnit4ClassRunner.class) // 2 指定Spring配置文件或Spring配置类 @ContextConfiguration("classpath:applicationContext.xml") public class TestTwo { // 3 注入需要测试的对象 @Autowired private BookServiceImpl bookService; // 3 注入需要测试的对象:若需要测试的bean有多个则与@Qualifier结合指定对应id的bean @Autowired @Qualifier("dataSourceDruid") private DataSource dataSourceDruid; // 3 注入需要测试的对象:若需要测试的bean有多个则与@Qualifier结合指定对应id的bean @Autowired @Qualifier("dataSourceCp30") private DataSource dataSourceCp30; // 4 创建测试方法进行测试 @Test public void test1() { bookServiceImpl.save(); System.out.println(dataSourceDruid); System.out.println(dataSourceCp30); } }
- Spring配置类的形式
package at.guigu.web; import at.guigu.config.SpringConfiguration; import at.guigu.service.BookService; import at.guigu.service.impl.BookServiceImpl; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import javax.sql.DataSource; // 1 集成 Spring 测试框架 @RunWith(SpringJUnit4ClassRunner.class) // 2 指定Spring配置文件或Spring配置类 @ContextConfiguration(classes = {SpringConfiguration.class}) public class TestTwo { // 3 注入需要测试的对象 @Autowired private BookService bookServiceImpl; // 3 注入需要测试的对象:若需要测试的bean有多个则与@Qualifier结合指定对应id的bean @Autowired @Qualifier("dataSourceDruid") private DataSource dataSourceDruid; // 3 注入需要测试的对象:若需要测试的bean有多个则与@Qualifier结合指定对应id的bean @Autowired @Qualifier("dataSourceCp30") private DataSource dataSourceCp30; // 4 创建测试方法进行测试 @Test public void test1() { bookServiceImpl.save(); System.out.println(dataSourceDruid); System.out.println(dataSourceCp30); } }
-
完整注解形式集成Junit代码实现
以Spring集成MyBatis项目为基础进行集成测试,要求:在maven项目的test包下完成测试(以测试业务层service包下的类为例)
-
Step1: 导入相关坐标,完整文件代码如下
- 导入Spring基础坐标:spring-context
- 导入Spring提供的监听器
ContextLoaderListener
的相关坐标:spring-web - 导入Spring集成Web环境相关坐标:servlet、jsp
- 导入Spring注解相关坐标:Annotation
- 导入数据库相关坐标:mysql、数据源坐标(druid、cp30)
- 导入Spring集成JUnit相关坐标:junit、spring-test
- 导入Spring集成MyBatis相关坐标:mybatis、spring-jdbc、mybatis-spring
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>org.example</groupId> <artifactId>SpringJunitDemo</artifactId> <packaging>war</packaging> <version>1.0-SNAPSHOT</version> <name>SpringJunitDemo Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--===================Spring基础坐标=======================--> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--===================Spring自带监听器ContextLoaderListener所需坐标=======================--> <!--spring-web--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>5.2.25.RELEASE</version> </dependency> <!--===================Spring集成Web环境相关坐标=======================--> <!-- servlet--> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>4.0.1</version> <scope>provided</scope> </dependency> <!--jsp--> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>javax.servlet.jsp-api</artifactId> <version>2.3.3</version> <scope>provided</scope> </dependency> <!--===================Spring注解相关坐标=======================--> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <!--=====================数据库相关坐标=========================--> <!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--===================Spring集成junit相关坐标=======================--> <!--junit坐标--> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> <!--spring-test坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>6.1.6</version> <scope>test</scope> </dependency> <!--=====================MyBatis相关坐标=========================--> <!--MyBatis坐标--> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> <version>3.5.16</version> </dependency> <!--mybatis-spring--> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis-spring</artifactId> <version>3.0.3</version> </dependency> <!--spring-jdbc--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>6.1.10</version> </dependency> </dependencies> <build> <finalName>SpringJunitDemo</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
-
Step2: 在test包下创建三层架构包的业务层service包,并在该包中创建
BrandServiceTest
测试类,完整代码如下- Step2-1: 该类要使用
@Runwith
注解替换原来的运行器,并设置新的类运行器- 其属性为
SpringRunner.class
或SpringJUnit4ClassRunner.class
:用于集成 Spring 测试框架
- 其属性为
- Step2-2: 该类使用
@ContextConfiguration
指定Spring配置文件或Spring配置类- 指定Spring配置文件:
@ContextConfiguration("classpath:applicationContext.xml")
- 指定单个Spring配置类:
@ContextConfiguration(classes = SpringConfiguration.class)
- 指定多个Spring配置类:
@ContextConfiguration(classes = {SpringConfiguration.class,...})
- 指定Spring配置文件:
- Step2-3: 在该类中使用
@Autowired
注入需要测试的对象 - Step2-4: 在该类中创建对应测试方法测试对应方法
package at.guigu.service; import at.guigu.config.SpringConfiguration; import at.guigu.pojo.Brand; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import java.util.List; @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {SpringConfiguration.class}) public class BrandServiceTest { @Autowired private BrandService brandService; @Test public void testGetAll( ) { List<Brand> brands = brandService.getAll(); for (Brand brand : brands) { System.out.println(brand); } } }
运行该测试类,截图如下
- Step2-1: 该类要使用
Spring集成Web环境
-
即集成
web/Controller
表现层步骤-
Step1: 在pom.xml文件中导入servlet和jsp坐标
- 导入servlet和jsp坐标后三层架构中的
web/Controller
表现层才会起效 - Spring需要导入的所有坐标:spring、spring-test、Annotation、servlet、jsp坐标。pom.xml文件完整代码如下
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.example</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>SpringWeb</artifactId> <packaging>war</packaging> <name>SpringWeb Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> <!--spring-test坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>6.1.6</version> <scope>test</scope> </dependency> <!-- servlet--> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>4.0.1</version> <scope>provided</scope> </dependency> <!--jsp--> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>javax.servlet.jsp-api</artifactId> <version>2.3.3</version> <scope>provided</scope> </dependency> </dependencies> <build> <finalName>SpringWeb</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
- 导入servlet和jsp坐标后三层架构中的
-
Step2: 创建三层架构,并写入对应接口和实现类,截图如下
-
Step3: 配置properties以及Spring配置文件,截图如下
-
Step4: 在
web/Controller
表现层下创建类BookServlet
,代码如下package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.annotation.*; import javax.sql.DataSource; import java.io.IOException; // 此时url不通过@WebServlet注解进行配置,而是在web.xml中进行配置 // @WebServlet("/bookServlet") public class BookServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //1 获取IOC容器 ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); //2 从IOC容器中获取bean对应的对象 BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid"); bookServiceImpl.save(); System.out.println(dataSourceDruid); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
-
Step5: 在web项目核心目录(即
WEB-INF
)下的web.xml文件中进行web配置,代码如下- 利用
<servlet>
标签声明Servlet - 利用
<servlet-mapping>
标签将URL模式映射到特定的Servlet上 <servlet>
标签一定要在<servlet-mapping>
标签之前,否则会报错
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app> <display-name>Archetype Created Web Application</display-name> <!--声明一个Servlet--> <servlet> <!--指定要声明的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--声明的Servlet的全限定名--> <servlet-class>at.guigu.web.BookServlet</servlet-class> </servlet> <!--将URL模式映射到特定的Servlet上--> <servlet-mapping> <!--指定的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--给指定的Servlet设置url,相当于@WebServlet("/bookServlet")--> <url-pattern>/bookServlet</url-pattern> </servlet-mapping> </web-app>
- 利用
-
Step6: 利用Tomcat运行该Web项目,运行截图如下
-
ApplicationContext应用上下文获取方式
-
定义:应用上下文对象是通过
new ClassPathXmlApplicationContext("spring配置文件");
方式获取的,但是每次从容器中获得bean时都要编写该句代码,这就会造成代码冗余,配置文件加载多次,应用上下文对象创建多次-
解释:在Spring集成web环境中,其中Web层代码是通过Spring容器获取Service层的,即
BookServlet
类中的如下两句代码//1 获取IOC容器 (ApplicationContext应用上下文获取方式) ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); //2 从IOC容器中获取bean对应的对象 BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid");
-
由于
web/Controller
表现层在后期会有很多业务(即很多Servlet类),所以就会造成代码冗余,配置文件加载多次,应用上下文对象创建多次
-
-
解决方式
在Web项目中,可以使用
ServletContextListener
监听Web应用的启动,我们可在Web应用启动时就加载Spring配置文件,创建Spring的应用上下文对象ApplicationContext
,然后将其存储到最大的域servletContext
对象中,这样就可以在任意位置从域中获取应用上下文ApplicationContext
对象了监听器Listener知识点内容可详见Filter&Listener&AJAX&Axios&JSON
自定义ContextListener
-
步骤
-
Step1: 在
web
包下创建listener
包,定义一个实现ServletContextListener
接口的实现类ContextLoaderListener
,代码及步骤如下package at.guigu.web.listener; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.annotation.*; @WebListener public class ContextLoaderListener implements ServletContextListener { public ContextLoaderListener() { } @Override public void contextInitialized(ServletContextEvent sce) { // 1 获取IOC容器,创建应用上下文对象`ApplicationContext` (ApplicationContext应用上下文获取方式) ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml"); // 2 获取最大域ServletContext对象 ServletContext servletContext = sce.getServletContext(); // 2 将Spring的应用上下文对象`ApplicationContext`存储到最大的域`servletContext`中 servletContext.setAttribute("app", app); } @Override public void contextDestroyed(ServletContextEvent sce) { } }
-
Step2: 在web项目核心目录(即
WEB-INF
)下的web.xml文件中进行监听器Listener配置,代码如下<listener>
标签一定要在<servlet>
标签之前,否则会报错
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app> <display-name>Archetype Created Web Application</display-name> <listener> <listener-class>at.guigu.web.listener.ContextLoaderListener</listener-class> </listener> <servlet> <servlet-name>BookServlet</servlet-name> <servlet-class>at.guigu.web.BookServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>BookServlet</servlet-name> <url-pattern>/bookServlet</url-pattern> </servlet-mapping> </web-app>
-
Step3:
web/Controller
表现层下创建类BookServlet
的代码更改如下package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import javax.sql.DataSource; import java.io.IOException; public class BookServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 1 获取最大域ServletContext对象 ServletContext servletContext = request.getServletContext(); // 2 通过最大域ServletContext对象获取Spring的应用上下文对象,即IOC容器 ApplicationContext app = (ApplicationContext) servletContext.getAttribute("app"); // 3 获取IOC容器中的bean BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid"); bookServiceImpl.save(); System.out.println(dataSourceDruid); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
-
然后利用Tomcat运行该Web项目即可
-
-
在上述过程中,其中监听器
ContextLoaderListener
类中创建应用上下文对象ApplicationContext
的那句代码需要传入Spring配置文件,这会造成耦合问题,因为我们可能会修改Spring配置文件的名称,为降低耦合,我们可将其进一步优化,步骤如下-
Step1:在web项目核心目录(即
WEB-INF
)下的web.xml文件中添加一个<context-param>
标签用于定义全局初始化参数,此处用于定义Spring配置文件供监听器使用,web.xml文件代码如下- 后期若Spring配置文件名字更改,我们就只需要更改web.xml文件中
<context-param>
标签中的子标签<param-value>
的参数即可
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app> <display-name>Archetype Created Web Application</display-name> <!--全局初始化参数--> <context-param> <!--定义参数的名称,必须是唯一的--> <param-name>contextConfigLocation</param-name> <!--定义参数的值--> <param-value>classpath:applicationContext.xml</param-value> </context-param> <!--监听器--> <listener> <!--监听器类的全限定名--> <listener-class>at.guigu.web.listener.ContextLoaderListener</listener-class> </listener> <!--声明一个Servlet--> <servlet> <!--声明的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--声明的Servlet的全限定名--> <servlet-class>at.guigu.web.BookServlet</servlet-class> </servlet> <!--将URL模式映射到特定的Servlet上--> <servlet-mapping> <!--指定的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--给指定的Servlet设置url,相当于@WebServlet("/bookServlet")--> <url-pattern>/bookServlet</url-pattern> </servlet-mapping> </web-app>
- 后期若Spring配置文件名字更改,我们就只需要更改web.xml文件中
-
Step2: 实现
ServletContextListener
监听器接口的实现类ContextLoaderListener
代码更改如下package at.guigu.web.listener; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.annotation.*; @WebListener public class ContextLoaderListener implements ServletContextListener { public ContextLoaderListener() { } @Override public void contextInitialized(ServletContextEvent sce) { // 1 获取最大域`servletContext`对象 ServletContext servletContext = sce.getServletContext(); // 2 通过最大域`servletContext`对象获取web.xml文件中的全局参数 String contextConfigLocation = servletContext.getInitParameter("contextConfigLocation"); // 3 获取IOC容器,创建应用上下文对象`ApplicationContext` (ApplicationContext应用上下文获取方式) ApplicationContext app = new ClassPathXmlApplicationContext(contextConfigLocation); // 4 将应用上下文对象`ApplicationContext`存入最大域中 servletContext.setAttribute("app", app); } @Override public void contextDestroyed(ServletContextEvent sce) { // 释放资源 } }
-
-
在上述优化后仍具有耦合,因为应用上下文对象的对象名是由开发人员在监听器对应的类中决定的,在Web层使用最大域对象来获取应用上下文对象时必须知道监听器类中所设置的对象名,这就具有耦合,为了解耦合,我们可以设置一个工具类,专门来返回应用上下文对象
ApplicationContext
在最大域中的键名,步骤如下-
Step1: 在util包下创建工具类
ApplicationContextUtil
,该类代码如下package at.guigu.util; import org.springframework.context.ApplicationContext; import javax.servlet.ServletContext; public class ApplicationContextUtil { public static ApplicationContext getApplicationContext(ServletContext servletContext) { return (ApplicationContext) servletContext.getAttribute("app"); } }
-
Step2:
web/Controller
表现层下创建类BookServlet
的代码更改如下package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import at.guigu.util.ApplicationContextUtil; import com.alibaba.druid.pool.DruidDataSource; import org.springframework.context.ApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import java.io.IOException; public class BookServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 1 获取最大域ServletContext对象 ServletContext servletContext = request.getServletContext(); // 2 获取应用上下文对象(即IOC容器) ApplicationContext app = ApplicationContextUtil.getApplicationContext(servletContext); //3 获取bean BookServiceImpl bookServiceImpl = app.getBean(BookServiceImpl.class); //等同于BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DruidDataSource dataSourceDruid = app.getBean(DruidDataSource.class); //等同于DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid"); bookServiceImpl.save(); System.out.println(dataSourceDruid); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
-
优化后完整步骤方式一(使用自己创建的监听器类)
-
Step1: 在pom.xml文件中导入坐标:spring、Annotation、spring-test、servlet、jsp以及Tomcat插件
-
若需要mysql及数据源(数据库连接池),则导入坐标:mysql、druid、c3p0
-
所有坐标代码如下
<!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> <!--spring-test坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>6.1.6</version> <scope>test</scope> </dependency> <!-- servlet--> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>4.0.1</version> <scope>provided</scope> </dependency> <!--jsp--> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>javax.servlet.jsp-api</artifactId> <version>2.3.3</version> <scope>provided</scope> </dependency>
-
-
Step2: 创建三层架构,并写入对应接口和实现类,截图如下
-
Step3: 配置properties以及Spring配置文件,截图如下
-
**Step4:**在util包下创建工具类
ApplicationContextUtil
,该类代码如下package at.guigu.util; import org.springframework.context.ApplicationContext; import javax.servlet.ServletContext; public class ApplicationContextUtil { public static ApplicationContext getApplicationContext(ServletContext servletContext) { return (ApplicationContext) servletContext.getAttribute("app"); } }
-
Step5: 在web包下创建listener包,并在listener包下创建实现
ServletContextListener
接口的监听器类ContextLoaderListener
,代码如下package at.guigu.web.listener; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.annotation.*; @WebListener public class ContextLoaderListener implements ServletContextListener { public ContextLoaderListener() { } @Override public void contextInitialized(ServletContextEvent sce) { // 1 获取最大的域`servletContext`对象 ServletContext servletContext = sce.getServletContext(); // 2 利用`servletContext`对象读取web.xml文件中的全局参数 String contextConfigLocation = servletContext.getInitParameter("contextConfigLocation"); // 3 获取IOC容器:获取应用上下文对象`ApplicationContext` 并将其存储到最大域中 servletContext.setAttribute("app", new ClassPathXmlApplicationContext(contextConfigLocation)); /* 等同于 ApplicationContext app = new ClassPathXmlApplicationContext(contextConfigLocation); servletContext.setAttribute("app", app); */ } @Override public void contextDestroyed(ServletContextEvent sce) { // 释放资源 } }
-
Step6: 在
web/Controller
表现层下创建类BookServlet
,代码如下package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import at.guigu.util.ApplicationContextUtil; import com.alibaba.druid.pool.DruidDataSource; import org.springframework.context.ApplicationContext; import javax.servlet.*; import javax.servlet.http.*; import java.io.IOException; public class BookServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 1 获取最大域ServletContext对象 ServletContext servletContext = request.getServletContext(); // 2 获取应用上下文对象(即IOC容器) ApplicationContext app = ApplicationContextUtil.getApplicationContext(servletContext); //3 获取bean BookServiceImpl bookServiceImpl = app.getBean(BookServiceImpl.class); //等同于BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DruidDataSource dataSourceDruid = app.getBean(DruidDataSource.class); //等同于DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid"); bookServiceImpl.save(); System.out.println(dataSourceDruid); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
-
Step7: 在web项目核心目录(即
WEB-INF
)下的web.xml文件中进行:全局初始化参数、监听器Listener配置、web配置。完整代码如下<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app> <display-name>Archetype Created Web Application</display-name> <!--全局初始化参数--> <context-param> <!--定义参数的名称,必须是唯一的--> <param-name>contextConfigLocation</param-name> <!--定义参数的值--> <param-value>classpath:applicationContext.xml</param-value> </context-param> <!--监听器--> <listener> <!--监听器类的全限定名--> <listener-class>at.guigu.web.listener.ContextLoaderListener</listener-class> </listener> <!--声明一个Servlet--> <servlet> <!--声明的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--声明的Servlet的全限定名--> <servlet-class>at.guigu.web.BookServlet</servlet-class> </servlet> <!--将URL模式映射到特定的Servlet上--> <servlet-mapping> <!--指定的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--给指定的Servlet设置url,相当于@WebServlet("/bookServlet")--> <url-pattern>/bookServlet</url-pattern> </servlet-mapping> </web-app>
-
Step8: Tomcat运行该Web项目即可
优化后完整步骤方式二(使用Spring提供的监听器类)
- 完整步骤方式二,已上传到Gitee,可自行下载
Spring配置文件的形式(非注解开发 )
-
在优化后完整步骤方式一中使用的是我们自己创建的继承
ServletContextListener
接口的监听器类ContextLoaderListener
-
此处我们使用Spring所提供的监听器
ContextLoaderListener
-
该监听器内部加载Spring配置文件,创建应用上下文对象,并存储到最大域
ServletContext
中 -
它提供了一个客户端工具类
WebApplicationContextUtils
供使用者获取应用上下文对象 -
使用该监听器的步骤
-
在web项目核心目录(即
WEB-INF
)下的web.xml文件中配置ContextLoaderListener
监听器(前提是要先导入**spring-web** 坐标)<!--必须为5.2.X.RELEASE版本,否则会报错使用WebApplicationContextUtils获取应用上下文对象会报错:需要的是Jakarta Servlet而不是Java Servlet--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>5.2.25.RELEASE</version> </dependency>
-
使用客户端工具类
WebApplicationContextUtils
获取应用上下文对象ApplicationContext
-
-
-
实现步骤
-
Step1: 在pom.xml文件中导入坐标:spring、spring-web、spring-test、Annotation、servlet、jsp以及Tomcat插件
- 若需要mysql及数据源(数据库连接池),则导入坐标:mysql、druid、c3p0
- 若需要MyBatis,则导入:spring-jdbc、mybatis-spring、mybatis
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.example</groupId> <artifactId>SpringDemo</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>SpringWeb</artifactId> <packaging>war</packaging> <name>SpringWeb Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!--===================Spring相关坐标=======================--> <!--spring坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>6.1.6</version> </dependency> <!--spring-web --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>5.2.25.RELEASE</version> </dependency> <!--spring-test坐标--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>6.1.6</version> <scope>test</scope> </dependency> <!--Annotation坐标--> <dependency> <groupId>javax.annotation</groupId> <artifactId>javax.annotation-api</artifactId> <version>1.3.2</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> <!-- servlet--> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>4.0.1</version> <scope>provided</scope> </dependency> <!--jsp--> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>javax.servlet.jsp-api</artifactId> <version>2.3.3</version> <scope>provided</scope> </dependency> <!--=====================数据库相关坐标=========================--> <!--mysql坐标--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.33</version> </dependency> <!--druid坐标--> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.2.18</version> </dependency> <!--c3p0坐标--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.5</version> </dependency> <!--=====================MyBatis相关坐标=========================--> <!--spring-jdbc--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>6.1.10</version> </dependency> <!--mybatis-spring--> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis-spring</artifactId> <version>3.0.3</version> </dependency> <!--MyBatis坐标--> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> <version>3.5.16</version> </dependency> </dependencies> <build> <finalName>SpringWeb</finalName> <plugins> <!-- Tomcat插件 --> <plugin> <groupId>org.apache.tomcat.maven</groupId> <artifactId>tomcat7-maven-plugin</artifactId> <version>2.2</version> </plugin> </plugins> </build> </project>
-
Step2: 创建三层架构,并写入对应接口和实现类,截图如下
-
Step3: 配置properties以及Spring配置文件,截图如下
-
Step4: 在web项目核心目录(即
WEB-INF
)下的web.xml文件中进行:全局初始化参数、配置Spring所提供的ContextLoaderListener
监听器、web配置。完整代码如下<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app> <display-name>Archetype Created Web Application</display-name> <!--全局初始化参数--> <context-param> <!--定义参数的名称,必须是唯一的--> <param-name>contextConfigLocation</param-name> <!--定义参数的值--> <param-value>classpath:applicationContext.xml</param-value> </context-param> <!--监听器--> <!--配置Spring所提供的`ContextLoaderListener` 监听器--> <listener> <!--监听器类的全限定名--> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!--声明一个Servlet--> <servlet> <!--声明的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--声明的Servlet的全限定名--> <servlet-class>at.guigu.web.BookServlet</servlet-class> </servlet> <!--将URL模式映射到特定的Servlet上--> <servlet-mapping> <!--指定的Servlet的类名--> <servlet-name>BookServlet</servlet-name> <!--给指定的Servlet设置url,相当于@WebServlet("/bookServlet")--> <url-pattern>/bookServlet</url-pattern> </servlet-mapping> </web-app>
-
Step5: 在
web/Controller
表现层下创建类BookServlet
,代码如下package at.guigu.web; import at.guigu.service.impl.BookServiceImpl; import com.alibaba.druid.pool.DruidDataSource; import org.springframework.context.ApplicationContext; import org.springframework.web.context.support.WebApplicationContextUtils; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; public class BookServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException { // 1 获取最大域ServletContext对象 ServletContext servletContext = request.getSession().getServletContext(); // 2 获取应用上下文对象(即IOC容器) ApplicationContext app = WebApplicationContextUtils.getWebApplicationContext(servletContext); //3 获取bean BookServiceImpl bookServiceImpl = app.getBean(BookServiceImpl.class); //等同于BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DruidDataSource dataSourceDruid = app.getBean(DruidDataSource.class); //等同于DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid"); bookServiceImpl.save(); System.out.println(dataSourceDruid); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
-
Spring配置类的形式(注解开发 )
-
Step1: 在pom.xml文件中导入坐标:spring、spring-web、spring-test、Annotation、servlet、jsp以及Tomcat插件
- 若需要mysql及数据源(数据库连接池),则导入坐标:mysql、druid、c3p0
- 若需要MyBatis,则导入:spring-jdbc、mybatis-spring、mybatis
- pom.xml代码详见Spring配置文件的形式
-
Step2: 创建三层架构,并写入对应接口和实现类,并使用Spring注解开发截图如下
-
Step3: 右键源代码配置文件目录(即资源文件
resources
)→New
→File
,创建properties配置文件,博主文件名为jdbc.properties
,该配置文件代码如下- 注意: properties配置文件中配置的各个属性前必须添加个
id.
(即id.属性
,比如:属性url
就设置为id.url
,博主设置的为jdbc.url
),以供Spring配置文件可以使用属性占位符${}
语法引用这些属性
#driverClassName代表数据库驱动,后跟驱动全类名(在MySQL驱动jar包下的META-INF下的services文件夹下的java.sql.Driver文件内) jdbc.driverClassName=com.mysql.cj.jdbc.Driver # 数据库连接URL jdbc.url=jdbc:mysql://localhost:3306/test02?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=Asia/Shanghai # 数据库用户名 jdbc.username=root # 数据库密码 jdbc.password=123456 # 初始化连接数量---即容器中初始的数据库连接数量 jdbc.initialSize=5 # 最大活跃连接数量---容器中初始为5个,但若5个用完了,此时可以在申请5个数据库连接数量 #也就是说容器中最多存放10个数据库连接 jdbc.maxActive=10 # 获取连接时的最大等待时间,单位:毫秒。---与数据库进行连接时若超过3s仍未连接成功,则会报错 jdbc.maxWait=3000 #最小空闲连接数量---minIdle=5 # 配置检测连接是否有效的SQL,可以是一个查询语句,如果不指定则默认为"SELECT 1"---validationQuery=SELECT 1 # 是否开启自动提交事务---defaultAutoCommit=true
- 注意: properties配置文件中配置的各个属性前必须添加个
-
Step4: 在三层架构包下创建一个
config
包,并在该包下创建一个**代替Spring配置文件的类**SpringConfiguration
,完整代码如下package at.guigu.config; import com.alibaba.druid.pool.DruidDataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource; import javax.sql.DataSource; // 该注解代表该类是Spring的核心配置类 @Configuration // 配置注解的组件扫描<context:component-scan base-package="at.guigu"></context:component-scan> @ComponentScan("at.guigu") // 加载properties配置文件<context:property-placeholder location="classpath:jdbc.properties"/> @PropertySource("classpath:jdbc.properties") public class SpringConfiguration { @Value("${jdbc.driverClassName}") private String driverClassName; @Value("${jdbc.url}") private String url; @Value("${jdbc.username}") private String username; @Value("${jdbc.password}") private String password; /** * Cp30对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceCp30") public DataSource getCp30DataSource() throws Exception{ // 创建数据源对象 ComboPooledDataSource dataSource = new ComboPooledDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClass(driverClassName); dataSource.setJdbcUrl(url); dataSource.setUser(username); dataSource.setPassword(password); return dataSource; } /** * Druid对应的bean * Spring会将当前方法的返回值以指定的id存储到Spring的IOC容器中 * @return * @throws Exception */ @Bean("dataSourceDruid") public DataSource getDruidDataSource() throws Exception{ // 创建数据源对象 DruidDataSource dataSource = new DruidDataSource(); // 设置数据源基本连接数据 dataSource.setDriverClassName(driverClassName); dataSource.setUrl(url); dataSource.setUsername(username); dataSource.setPassword(password); return dataSource; } }
-
Step5: 在web项目核心目录(即
WEB-INF
)下的web.xml文件中进行:全局初始化参数、配置Spring所提供的ContextLoaderListener
监听器、web配置。完整代码如下<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app> <display-name>Archetype Created Web Application</display-name> <!--配置Spring配置类的全局初始化参数--> <context-param> <param-name>contextClass</param-name> <param-value>org.springframework.web.context.support.AnnotationConfigWebApplicationContext</param-value> </context-param> <context-param> <!--定义参数的名称,必须是唯一的--> <param-name>contextConfigLocation</param-name> <!--定义参数的值--> <param-value>at.guigu.config.SpringConfiguration</param-value> </context-param> <!--监听器--> <!--配置Spring所提供的`ContextLoaderListener` 监听器--> <listener> <!--监听器类的全限定名--> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!--声明一个Servlet--> <servlet> <!--声明的Servlet的类名--> <servlet-name>UserServlet</servlet-name> <!--声明的Servlet的全限定名--> <servlet-class>at.guigu.web.UserServlet</servlet-class> </servlet> <!--将URL模式映射到特定的Servlet上(即UserServlet)--> <servlet-mapping> <!--指定的Servlet的类名--> <servlet-name>UserServlet</servlet-name> <!--给指定的Servlet设置url,相当于@WebServlet("/userServlet")--> <url-pattern>/userServlet</url-pattern> </servlet-mapping> </web-app>
-
Step6: 在web层下创建
UserServlet
类,代码如下package at.guigu.web; import at.guigu.service.impl.UserServiceImpl; import com.alibaba.druid.pool.DruidDataSource; import org.springframework.context.ApplicationContext; import org.springframework.web.context.support.WebApplicationContextUtils; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; public class UserServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 1 获取最大域ServletContext对象 ServletContext servletContext = request.getServletContext(); //等同于ServletContext servletContext = request.getSession().getServletContext(); // 2 获取应用上下文对象(即IOC容器) ApplicationContext app = WebApplicationContextUtils.getWebApplicationContext(servletContext); //3 获取bean UserServiceImpl userServiceImpl = app.getBean(UserServiceImpl.class); //等同于BookServiceImpl bookServiceImpl = (BookServiceImpl) app.getBean("bookServiceImpl"); DruidDataSource dataSourceDruid = app.getBean(DruidDataSource.class); //等同于DataSource dataSourceDruid = (DataSource) app.getBean("dataSourceDruid"); userServiceImpl.save(); System.out.println(dataSourceDruid); } @Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doGet(request, response); } }
Tomcat运行该Web项目后截图如下所示