Java之SSM中bean相关知识汇总案例讲解

bean 的生命周期

对象创建

  1. 实例化Bean对象,默认选择无参构造方法,如果只有一个有参构造那么调用有参构造,如果只有多个有参构造那么报错,除非其中一个有参构造添加了@AutoWired注解;
  2. 设置Bean的属性;
  3. 依赖注入以及判断是否实现了Aware相关接口(BeanNameAware, BeanFactoryAware, ApplicationContextAware)
  4. 如果这个 Bean 关联了 BeanPostProcessor 接口,将会调用BeanPostProcessor.postProcessorBeforeInitialization()
  5. 判断是否实现了InitalizingBean接口,实现了就执行 InitalizingBean.afterPropertiesSet() 方法
  6. 如果Bean在配置文件中的定义包含init-method属性(或者添加了@PostConstruct注解),执行指定的方法;
  7. 执行方法BeanPostProcessor.postProcessorAfterInitialization():例如判断有没有实现AOP
  8. 创建对象完毕;

对象销毁

  1. 当要销毁Bean的时候,如果Bean实现了DisposableBean接口,执行destroy()方法。
  2. 当要销毁Bean的时候,如果Bean在配置文件中的定义包含destroy-method属性(或者添加了@PreDestroy注解),执行指定的方法
  3. 销毁对象完毕

Bean 的作用域

spring 支持 5 种作用域,如下:

request:每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTP request内有效。

  • singleton::单例模式,Spring IoC 容器中只会存在一个共享的 Bean 实例,无论有多少个Bean 引用它,始终指向同一对象。该模式在多线程下是不安全的。Singleton 作用域是Spring 中的默认作用域
  • prototype:每次通过 Spring 容器获取 prototype 定义的 bean 时,容器都将创建一个新的 Bean 实例,每个 Bean 实例都有自己的属性和状态,而 singleton 全局只有一个对象。根据经验,对有状态的bean使用prototype作用域,而对无状态的bean使用singleton作用域。
    • 用于与数据库交互的存储数据的bean等,均是有状态的bean。
    • 而仅仅用于操作其他资源的bean,如service controller,就是无状态的bean。
    • 当然,由于spring使用了ThreadLocal进行多线程处理,绝大多数bean都可以声明为singleton作用域。这是后话。
  • session:在一次 Http Session 中,容器会返回该 Bean 的同一实例。而对不同的 Session 请求则会创建新的实例,该 bean 实例仅在当前 Session 内有效。同 Http 请求相同,每一次session 请求创建新的实例,而不同的实例之间不共享属性,且实例仅在自己的 session 请求内有效,请求结束,则实例将被销毁。如用户购物车
  • global-session:全局session作用域,仅仅在基于Portlet的Web应用中才有意义,Spring5中已经没有了。Portlet是能够生成语义代码(例如HTML)片段的小型Java Web插件。它们基于Portlet容器,可以像Servlet一样处理HTTP请求。但是与Servlet不同,每个Portlet都有不同的会话。

bean的循环依赖

什么是循环依赖

一个AService里面引用了BService的一个对象,BService里面又引用了AService的一个对象。

那么在构造AService的bean的时候,会填充属性以及注入依赖,那么就需要注入BService的bean,spring发现BService的bean还没有创建,又会去构造BService的bean。同理BService的bean又需要AService的bean,这时候因为AService的bean还没有构建好,所以他也会去创建AService的bean。一直循环

怎么解决:使用二级缓存

  • singletonObjects:第一级缓存,里面放置的是实例化好的单例对象;这个是一直存在的
  • earlySingletonObjects:第二级缓存,里面存放的是提前曝光的单例对象;就是下面图中的那个缓存

有什么问题

如果上述的bean不存在AOP,那么是没有什么问题的,但是如果存在AOP的话,那么构造出来的bean对象就不是原始对象了,而是AOP生成的代理对象。如果还是使用二级缓存的话,那么B从缓存取的是A的原始对象而不是构造好的A的bean对象

