IoC(Inverse of Control: 控制反转)是⼀种设计思想,就是将原本在程序中⼿动创建对象的控制权,交由 Spring 框架来管理。IoC 在其他语⾔中也有应用,并非Spring 特有。
IoC 容器是 Spring 用来实现 IoC 的载体, IoC 容器实际上就是个 Map(key,value),Map 中存放的是各种对象。将对象之间的相互依赖关系交给 IoC 容器来管理,并由 IoC 容器完成对象的注⼊。这样可以很大程度上简化应用的开发,把应用从复杂的依赖关系中解放出来。IoC 容器就像是⼀个工厂⼀样,当我们需要创建⼀个对象的时候,只需要配置好配置文件 / 注解即可,完全不用考虑对象是如何被创建出来的。
DI 依赖注入
DI:(Dependancy Injection:依赖注入) 站在容器的角度,将对象创建依赖的其他对象注入到对象中。
AOP(Aspect-Oriented Programming: 面向切面编程) 能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任(例如事务处理、日志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可拓展性和可维护性。
Spring AOP 就是基于动态代理的,如果要代理的对象,实现了某个接口,那么 Spring AOP 会使用 JDKProxy,去创建代理对象,而对于没有实现接口的对象,就无法使用 JDK Proxy 去进行代理了,这时候 Spring AOP 会使用基于 asm 框架字节流的 Cglib 动态代理 ,这时候 Spring AOP 会使用 Cglib ⽣成⼀个被代理对象的⼦类来作为代理。
单例对象: singleton
总结:单例对象的生命周期和容器相同
多例对象: prototype
出生:使用对象时 spring 框架为我们创建
活着:对象只要是在使用过程中就一直活着
死亡:当对象长时间不用且没有其它对象引用时,由 java 的垃圾回收机制回收
IOC 容器初始化加载 Bean 流程:
@Overridepublic
void refresh() throws BeansException, IllegalStateException {
synchronized (this.startupShutdownMonitor) {
// 第一步:刷新前的预处理
prepareRefresh();
//第二步: 获取BeanFactory并注册到
BeanDefitionRegistry ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
// 第三步:加载BeanFactory的预准备工作(BeanFactory进行一些设置,比如context的类加载器等)
prepareBeanFactory(beanFactory);
try {
// 第四步:完成BeanFactory准备工作后的前置处理工作
postProcessBeanFactory(beanFactory);
// 第五步:实例化BeanFactoryPostProcessor接口的Bean
invokeBeanFactoryPostProcessors(beanFactory);
// 第六步:注册BeanPostProcessor后置处理器,在创建bean的后执行
registerBeanPostProcessors(beanFactory);
// 第七步:初始化MessageSource组件(做国际化功能;消息绑定,消息解析);
initMessageSource();
// 第八步:注册初始化事件派发器
initApplicationEventMulticaster();
// 第九步:子类重写这个方法,在容器刷新的时候可以自定义逻辑
onRefresh();
// 第十步:注册应用的监听器。就是注册实现了ApplicationListener接口的监听器
registerListeners();
//第十一步:初始化所有剩下的非懒加载的单例bean 初始化创建非懒加载方式的单例Bean实例(未设置属性)
finishBeanFactoryInitialization(beanFactory);
//第十二步: 完成context的刷新。主要是调用LifecycleProcessor的onRefresh()方法,完成创建
finishRefresh();
}
……
}
默认作用域是 singleton,多个线程访问同一个 bean 时会存在线程不安全问题
保障线程安全方法:
ThreadLocal:
每个线程中都有一个自己的 ThreadLocalMap 类对象,可以将线程自己的对象保持到其中,各管各的,线程可以正确的访问到自己的对象。
将一个共用的 ThreadLocal 静态实例作为 key,将不同对象的引用保存到不同线程的 ThreadLocalMap 中,然后在线程执行的各处通过这个静态 ThreadLocal 实例的 get() 方法取得自己线程保存的那个对象,避免了将这个对象作为参数传递的麻烦。
循环依赖其实就是循环引用,也就是两个或者两个以上的 Bean 互相持有对方,最终形成闭环。比如 A 依赖于 B,B 又依赖于 A
Spring 中循环依赖场景有:
懒加载 @Lazy 解决循环依赖问题
Spring 启动的时候会把所有 bean 信息 (包括 XML 和注解) 解析转化成 Spring 能够识别的 BeanDefinition 并存到 Hashmap 里供下面的初始化时用,然后对每个 BeanDefinition 进行处理。普通 Bean 的初始化是在容器启动初始化阶段执行的,而被 lazy-init=true 修饰的 bean 则是在从容器里第一次进行 context.getBean() 时进行触发。
三级缓存解决循环依赖问题
声明 bean 的注解
@Component 通用的注解,可标注任意类为 Spring 组件
@Service 在业务逻辑层使用(service 层)
@Repository 在数据访问层使用(dao 层)
@Controller 在展现层使用,控制器的声明(controller 层)
注入 bean 的注解
@Autowired:默认按照类型来装配注入,@Qualifier:可以改成名称
**@Resource:**默认按照名称来装配注入,JDK 的注解,新版本已经弃用
@Autowired 注解原理
@Autowired 的使用简化了我们的开发,
实现 AutowiredAnnotationBeanPostProcessor 类,该类实现了 Spring 框架的一些扩展接口。
实现 BeanFactoryAware 接口使其内部持有了 BeanFactory(可轻松的获取需要依赖的的 Bean)。
实现 MergedBeanDefinitionPostProcessor 接口,实例化 Bean 前获取到 里面的 @Autowired 信息并缓存下来;
实现 postProcessPropertyValues 接口, 实例化 Bean 后从缓存取出注解信息,通过反射将依赖对象设置到 Bean 属性里面。
@SpringBootApplication
@SpringBootApplication
public class JpaApplication {
public static void main(String[] args) {
SpringApplication.run(JpaApplication.class, args);
}
}
其中 @EnableAutoConfiguration 是关键 (启用自动配置),内部实际上就去加载 META-INF/spring.factories 文件的信息,然后筛选出以 EnableAutoConfiguration 为 key 的数据,加载到 IOC 容器中,实现自动配置功能!
它主要加载了 @SpringBootApplication 注解主配置类,这个 @SpringBootApplication 注解主配置类里边最主要的功能就是 SpringBoot 开启了一个 @EnableAutoConfiguration 注解的自动配置功能。
@EnableAutoConfiguration 作用:
它主要利用了一个
EnableAutoConfigurationImportSelector 选择器给 Spring 容器中来导入一些组件。
@Import(EnableAutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration
@Controller 声明该类为SpringMVC中的Controller
@RequestMapping 用于映射Web请求
@ResponseBody 支持将返回值放在response内,而不是一个页面,通常用户返回json数据
@RequestBody 允许request的参数在request体中,而不是在直接连接在地址后面。
@PathVariable 用于接收路径参数
@RequestMapping("/hello/{name}")申明的路径,将注解放在参数中前,即可获取该值,通常作为Restful的接口实现方法。
@Insert :插入sql ,和xml insert sql语法完全一样
@Select :查询sql, 和xml select sql语法完全一样
@Update :更新sql, 和xml update sql语法完全一样
@Delete :删除sql, 和xml delete sql语法完全一样
@Param :入参
@Results :设置结果集合
@Result :结果
@ResultMap :引用结果集合
@SelectKey :获取最新插入id
简单的说就是 #{} 是经过预编译的,是安全的,
**${} 是未经过预编译的,仅仅是取变量的值,是非安全的,存在 SQL 注入。在编写 mybatis 的映射语句时,尽量采用 **“#{xxx}” 这样的格式。如果需要实现动态传入表名、列名,还需要做如下修改:添加属性 statementType=“STATEMENT”,同时 sql 里的属有变量取值都改成 ${}
Hibernate 是一个开放源代码的对象关系映射框架, 它对 JDBC 进行了非常轻量级的对象封装, 建立对象与数据库表的映射。是一个全自动的、完全面向对象的持久层框架。
Mybatis 是一个开源对象关系映射框架,原名:ibatis,2010 年由谷歌接管以后更名。是一个半自动化的持久层框架。
区别:
开发方面
在项目开发过程当中,就速度而言:
hibernate 开发中,sql 语句已经被封装,直接可以使用,加快系统开发;
Mybatis 属于半自动化,sql 需要手工完成,稍微繁琐;
但是,凡事都不是绝对的,如果对于庞大复杂的系统项目来说,复杂语句较多,hibernate 就不是好方案。
sql 优化方面
Hibernate 自动生成 sql, 有些语句较为繁琐,会多消耗一些性能;
Mybatis 手动编写 sql,可以避免不需要的查询,提高系统性能;
对象管理比对
Hibernate 是完整的对象 - 关系映射的框架,开发工程中,无需过多关注底层实现,只要去管理对象即可;
Mybatis 需要自行管理映射关系;
@EnableTransactionManagement
@Transactional
注意事项:
①事务函数中不要处理耗时任务,会导致长期占有数据库连接。
②事务函数中不要处理无关业务,防止产生异常导致事务回滚。
事务传播属性
1、Spring 中的设计模式
单例设计模式 : Spring 中的 Bean 默认都是单例的。
工厂设计模式 : Spring 使用工厂模式通过 BeanFactory 、 ApplicationContext 创建 bean 对象。
代理设计模式 : Spring AOP 功能的实现。
观察者模式: Spring 事件驱动模型就是观察者模式很经典的⼀个应用。
适配器模式:Spring AOP 的增强或通知 (Advice) 使用到了适配器模式、spring MVC 中也是用到了适配器模式适配 Controller 。