Spring的总结
Spring的核心:
1. Spring的两大核心:
1. IOC: 控制反转,将创建(实例化)对象的权利交给Spring容器去进行管理.
2. AOP: 面向切面编程(将切面织入到连接点,使连接点称为切入点).
2. Spring是一线式框架:
简单的来说,可以单单使用Spring实现MVC三层架构,Mybtais的数据访问层可以使用JdbcTemplate来代替.web层有SpringMVC,Spring贯穿了三层架构.
3. IOC:
* IOC的目的是实现解耦,来降低程序的耦合度(高内聚,低耦合).
* 实现的思想是: 反射 + 配置文件 + 工厂,使用工厂来读取配置文件,利用反射来实例化对象.
4. 初始化Spring容器:
* ApplicationContext的三个实现接口:
1. ClassPathXmlApplicationContext: 从类路径下加载配置文件
2. FileSystemXmlApplicationContext: 从文件系统中加载配置文件
3. AnnotationConfigApplicationContext: 注解开发
* ApplicationContext与BeanFactory的区别和联系:
ApplicationContext是BeanFactory的子接口(BeanFactory是ApplicationContext的顶级父接口).
ApplicationContext采用立即加载的方式,加载配置文件的时候就初始化bean.
BeanFactory采用延迟加载,使用bean的时候才会初始化.
ApplicationContext还实现了其他的接口,具有更为丰富的功能(国际化,自动装配等)
5. Bean的实例化的三种方式:
<!--初始化Bean有三种方式
1. 使用Bean的id class,需要默认的构造方法
-->
<bean id="testService" class="cn.itcast.spring.service.impl.TestServiceImpl"></bean>
<!--第二种方式 使用BeanFactory的普通方法-->
<bean id="factory" class="cn.itcast.spring.BeanFactory"></bean>
<bean id="testService2" factory-bean="factory" factory-method="getBean"></bean>
<!--第三种方法 使用BeanFactory的静态方法-->
<bean id="testService3" class="cn.itcast.spring.BeanFactory" factory-method="getStaticBean"></bean>
6. Bean的作用范围:
singleton: 单例的(Spring的默认),在IOC容器中唯一
prototype: 多例的,每次获取bean都是新对象
request: 在web开发中,与request域绑定,获取的bean作为request的属性
session: 在web开发中,与session域进行绑定,获取的bean作为session的属性
global-session: 在web开发中,在集群环境下,才可以使用,如果是单个服务器,与session没有区别.
7. Bean的生命周期(指定init-method和destroy-method):
单例:
初始化:加载配置文件的时候初始化,执行init-method方法
活着:容器存在,Bean存在
销毁:容器销毁,Bean消失,执行destroy-method方法.
总结: 单例模式的Bean的生命周期与容器相同(容器的销毁需要手动销毁).
多例:
初始化: 调用getBean使用的时候初始化
活着:对象使用就一直活着
销毁:有java的JVM的垃圾回收机制决定
总结:使用时初始化,销毁由JVM的垃圾回收机制决定
Spring的DI
2. 注入的三种方式
* 构造器注入(constructor-arg标签),必须指定构造函数所有的参数
1. type:根据参数的类型来注入数据
2. index:根据参数的索引来注入,从0开始
3. name:根据参数的名称来注入数据(建议使用)
4. value:只能注入基本类型,基本类型的包装类和String
5. ref:用于注入其他的Bean类型数据(尤其注意Date类型)
* setter注入(依赖于set方法,使用property标签)----重要
1. name:用于指定注入时所调用的set方法后面的变量名称
2. value:注入基本类型,基本类型的包装类和String
3. ref:用于注入其他的Bean类型数据(尤其注意Date类型<bean id="now" class="java.util.Date"></bean>)
* 注解注入
3. 集合的注入(前三个为值类型的集合,后两个为键值对类型的集合,同类型的集合,使用的标签可以互换,但是建议使用的时候标签与实际类型对应)
* 数组
<property name="myArrs">
<array>
<value>AAA</value>
<value>BBB</value>
</array>
</property>
* List集合
<property name="myLists">
<list>
<value>AAA</value>
<value>BBB</value>
</list>
</property>
* Set集合
<property name="mySets">
<set>
<value>AAA</value>
<value>BBB</value>
</set>
</property>
* Map
<property name="myMaps">
<map>
<!--如果是复杂数据类型,可以使用key-ref和value-ref-->
<entry key="testA" value="AAA"></entry>
<entry key="testB" value="BBB"></entry>
</map>
</property>
* Properties
<property name="properties">
<props>
<prop key="testA">AAA</prop>
<prop key="testB">BBB</prop>
</props>
</property>
Spring的IOC注解 + xml
1. Spring的注解是为了替代xml配置文件而开发的.
2. 在xml配置文件中,修改约束,并开启注解扫描
<context:component-scan base-package=""/>
3. 注解定义容器中的Bean
@Component ----> 如果不指定value,默认的bean的名称是类名的首字母小写,指定value属性bean的名称就是指定的值
@Controller ----> 作用于表现层
@Service ---> 作用于业务层
@Repository ---> 作用于数据访问层
4. 注入属性的注解
@Autowired: 按照类型注入,多个类型使用变量名来判定注入的bean.
@Qualifiler: 配合@Autoeired注解完成注入,一起使用,使用该注解指定注入的名称.
@Resource注解指定name属性来指定注入的名称
@Value: 用于注入基本类型和String,上面三种是注入其他的Bean的.
5. @Scope的注解是指定作用范围
singleton: 单例
prototype: 多例
request
session
global-session
6. 生命周期的注解:
@PostConstruct: 指定初始化
@PreDestroy: 指定销毁方法
全注解方式
1. 在前面的开发基础上,替换xml文件,使用配置类
配置类就是使用@Configuration注解标识的类
2. 那包扫描使用哪个注解?
@ComponentScan : 指定扫描的包
3. @Bean: 在方法上使用,可以将返回的值对象放入Spring容器中,需要指定name属性
4. 使用注解开发如何加载Spring的容器呢?
想到ApplicatoinContext的三个实现类的第三个,使用AnnotationConfigApplicationCOntext(配置类的字节码)来获取Spring容器
5. 关于配置类的相关注解
* @Configuration
* @Import : 导入子配置类,使用该注解的类是总配置类
* @ComponentScan : 指定扫描的包
* @Bean: 在方法上使用,可以将返回的值对象放入Spring容器中,需要指定name属性
* @PropertySource: 指定文件的名称和路径 @PropertySource("classpath:jdbcConfig.properties")
* @Qualifiler 用于方法参数,可以指定注入的值,用于多个数据源等参数有多个实现类的情况.
6. 为什么使用子配置类
将同一个业务的代码放在同比一个配置文件,不同的业务放在不同的配置文件,高内聚. 所以使用子配置类,总配置类使用@Import注解来加载子配置类
Spring整个Junit
1. 两个注解
@Runwith
@ContextConfiguration
SpringAOP
1. 概念:
AOP: 面向切面编程,与OOP类似,是一种编程思想(AspectJ是基于AOP思想的一种技术)
作用: 在不更改源码的情况下,对原始方法进行增强
底层原理: 使用动态代理实现
2. 相关术语:
Joinpoint(连接点): 简单来说业务层接口的所有的方法都可以作为连接点
Pointcut(切入点): 实际被切面织入的连接点(被增强的方法)
Advice(通知): 切面的整个实现(切面织入的目标对象,时间点以及内容),通知是共性代码抽取出来的方法
前置,后置,异常,最终,环绕
Target(目标对象): 切入点和连接点所属的类
weaving(织入): 将切面代码插入到目标对象的某个方法的过程
Aspect(切面): 交叉在各个业务逻辑中的系统服务,类似于安全验证,事务处理等等
Advisor(顾问): 就是通知的封装和延伸,可以将通知以更为复杂的方式织入到某些方法中去
Proxy(代理): 创建代理对象的整个过程
3. 简单总结:
将切面织入到连接点,使得连接点成为切入点
SpringAOP的XML的配置
最前面,pom.xml的配置文件引入的依赖:
spring-context
spring-tx
spring-jdbc
mysql-connection-java
aspectjweaver
spring-test
junit
1. 核心配置文件的内容
1. 目标类的Bean的注入
2. 通知类的Bean的注入
3. 表明AOP的配置 <aop:config>
4. 在config的内部配置切面 <aop:aspect>
注意的属性有两个,一个是id(唯一标识),一个是ref(指向通知类).
5. 在aspect内部配置通知类型
前置通知
后置通知
异常通知
最终通知
环绕通知
注意两个属性:
method: 指明通知(也就是注入的方法)
pointcut: 指明切入点(使用切入点表达式 * 包名.包名..*.*(..))
6. 可以统一配置切入点表达式<aop:pointcut>,两个属性注意
id: 唯一标识
expression: 配置切入点表达式
SpringAOP的注解的配置
1. 核心配置文件:
1. 开启注解扫描: <context:component-scan base-package="com.itheima"></context:component-scan>
2. 开启Spring对AOP注解的支持 <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
2. 对于目标类,需要添加bean的注解
3. 通知类需要添加bean的注解支持
1. 通知类需要添加切面注解 @Aspect
2. 通知类需要定义切面表达式(利用方法,唯一标识就是方法名)
3. 对通知添加通知类型的注解(前置,后置,最终,异常,环绕),注意的是环绕通知的ProceedingJoinPoint
4. 测试类:
使用ClassPathXmlApplicationContext来创建ApplicationContext的实例
5. 注意顺序问题:
对于注解来说,前置,后置,最终,异常的顺序是=====> 前置 -> 最终 -> 后置 注意的是后置和异常同一时刻只能执行一个,如果有环绕通知,环绕通知最先执行
我是前置通知
添加账户
我是最终通知
我是后置通知
有异常的时候:
我是前置通知
添加账户
我是最终通知
我是异常通知
环绕通知的应用场景:
事务管理
日志记录
性能检测
SpringJdbcTemplate
1. 首先引入依赖:
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-jdbc -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.1.0.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-tx -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.1.0.RELEASE</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.46</version>
</dependency>
2. 对于dao接口的实现类,继承JdbcDaoSupport,直接使用getJdbcTemplate()就可以获取JdbcTemplate对象
3. 核心配置文件的配置:
1. 注入数据访问层接口实现类的bean,中间使用setter注入JdbcTemplate
2. 注入JdbcTemplate的bean,中间使用setter注入DataSource
3. 注入dataSource,使用Spring自带的连接池DriverManagerDataSource,使用setter注入四要素
4. 测试执行就好
Spring事务管理
1. 事务控制在业务(Service)层
2. Spring事务管理的API三大接口 -----> spring-tx包下
1. PlatformTransactionManager : 平台事务管理器
DataSourceTransactionManager ----> springjdbc,mybatis
HibernateTransactionManager ----> hibernate
2. TransactionDefinition : 事务定义接口(包含了事务的基本属性信息)
3. TransactionStatus : 事务状态接口(事务状态的一些基本信息)
3. 事务管理的方式:
* 声明式事务管理(重点掌握)
1. XML
2. 注解
* 编程式事务管理(了解)
事务管理之环境依赖
1. 环境搭建(依赖):
spring-context : 核心
spring-tx : 事务管理
spring-jdbc : 数据库
mysql-connection-java : 数据库连接
aspectjweave : aop
spring-test : 测试,与junit进行整合
junit : 测试
事务管理之基于XML的声明式事务控制
pom文件的依赖:
spring-context
spring-tx
spring-jdbc
mysql-connection-java
aspectjweaver
spring-test
junit
1. 核心配置文件配置(aop,tx的约束):
1. 声明service的bean(注入dao)
2. 声明dao的bean(注入dataSource),dao实现类继承JdbcDaoSupport,使用Spring提供的模板
3. 声明dataSource,使用spring的连接池 DriverManagerDataSource,注入四要素
4. 配置事务管理器(<bean>) DataSourceTransactionManager,需要注入dataSource
5. 配置事务的通知 <tx:advice>,其实就是事务管理的通知
id属性: 唯一标识
transaction-manager: 给事务通知提供一个事务管理器的引用
6. 配置AOP的通用切入点表达式<aop:config>
配置切入点表达式 <aop:pointcut>
建立切入点表达式与事务通知之间的关系 <aop:advisor>,内部其实是环绕通知
advice_ref
pointcut-ref
7. 配置事务的属性(在事务通知标签内部配置)
<tx:attribute><tx:method name="find*" propagation="" read-only="" /></tx:attribute>,name用来配置方法,可以使用表达式
isolation: 事务的隔离级别,默认值是default
propagation: 指定事务的传播行为,默认是REQUIRED,表示一定会有事务,增删改的选择.如果是查询,可以说那个SUPPORTS
read-only: 用于指定事务是否只读,只有查询方法才能设置为true,默认是false,表示读写
timeout: 指定事务的超时时间,默认值是-1,表示永不超时,单位是秒
rollback-for: 用于指定一个异常,当产生该异常时,事务回滚,产生其他异常时,事务不回滚。没有默认值。默认回滚运行期异常
no-rollback-for: 用于指定一个异常,当产生该异常时,事务不回滚,产生其他异常时事务回滚。没有默认值。默认回滚运行期异常
8. 测试的时候使用Junit和Spring的整合进行测试
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:bean.xml"})
public class TxTest {
@Autowired
private AccountService accountService;
@Test
public void testTransfer(){
accountService.tranfer("aaa","bbb",100f);
}
}
如果使用main方法进行测试注意的是获取bean的时候获取的应该是接口,因为使用的是java的动态代理,获取的代理对象与原来对象的关系是兄弟关系,如果直接获取实现类的对象,会报代理异常错误
2. 关于Spring事务控制异常回滚
默认回滚运行期异常,如果想要控制非运行期异常也回滚,则使用rollback-for="java.lang.Exception"来指定,指定多个异常使用逗号隔开
事务管理之基于注解的事务控制(在前一个项目的基础上修改)
1. 核心配置文件(aop,tx,context)
1. 开启注解扫描: <context:component-scan>
2. 配置平台事务管理器 <bean>指定事务管理器 DataSourceTransactionManager
3. 开启Spring对注解事务的支持: <tx:annotation-driver transaction-manager="">
4. 在目标类上面添加 @Transactional() 还可以配置属性,属性一般都有默认值,可以查看源码试试
注意的是 @Transactional()可以指定在方法上面,单独为这个方法指定事务属性
2. 替换核心配置文件的内容:
1. 替换service的bean的注入
2. 删除事务通知和AOP切面就可以了
3. 直接进行测试
4. @Transactional作用位置的区别
> 作用于类上,当前类所有的方法具有事务控制
> 作用在方法上,当前类的所有方法都具有事务
> 接口上,当前接口的实现类的所有方法具有事务
Spring的全注解配置方式(使用下面的方式替换xml配置文件)
首先先删除xml配置文件
1. 核心配置类
@Configuration
@ComponentScan
@Import // 导入其他两个配置文件
@PropertySource({"classpath:"})
@EnableTransactionMangement // 开启事务注解支持
2. 子配置类(两个子配置类,一个事务,一个JDBC连接)
@Configuration
@Bean(name="")
@Value
在jdbc连接配置类中配置DataSource和JdbcTemplate
在事务配置类中配置 用于创建事务管理器对象PlatformTransactionManager Bean的名字为transactionManager