怎么解决

添加一层缓存,singletonFactories:第三级缓存,里面存放的是要被实例化的对象的对象工厂。

主要代码如下:

protected Object getSingleton(String beanName, boolean allowEarlyReference) {
    Object singletonObject = this.singletonObjects.get(beanName);
    if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
        synchronized (this.singletonObjects) {
            singletonObject = this.earlySingletonObjects.get(beanName);
            if (singletonObject == null && allowEarlyReference) {
                ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                if (singletonFactory != null) {
                    singletonObject = singletonFactory.getObject();
                    this.earlySingletonObjects.put(beanName, singletonObject);
                    this.singletonFactories.remove(beanName);
                }
            }
        }
    }
    return (singletonObject != NULL_OBJECT ? singletonObject : null);
}

分析getSingleton()的整个过程,Spring首先从一级缓存singletonObjects中获取。如果获取不到,并且对象正在创建中,就再从二级缓存earlySingletonObjects中获取。如果还是获取不到且允许singletonFactories通过getObject()获取,就从三级缓存singletonFactory.getObject()(三级缓存)获取,如果获取到了则从singletonFactories中移除,并放入earlySingletonObjects中。其实也就是从三级缓存移动到了二级缓存。

总结一下流程:

  • A在第一步实例化对象之后将自己提前曝光到singletonFactories中
  • 在填充属性时发现自己依赖对象B,此时就尝试去get(B),发现B还没有被create,所以走create流程
  • B在填充属性的时候发现自己依赖了对象A,于是尝试get(A),尝试一级缓存singletonObjects(肯定没有,因为A的bean还没有构造完),尝试二级缓存earlySingletonObjects(也没有),尝试三级缓存singletonFactories,由于A通过ObjectFactory将自己提前曝光了,所以B能够通过ObjectFactory.getObject拿到A对象
  • 通过提前引用,直接创建出A的动态代理对象也就是实例化好的A的bean放到第二个缓存中,这样B的bean就直接实例化完成进入一级缓存。
  • 此时回调到A中,A填充属性这一步完成了继续往下执行,因为bean是单例的,所以A不会又去调用动态代理再创建一个bean,而是直接从第二个缓存里拿出实例化好的那个bean出来直接用放进一级缓存中。

