Spring是一个全面的全面的、企业应用开发一站式的解决方案,贯穿表现层、业务层、持久层。但是 Spring 仍然可以和其他的框架无缝整合。
1、Spring的核心组件
(1)数据层: JDBC、ORM、OXM、JMS、Transations
(2)Web层: Web、Servlet、Portlet、Struts
(3)中间层: AOP、Aspects、Instrumentation
(4)核心容器: Beans、Core、Context、Expression Language
(5)测试层: Test
2、Spring IOC 原理
(1)概念: Spring通过一个配置文件描述Bean及Bean之间的依赖关系,利用Java语言的反射功能实例化 Bean并建立Bean之间的依赖关系。Spring的IOC容器在完成这些底层工作的基础上,还提供 了 Bean实例缓存、生命周期管理、Bean实例代理、事件发布、资源装载等高级服务。
(2)Spring容器高层视图
(3)IOC容器实现
IOC容器依赖于BeanFactory,是Spring框架的基础程序,所以开发者操作的是ApplicationContext而非底层的BeanFactory。
Bean注册表
首先,Spring配置文件中的每一个节点元素在Spring容器中都通过一个 BeanDefinition 对象表示,它表述了 Bean 的配置信息,而BeanDefinitionRegistry 接口提供了向容器手动注册 BeanDefinition 对象的方法。
获取Bean实例对象
BeanFactory 提供的最主要方法就是getBean(String beanName)
方法,调用该方法能从容器中返回一个特定名称的Bean。
访问容器中关于Bean的基本信息——ListableBeanFactory
BeanFactory 的子类 ListableBeanFactory,提供了访问容器内关于Bean 的基本信息,如Bean的个数,获取某一类型Bean的配置名、查看容器中是否包含某一Bean等方法。
父子级容器——HierarchicalBeanFactory
父子级联IOC容器实现的接口是HierarchicalBeanFactory
,子容器可以通过接口方法访问父容器,由此建立的IOC级联容器体系中,子容器可以访问父容器,而父容器无法访问子容器。
Spring利用父子级联容器实现了很多功能,例如 SpringMVC中,View层的Bean位于一个子容器中,而Service层和Dao层的Bean位于父容器中,这样View层的Bean就可以引用Service层和Dao层的Bean了
拓展接口 —— ConfigurableBeanFactory
这是一个BeanFactory子类系列十分重要的接口,它增强了IOC的可定制性,它定义了设置类加载器、属性编辑器、容器初始化后置处理器等方法
自动装配——AutowireCapableBeanFactory
这个接口定义了容器中的Bean按某种规则(如按名字匹配、按类型匹配等)进行自动装配的方法。
注册单例Bean——SingletonBeanRegistry
这个接口定义了允许在运行期间向容器注册单例Bean的方法。
对于单例Bean对象来说,BeanFactory会缓存Bean实例,所以第二次调用getBean()获取Bean对象时会直接从IOC容器的缓存中获取Bean实例。
Spring 在 DefaultSingletenBeanFactory 类中提供了用于缓存单例Bean 缓存器,这个缓存器使用HashMap实现,单例Bena以beanName作为key进行存储。
BeanFactory依赖日志框架
在初始化 BeanFactory 时,必须提供一种日志框架,例如使用Log4j,这样启动Spring容器时才不会报错。
BeanFactory提供给开发者的操作类——ApplicationContext
ApplicationContext 继承于 HierarchicalBeanFactory 和 ListableBeanFactory 接口,并由此拓展出其它功能接口:
WebApplicationContext
WebApplicationContext是专门为Web应用准备的,它允许从相对于Web根目录的路径中装载配置文件完成初始化工作,从WebApplicationContext中可以获得ServletContext的引用,整个Web应用上下文对象将作为属性放置到ServletContext 中,以便Web应用环境可以访问Spring应用上下文。
(4)Spring Bean的作用域
Spring 3.0 定义了五种Bean的作用域:单例、原型、请求、会话、全局会话
(5)Spring Bean的生命周期
SpringBean的生命周期:实例化 —— 属性赋值 —— 初始化 —— 销毁
实例化: 实例化一个Bean,也就是我们常说的new
属性赋值
BeanNameAware
接口,调用setBeanName(String beanId)
方法BeanFactoryAware
接口,调用setBeanFactory(BeanFacrtory beanFactory)
方法ApplicationContextAware
接口,调用setApplicationContext(ApplicationContext applicationContext)
方法,可以实现设置BeanFactory方法,并比BeanFactoryAware更加优秀BeanPostProcessor
接口,将调用postProcessBeforelnitialization(Object obj, String s)
方法,BeanPostProcessor经常用于Bean内容的修改,并且Bean初始化结束后也会调用这个方法,也可以应用于内存或缓存技术。初始化
BeanPostProcessor
接口,调用postProcessAfterInitialization(Object obj, String s)
方法。销毁
destroy()
方法进行销毁。(6)Spring 依赖注入的方式
Spring 提供的依赖注入的方式:构造器注入、set方法注入、静态工厂注入、实例工厂注入
构造器注入案例
public CatDaoImpl(String message){
this.message = message;
}
<bean id="CatDaoImpl" class="class.CatDaoImpl">
<constructor-arg value="message"></constructor-arg>
</bean>
setter方法注入
public class Id{
private int id;
public int getId(){
return id;
}
public void setId(int id){
this.id = id;
}
}
<bean id="id" class="com.Id">
<property name="id" value="123"></property>
</bean>
静态工厂注入
// 对外提供的静态工厂
public class DaoFactory{
public static final FactoryDao getStaticFactoryDaoImpl(){
return new StaticFactoryDaoImpl;
}
}
//使用静态工厂的类
public class SpringAction{
//注入对象
private DaoFactory staticFactoryDao;
//注入对象的set方法
public void setStaticFactoryDao(FactoryDao staticFactoryDao){
this.staticFactoryDao = staticFactoryDao;
}
}
//给调用类注入静态工厂
<bean name="springAction" class="com.SpringAction">
<property name="staticFactoryDao" ref="staticFatoryDao"></property>
</bean>
//给静态工厂中的成员注入
<bean name="staticFatoryDao" class="com.DaoFactory" factory-method="getStaticFactoryDaoImpl"></bean>
实例工厂注入
//创建一个实例工厂
public class DaoFactory{
public DaoFactory getFactoryDaoImpl(){
return new FactoryDaoImpl();
}
}
//调用类使用实例工厂
public class SpringAction{
private DaoFactory daoFactory;
public void setDaoFactory(DaoFactory daoFactory){
this.daoFactory = daoFactory;
}
}
//为调用类注入实例工厂Bean对象
<bean name="springAction" class="SpringAction">
<property name="daoFactory" ref="daoFactory"></property>
</bean>
//管理实例工厂,先声明注入的类,再配置实例工厂的成员
<bean name="daoFactory" class="com.DaoFactory"></bean>
<bean name="factoryDao" factory-bean="daoFactory" factory-method="getFactoryDaoImpl"></bean>
(7)自动装配的实现方式
Spring装配包括手动装配和自动装配,手动装配是基于 xml 装配、构造方法、setter方法等。
自动装配一共有5种实现方式:
3、Spring AOP 的原理
(1)概念: AOP是一种“横切”技术,将多个类的公共模块抽取到一个模块中,并将其命名为**“Aspect”**,即切面。
(2)AOP的两个部分: 核心关注点、横切关注点
(3)AOP的应用场景:
(4)AOP的核心概念:
(5)AOP的代理方式
Spring 提供了两种AOP的代理方式:JDK接口动态代理、CGLib动态代理
JDK接口动态代理:
InvocationHandler
,通过实现该接口定义横切逻辑,并通过反射机制调用目标类的代码,动态地将横切逻辑和业务逻辑编织在一起InvocationHandler
接口动态创建一个实例,生成目标类的代理对象GCLib动态代理:
GCLib的全称是Code Generation Library,是一个高性能、高质量的代码生成类库,可以在运行期间动态扩展Java类和实现Java接口,CGLib封装了ASM,可以在运行期间生成新的class,对比JDK动态代理,CGLib可以动态代理未实现接口的类。
(6)AOP的实现原理
@Aspect
public class TransactionDemo {
@Pointcut(value="execution(* com.yangxin.core.service.*.*.*(..))")
public void point(){}
@Before(value="point()")
public void before(){
System.out.println("transaction begin");
}
@AfterReturning(value = "point()")
public void after(){
System.out.println("transaction commit");
}
@Around("point()")
public void around(ProceedingJoinPoint joinPoint) throws Throwable{ System.out.println("transaction begin");
joinPoint.proceed();
System.out.println("transaction commit");
}
}
4、Spring MVC 的原理
MVC指的是Modal(模型),View(视图),Controller(控制器),Spring MVC框架是围绕 DispatcherServlet 而设计的,这个Servlet会将请求分发到各个控制器,并支持可配置的处理器映射、视图渲染、本地化、时区与主题渲染等,甚至还能支持文件上传。
(1)浏览器请求的全过程:
(2)MVC常用注解
组件注解: @Controller、@RestController、@Component、@Repository、@Service
请求注解: @RequestMapping、@Autowired、@PathVariable、@RequestParam、@RequestHeader
5、MyBatis缓存
MyBatis拥有两级缓存,默认情况下开启一级缓存,二级缓存由开发者手动开启。
一级缓存是会话级别的缓存,当同一会话调用同一个SQL时,优先从缓存中读取数据。
二级缓存是映射级别的缓存,不同的SQL会话可以共享该缓存
(1)一级缓存的原理
当用户线程第一次发出查询sql,sql的查询结果会写入到sqlSession的一级缓存中,缓存使用的是Map结构,其中key = mapperID + offset + limit + SQL + 所有入参,value = 用户信息
,当同一个sqlSession再次发出同一个sql请求,就从缓存中取出数据,若两次查询中间发生更新操作,则本sqlSession中的一级缓存区域全部清空,所以第二次请求会从数据库中查询,并写入到新的缓存中去。
(2)二级缓存的原理
二级缓存的范围是mapper同命名空间的mapper,mapper以命名空间为单位创建缓存数据结构,结构是 map,通过 CacheExecutor 实现的。
CacheExecutor是Executor的代理对象,所有的查询操作,在CacheExecutor都会先查询缓存,再查询数据库。
(3)如何配置二级缓存