到此这篇关于Java之SSM中bean相关知识汇总案例讲解的文章就介绍到这了,更多相关Java之SSM中bean相关知识内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java使用BeanUtils.copyProperties踩坑经历

    1. 原始转换 提起对象转换,每个程序员都不陌生,比如项目中经常涉及到的DO.DTO.VO之间的转换,举个例子,假设现在有个OrderDTO,定义如下所示: public class OrderDTO { private long id; private Long userId; private String orderNo; private Date gmtCreated; // 省略get.set方法 } 有个OrderVO,定义如下所示: public class OrderVO { pr

  • Java 确保某个Bean类被最后执行的几种实现方式

    一.事出有因 ​ 最近有一个场景,因同一个项目中不同JAR包依赖同一个组件,但依赖组件的版本不同,导致无论使用哪个版本都报错(无法同时兼容两个JAR包中所需的方法调用),经过分析发现差异的部份是在一个BEAN中的方法出入参不同而矣,故考虑通过动态替换掉这个存在兼容性的BEAN,换成我们自己继承自该BEAN类并实现适配兼容方法,从而最终解决组件版本不兼容问题: 二.解决方案困境 但在实现的编码过程中发现,原依赖的那个BEAN并不是普通的通过标注@Compent之类的注解实现的注册的BEAN,而是由

  • 利用Java实体bean对象批量数据传输处理方案小结

    javaBean在MVC设计模型中是model,又称模型层,在一般的程序中,我们称它为数据层,就是用来设置数据的属性和一些行为,然后我会提供获取属性和设置属性的get/set方法JavaBean是一种JAVA语言写成的可重用组件.为写成JavaBean,类必须是具体的和公共的,并且具有无参数的构造器. 下面通过本文给大家分享利用Java实体bean对象批量数据传输处理的解决方案. 需求 现在有两方数据库表结构相同,一方A.另一个方B,现想从A处查询出多个表的数据,传输到B地保存起来. 解决方案1

  • IDEA使用GsonFormat完成JSON和JavaBean之间的转换

    最近一直在对接接口,上游返回的都是 JSON 数据,我们需要将这些数据进行保存,我们可以解析成 Map 通过 key 的方式进行获取,然后 set 到实体类对象中,说到这里我开始想吐了,这样就造成了代码过多,没有可读性,如果有100个值,要 get 100次, set 100次吗? 所以最简单的方式是封装成对象,通过对象操作工具进行对象中属性值的映射,但是封装对象过程又繁琐了,属性过多极大的浪费时间,记得初中历史学过的一段话,人和动物最根本的区别就是会不会制造和使用工具,大神和菜鸟之间的差距也莫

  • 如何动态修改JavaBean中注解的参数值

    我这里有一个需求需要修改Person类中的一个属性上的注解的值进行修改,例如: public class Person { private int age; @ApiParam(access="lala") private String name; //get set 方法忽略 } 将@ApiParam(access="lala") 修改为@ApiParam(access="fafa"),经过分析是可以实现的,需要用到动态代理进行操作. 具体源码

  • Java 如何使用@Autowired注解自动注入bean

    Java @Autowired注解自动注入bean annotationWire.xml (一定记得配置context:annotation-config/) <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001

  • 常用json与javabean互转的方法实现

    JSONObject 与 JSONArray区别 JSONObject: { "area": "武汉", "name": "张三", "age": 25 } JSONArray: [{ "area": "武汉", "name": "张三", "age": 25 }, { "area": &

  • Java基础之Bean的创建、定位和使用

    一.前言 Bean是一个由Spring IoC容器实例化.组装和管理的对象.在 Spring 中,类的实例化.依赖的实例化.依赖的传入都交由 Spring Bean 容器控制,而不是用new方式实例化对象.通过非构造函数方法传入依赖等常规方式.这样可以减少垃圾回收对大量实例的回收工作. 在举例中使用到了三个类AAA,BBB和CCC.其中AAA和BBB是平等的两个类,可以相互调用.CCC是以BBB为构造参数的类. 二.自动装配Bean 2.1 注册Bean 为所有想要创建Bean的类添加@Comp

  • Java之SSM中bean相关知识汇总案例讲解

    bean 的生命周期 对象创建 实例化Bean对象,默认选择无参构造方法,如果只有一个有参构造那么调用有参构造,如果只有多个有参构造那么报错,除非其中一个有参构造添加了@AutoWired注解: 设置Bean的属性: 依赖注入以及判断是否实现了Aware相关接口(BeanNameAware, BeanFactoryAware, ApplicationContextAware) 如果这个 Bean 关联了 BeanPostProcessor 接口,将会调用BeanPostProcessor.pos

  • jQuery中Form相关知识汇总

    form中的单行文本获取和失去焦点 复制代码 代码如下: <!DOCTYPE html> <html> <head lang="en">     <meta charset="UTF-8">     <script type="text/javascript" src="../../js/jquery-2.1.3.js"></script>     <

  • JAVA内存空间相关知识汇总

    Java内存分配与管理是Java的核心技术之一,之前我们曾介绍过Java的内存管理与内存泄露以及Java垃圾回收方面的知识,今天我们再次深入Java核心,详细介绍一下Java在内存分配方面的知识.一般Java在内存分配时会涉及到以下区域: ◆寄存器:我们在程序中无法控制 ◆栈:存放基本类型的数据和对象的引用,但对象本身不存放在栈中,而是存放在堆中 ◆堆:存放用new产生的数据 ◆静态域:存放在对象中用static定义的静态成员 ◆常量池:存放常量 ◆非RAM存储:硬盘等永久存储空间 Java内存

  • JAVA 枚举相关知识汇总

    Java 枚举 知识点 概念 enum 的全称为 enumeration, 是 JDK 1.5 中引入的新特性. 在Java中,被 enum 关键字修饰的类型就是枚举类型.形式如下: enum Color { RED, GREEN, BLUE } 如果枚举不添加任何方法,枚举值默认为从0开始的有序数值.以 Color 枚举类型举例,它的枚举常量依次为 RED:0,GREEN:1,BLUE:2. 枚举的好处:可以将常量组织起来,统一进行管理. 枚举的典型应用场景:错误码.状态机等. 枚举类型的本质

  • JAVA 线程通信相关知识汇总

    两个线程之间的通信 多线程环境下CPU会随机的在线程之间进行切换,如果想让两个线程有规律的去执行,那就需要两个线程之间进行通信,在Object类中的两个方法wait和notify可以实现通信. wait方法可以使当前线程进入到等待状态,在没有被唤醒的情况下,线程会一直保持等待状态. notify方法可以随机唤醒单个在等待状态下的线程. 来实现这样的一个功能: 让两个线程交替在控制台输出一行文字 定义一个Print类,有两个方法print1和print2,分别打印一行不同的内容 package c

  • Java基础之反射技术相关知识总结

    一.反射概念 Java的反射(reflection)机制是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法.这种动态获取程序信息以及动态调用对象的功能称为Java语言的反射机制.反射被视为动态语言的关键. 二.反射应用场景 1.几乎所有的框架都会用到反射 2.程序解耦合使用 3.代码更加的优雅 三.反射更多细节 1.Jdk中的位置: java.lang.reflect包下 2.获取字节码方式 //

  • Java集合的总体框架相关知识总结

    一.集合概述 数组其实就是一个集合.集合实际上就是一个容器.可以来容纳其它的数据. 二.集合在开发中的应用 集合是一个容器,是一个载体,可以一次容纳多个对象.在实际开发中,假设连接数据库,数据库当中有10条记录,那么假设把这10条记录查询出来,在Java程序中会将10条数据封装成10个Java对象,然后将10个Java对象放到某一个集合当中,将集合传到前端,然后遍历集合,将一个数据一个数据展现出来 三.集合存储的数据 Java集合中实际存放的只是对象的引用,每个集合元素都是一个引用变量,实际内容

  • Java并发编程之死锁相关知识整理

    一.什么是死锁 所谓死锁是指多个线程因竞争资源而造成的一种僵局(互相等待),若无外力作用,这些进程都将无法向前推进 二.死锁产生的条件 以下将介绍死锁的必要条件,只要系统发生死锁,这些条件必然成立,而只要上述条件之一不满足,就不会发生死锁 互斥条件 进程要求对所分配的资源(如打印机〉进行排他性控制,即在一段时间内某资源仅为一个进程所占有.此时若有其他进程请求该资源,则请求进程只能等待 不可剥夺条件 进程所获得的资源在未使用完毕之前,不能被其他进程强行夺走,即只能由获得该资源的进程自己来释放(只能

  • Java泛型机制与反射原理相关知识总结

    一.泛型的概念 1.1 基础案例 泛型在Java中的应用非常广泛,最常见则是在集合容器中,先看下基础用法: public class Generic01 { public static void main(String[] args) { Map<Integer,String> map = new HashMap<>() ; map.put(88,"hello") ; // map.put("99","world") ;

  • 深入理解Java显式锁的相关知识

    目录 一.显式锁 二.Lock的常用api 三.Lock的标准用法 四.ReentrantLock(可重入锁) 五.ReentrantReadWriteLock(读写锁) 六.Condition 一.显式锁 什么是显式锁? 由自己手动获取锁,然后手动释放的锁. 有了 synchronized(内置锁) 为什么还要 Lock(显示锁)? 使用 synchronized 关键字实现了锁功能的,使用 synchronized 关键字将会隐式地获取锁,但是它将锁的获取和释放固化了,也就是先获取再释放.

随机推荐