联通数科面试准备

news2024/11/22 21:34:08

Spring中Bean的生命周期

Spring Bean的生命周期全过程分为5个阶段,创建前准备阶段、创建实例阶段、依赖注入阶段和容器缓存阶段以及销毁实例阶段。
阶段1:创建前准备阶段这个阶段主要是在开始Bean加载之前,从Spring上下文中去获取相关的配置并且解析,然后找到Bean有关的配置内容,比如说"init-method"容器在初始化Bean的时候去调用‍这个方法。‍"destory-method"容器在销毁Bean的时候去调用这个方法以及BeanFactoryPostProcessor这个类的Bean加载过程中的前置和后置的处理,这些类或者配置是Spring提供给开发者用来实现Bean加载过程中的一个扩展的机制。在很多和Spring集成的中间件中也经常‍‍使用到,比如Dubbo。

阶段2:创建实例阶段这个阶段主要是通过反射来创建Bean的实际对象,并且扫描和解析Bean的声明的一些属性。‍

阶段3:依赖注入阶段在这个阶段会检测被实例化的Bean是否存在其他依赖,如果存在其他依赖的话,‍‍就需要将这些依赖注入到Bean里面,比如说通过读取"@Autowired", “@Setter” 等注解去完成依赖注入的配置,‍‍这个阶段会触发一些扩展的调用,如常见的扩展类BeanPostProcessors它用来实现Bean初始化‍前后的一个回调。如InitializingBean的afterPropertiesSet()方法,它可以给属性赋值。还有BeanFactoryAware等等。‍

‍阶段4:容器缓存阶段容器缓存阶段主要是把Bean保存到IoC容器中缓存起来,‍‍到了这个阶段Bean就可以去被开发者使用了,这个阶段涉及到的操作有常见的“init-method”属性配置的方法会在这个阶段被调用,比如BeanPostProcessors它的后续处理方法postProcessAfterInitialization也会在这个阶段被触发。
步骤一:执行前置处理方法,当正在初始化的Bean对象被传递进来,postProcessBeforeInitialization()会吸纳与执行初始化调用方法执行,所有xxxAware接口的注入,即对引用的容器级别对象的属性的赋值/依赖注入操作,就是在这一步完成的。
步骤二:执行初始化调用方法–invokeInitMethods:
如果在Spring配置文件中配置了init-method 属性,会自动对该Bean进行增强实现步骤中的初始化步骤。“执行初始化调用方法”这一个步骤在Java 语言里边其实是没有多大实际意义的。 因为在其他编程语言里边比如说Python 里边,会在这一个步骤中进行属性赋值工作,但是在Java里边,刚才已经刚刚完成了属性赋值的工作了。
步骤三:该后置处理器的执行是在 “执行初始化调用方法” 后面进行执行,主要是判断该bean是否需要被AOP代理增强,如果需要的话,则会在该步骤返回一个代理对象。这个函数会在 “执行初始化调用方法” 完成后执行,因此称为后置处理方法。
这一步完成以后,就可以正常使用这个 Bean 了。

(5)销毁 Destruction

详解:
(1)通过xml配置或者注解配置的类,得到BeanDefinition;
(2)通过BeanDefinition反射创建Bean对象
(3)对Bean对象的属性进行填充。
(4)回调实现了Aware接口的方法,如BeanNameAware;
(5)调用BeanPostProcessor的初始化前方法。
(6)调用inti初始化方法。
(7)调用BeanPostProcessor的初始化方法,此处会进行AOP;
(8)将创建的Bean对象放入一个Map。
(9)业务使用Bean对象。
(10)Spring 容器关闭时调用DisposableBean的destory()方法。

Spring如何解决循环依赖问题

一级缓存:singletonObjects存储的是所有创建好了的单例Bean(实例化、依赖注入、初始化完成的bean实例)。
二级缓存:earlySingletonObjects保存的是完成实例化,但是还未进行属性注入以及初始化的对象。
三级缓存:singletonFactories提前暴露的一个单例工厂,二级缓存中存储的就是从这个工厂中获取到的对象。

三级缓存解决循环依赖流程:‘
(1)获取A时首先会尝试从一级缓存sigletonOjects中获取,有就返回,没有就找“二级缓存”;
(2)获取不到就从二级缓存中获取,有就返回,没有就找“三级缓存”;
(3)若还没有则从三级缓存singletonFactories中获取,找到了,就获取对象,放到“二级缓存”,从“三级缓存”移除。
(4)还有没有则再次创建该对象。
(5)会依次执行doGetBean->createBean->createBeanInstance并使用构造器实例化。
(6)在尝试给A进行初始化时,由于B不存在无法完成初始化,则将A的代理工厂放入三级缓存中,进行创建B的创建流程。
(7)与之前的过程相似,在第三级缓存中放入beanName和表达式sharedInstance,进入B的初始化过程。
(8)由于在第三级缓存中可以找到A的代理工厂,直接从“三级缓存”中拿到 A 的代理工厂,获取 A 的代理对象,放入“二级缓存”,并清除“三级缓存”;
(9)则B可以完成初始化,完成A对象的属性注入,将B放入一级缓存,然后再填充A的其他属性,以及A的其他步骤(包括AOP),完成A的初始化功能(包括AOP),完成对A的初始化功能。
(9)同时完成A的初始化,并删除二级缓存中的半成品A,将A放入一级缓存。

总结:
Spring通过三级缓存解决了循环依赖,其中一级缓存是单例池,二级缓存是早期曝光对象earlySingletonObjects,三级缓存为早期曝光对象工厂(singletonFactories)。当A、B两个类发生循环依赖时,在A完成实例化后,就使用实例化后的对象去创建一个对象工厂,添加到三级缓存中,如果A被AOP代理,那么通过这个工厂获取到的就是A代理后的对象,如果A没有被AOP代理,那么这个工厂获取到的就是A实例化后的对象。当A进行属性注入时,回去创建B,同时B有依赖了A,所以创建B的同时又去调用getBean(a)来获取所需要的依赖,此时的getBean(a)会从缓存中获取:
(1)先获取到三级缓存中的工厂。
(2)调用对象工厂的getObject方法来获取到对应的对象,得到这个对象后将其注入到B中,紧接着B就会走完它的声明周期流程,包括初始化、后置处理器。
(3)当B创建完成后,会将B再次注入到A中,此时A再完成它的整个生命周期,至此,循环依赖结束。
在这里插入图片描述

Spring AOP的实现原理

Spring的AOP是通过动态代理实现的,如果我们为Spring的某个bean配置了切面,那么Spring在创建这个bean的时候,实际上创建的就是这个bean的一个代理对象,我们后续对bean中的方法的调用,实际上调用的就是代理类重写的代理方法。而Spring的AOP使用了两种动态代理,分别是JDK的动态代理,以及CGLib的动态代理。

JDK动态代理:
如果目标类实现了接口,Spring AOP会选择使用JDK动态代理目标类。JDK动态代理的代理类根据根据目标类实现的接口动态生成,不需要自己编写。生成的动态代理类和目标类都实现相同的接口。JDK动态代理的核心是InvocationHandler接口和Proxy类。JDK动态代理的缺点是目标类必须有实现的接口。如果这个类没有实现接口,那么这个类就不能使用JDK动态代理。

CGLIB动态代理:
如果目标类没有实现接口,那么Spring AOP会选择使用CGLIB来动态代理目标类。CGLIB英文全称是Code Generation Library,可以再运行时动态生成类的字节码,动态创建目标类的子类对象,在子类对象中增强目标类。CDLIB是通过继承的方式实现的动态代理,因此如果某个类被标记为final,那么它无法使用CGLIB做动态代理。相对于JDK动态代理,CGLIB动态代理的优点是:目标类不需要实现特定的几口,更加灵活。

如果目标对象实现了接口,默认情况下会使用JDK动态代理实现AOP,但是也可以强制使用CGLIB实现AOP。如果目标对象没有实现接口,必须采用CGLIB做动态代理。

动态代理的作用

动态代理是设计模式的一种:
(1)调用方不用知道具体目标类是什么(安全、解耦),只需要知道统一的代理类。
(2)可以对目标类做增强(在方法运行:前、后、成功、异常时,做额外处理,并且代码相对独立,也是为了解耦)

java如何删除list中的元素

(1)正序删除
使用正序删除,如果只删除至多一个元素,那只需要在删除这个元素后使用break语句跳出循环即可,如果要删除多个元素,若不注意控制当前列表的size和下一个元素的index,容易报java.lang.IndexOutOfBoundsException异常。

public static void remove(List<String> list, String target) {    
for(int i = 0, length = list.size(); i < length; i++){
        String item = list.get(i);        
        if(target.equals(item)){
            list.remove(item);
            length--;
            i--;
        }
    }
}

(2)倒序删除
倒序删除可以客服正序删需要额外管理列表size和下一个元素index的问题。

public static void remove(List<String> list, String target) {    
for(int i = list.size() - 1; i >= 0; i--){
        String item = list.get(i);
        if(target.equals(item)){
            list.remove(item);
        }
    }
}

(3)迭代器remove()方法删除

public static void remove(List<String> list, String target) {
    Iterator<String> iter = list.iterator();
    while (iter.hasNext()) {
        String item = iter.next();
            if (item.equals(target)) {
            iter.remove();
        }
    }
}

(4)CopyOnWriteArrayList线程安全删除
利用CopyOnWrite容器。CopyOnWrite容器就是写时复制容器。通俗的理解就是当我们往一个容器添加元素的时候,不直接往当前元素添加,而是先将当前容器进行cooy,复制出一个新的容器,然后往新的容器里面添加元素,添加完元素后,再讲原来的容器的引用指向新的容器。这样做的好处是我们可以对CopyOnWrite容器进行并发的读,而不需要加锁,因为当前容器不会添加任何元素。所以CopyOnWrite容器也是一种读写分离的思想,读与写不同的容器。

public static List<String> remove(ArrayList<String> list, String target) {
    CopyOnWriteArrayList<String> cowList = new CopyOnWriteArrayList<String>(list);
    for (String item : cowList) {
            if (item.equals(target)) {
            cowList.remove(item);
        }
    }
                return cowList;
}

(5)增强for循环删除
增强for循环中删除元素后继续循环会报java.util.ConcurrentModification异常,因为元素在使用的时候发生了并发的修改,导致异常抛出,但是删除完毕马上使用break语句跳出循环,则不会触发报错,所以它只适合删除一个元素。

public static void remove(List<String> list, String target) {
    for (String item : list) {
            if (item.equals(target)) {
            list.remove(item);
                        break;
        }
    }
}

(6)stream API filter
java8引入的stream API带来了新的比较简洁的删除List元素的方法filter,该方法不会改变原List对象,会返回新的对象。

HashMap与HashTable的区别

(1)线程安全:HashMap是非线程安全的,HashTable是线程安全的。
(2)效率: 因为线程安全的问题,HashMap 要比 HashTable 效率高一点。另外,HashTable 基本被淘汰,不要在代码中使用它;
(3) 对Null key 和Null value的支持: HashMap 中,null 可以作为键,这样的键只有一个,可以有一个或多个键所对应的值为 null。但是在HashTable 中 put 进的键值只要有一个 null,直接抛NullPointerException。
(4)4. 初始容量大小和每次扩充容量大小的不同 : ①创建时如果不指定容量初始值,Hashtable 默认的初始大小为11,之后每次扩充,容量变为原来的2n+1。HashMap 默认的初始化大小为16。之后每次扩充,容量变为原来的2倍。②创建时如果给定了容量初始值,那么 Hashtable 会直接使用你
给定的大小,而 HashMap 会将其扩充为2的幂次方大小。也就是说 HashMap 总是使用2的幂作为哈希表的大小。
5. 底层数据结构: JDK1.8 以后的 HashMap 在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为8)时,将链表转化为红黑树,以减少搜索时间。Hashtable 没有这样的机制。

说一下HashMap的实现原理

HashMap概述:HashMap是基于哈希表的Map接口的非同步实现。此实现提供所有可选地映射操作,并且允许使用null值和null键。此类并不保证映射的顺序,特别是它不保证顺序恒久不变。
HashMap的数据结构:在java编程语言中,基本的数据结构就是两种,一个是数组,另一个是模拟指针(引用),所有数据结构都可以用这两种基本结构来进行构造,Hashmap也不例外。HashMap实际上是一个链表散列的数据结构,即数组和链表的集合。
HashMAP是基于Hash算法实现的:
(1)当我们向Hashmap中put元素的时候,利用key的hashcode重新hash出当前对象的元素在数组中的下标。
(2)存储时,如果出现hash值相同的key,此时有两种情况。如果key相同,此时覆盖原值。如果key不相同,也就是出现hash冲突,则将key-value放入链表中。
(3)获取时,之际找到hash值对应的下标,再进一步判断key是否相同,从而找到对象的值。
(4) 理解了以上过程就不难明白HashMap是如何解决hash冲突的问题,核心就是使用了数组的存储方式,然后将冲突的key的对象放入链表中,一旦发现冲突就在链表中做进一步的对比。

jdk1.8中对HashMap实现了优化,当链表中的节点超过8个之后,该联保转化为红黑树来提高查询效率,由曾经的O(N)转化为O(logn)

简单总结下HashMap是使用了哪些方法来有效解决哈希冲突的:

(1)使用链地址法(散列表)来链接具有相同哈希值的数据。
(2)使用二次扰动函数(hash函数)来降低哈希冲突的概率,使得数据分布更加均匀。
(3)当链表长度超过八个节点时,使用红黑树来降低时间复杂度,使得遍历更快。

ConcurrentHashMap

Segment段:
整个ConcurrentHashMap是由一个个的Segment组成,Segment代表部分或者一段的意思,很多地方会将其描述非分段锁。
线程安全:
简单理解就是,ConcurrentHashMap是一个Segment数组,Segment通过继承ReenTranLock拉进行加锁,所以每次需要加锁的操作锁住的是一个Segment,这样就保证了每个Segmnet是线程安全的,也就实现了全局线程安全。
并行度concurrencyLevel:并行级别、并发数、Segment数。默认是16,也就是说ConcurrentHashMap有16个分段锁,所以理论上最多支持16个线程并发写,只要他们的操作分布在不同的Segment上,这个值是可以再初始化的时候设置为其他值,但是一旦初始化后,不能进行扩容。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/519179.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何使用Alchemy开发NFT智能合约(ERC721)

&#x1f978; 本教程翻译自官网&#xff1a;[https://docs.alchemy.com/docs](https://docs.alchemy.com/docs)。对部分内容进行了修改。教程中所有实例经过本人实践&#xff0c;代码可见&#xff1a;https://github.com/ChuXiaoYi/web3Study 使用Solidity开发智能合约并在区块…

2022 第十三届蓝桥杯大赛软件赛决赛, 国赛,C/C++ 大学B组题解

2022 第十三届蓝桥杯大赛软件赛决赛, 国赛&#xff0c;C/C 大学B组题解 文章目录 第1题 —— 2022 &#xff08;5分&#xff09;第2题 —— 钟表 &#xff08;5分&#xff09;第3题 —— 卡牌 &#xff08;10分&#xff09;第4题 —— 最大数字 &#xff08;10分&#xff09;第…

bbys_tu_2016

1,三连 思路&#xff1a;栈溢出 2&#xff0c;IDA分析 利用函数&#xff1a; 思路:ret2text 偏移&#xff1a;24 3&#xff0c;payload from pwn import * context.log_level"debug"rremote(node4.buuoj.cn,29195)flag 0x804856Dpayload 24 * a p32(flag) r.se…

Unittest单元测试框架之unittest构建测试套件

构建测试套件 在实际项目中&#xff0c;随着项目进度的开展&#xff0c;测试类会越来越多&#xff0c;可是直到现在我 们还只会一个一个的单独运行测试类&#xff0c;这在实际项目实践中肯定是不可行的&#xff0c;在 unittest中可以通过测试套件来解决该问题。 测试套件&…

七、Zookeeper注册中心

目录 1、下载Zookeeper的服务jar包 2、下载好jar包后解压放到合适的目录&#xff08;目录最好不要有中文及空格&#xff09; 3、进入解压后的conf目录&#xff0c;复制zoo_sample.cfg文件并重命名为zoo.cfg&#xff0c;修改zoo.cfg文件内容如下 4、运行bin目录下的zkServer…

split,paste,eval命令及正则表达式

一、split命令 将 linux 下的一个大文件拆分成若干小文件 1.语法格式 格式&#xff1a;split 选项 参数 原始文件 拆分后文件名前缀 常用选项: -l&#xff1a;以行数拆分 -b&#xff1a;以大小拆分 2.命令演示 2.1选项 -l &#xff1a;以行数分隔 cat -n anaconda-ks.cfg…

Windows系统安装好MongoDB后运行方法

文章目录 1、先找到安装MongoDB的文件位置的bin文件夹&#xff1a;2、找到data/db文件夹位置&#xff08;如果data文件夹中没有db文件夹需要创建一个&#xff09;&#xff1a;3、在刚刚打开的cmd窗口中运行以下命令&#xff1a;4、再另外从bin文件夹位置开一个cmd窗口&#xff…

图书管理系统(Java简单版)(完整代码+详解)

目录 详解&#xff1a; BookList类&#xff1a; InOperation接口 User类&#xff08;父类&#xff09; 和 Main类&#xff08;这俩要一起看&#xff09; 完整代码 book包 Book类 BookList类 operation包 AddBook类 BorrowBook类 DeleteBook类 FindBook类 Pr…

ESP32 FreeRTOS学习总结

2023.5.11 1.Task 创建任务常用API&#xff1a; 任务函数描述xTaskCreate()使用动态的方法创建一个任务xTaskCreatePinnedToCore指定任务的运行核心(最后一个参数)vTaskDelete(NULL)删除当前任务 BaseType_t xTaskCreate(TaskFunction_t pxTaskCode, // 任…

【HBase】架构

文章目录 整体架构Master负载均衡器元数据管理器预写日志处理器 Region ServerZookeeperHDFS Master架构Meta 表格 RegionServer 架构MemStoreWALBlockCache 读写流程HFile结构写流程读操作 整体架构 Master 实现类为 HMaster。 负责监控集群中所有的 RegionServer 实例。 &…

在Fedora-Workstation-Live-x86_64-36-1.5中使用佳能喷墨打印机ip2780

在Fedora-Workstation-Live-x86_64-36-1.5中使用佳能喷墨打印机ip2780 操作系统是64位的Fedora-Workstation-Live-x86_64-36-1.5.iso&#xff0c;实物打印机是佳能ip2780&#xff0c;USB接口 应用程序——其它——设置——打印机——解锁——输入root密码——将打印机USB插入电…

Redis数据结构——动态字符串、Dict、ZipList

一、Redis数据结构-动态字符串 我们都知道Redis中保存的Key是字符串&#xff0c;value往往是字符串或者字符串的集合。可见字符串是Redis中最常用的一种数据结构。 不过Redis没有直接使用C语言中的字符串&#xff0c;因为C语言字符串存在很多问题&#xff1a; 获取字符串长度…

智慧产业城彰显中国智造魅力,中联重科踏出“走上去”的关键一步

5月11日至14日&#xff0c;中联重科在长沙举行了“科技献礼新时代”系列活动。 借着活动&#xff0c;松果财经实地探访了全球规模最大、品种最全的工程机械产业基地&#xff0c;领略了前沿技术赋能先进制造的魅力。 作为湖南省“一号工程”和“十大重点项目之首”&#xff0c…

好家伙,又一份牛逼笔记面世了...

最近网传的一些裁员的消息&#xff0c;搞的人心惶惶。已经拿到大厂offer的码友来问我&#xff1a;大厂还能去&#xff0c;去了会不会被裁。 还在学习的网友来问我&#xff1a;现在还要冲互联网么&#xff1f; 我是认为大家不用恐慌吧&#xff0c;该看啥看啥&#xff0c;该学啥…

你一定要知道的unittest自动化测试框架详解

目录 框架的概念 Unittest单元测试框架 常用的assert语句 unittest创建测试代码的方式&#xff1a; unittest构建测试套件&#xff08;测试用例集合&#xff09;&#xff1a; unittest忽略测试用例&#xff1a; 运行测试集 批量执行测试用例 生成HTMLTestRunner测试报告…

【运维知识进阶篇】集群架构-Nginx基础(安装+启动+配置+多业务实现+日志管理)

本篇文章介绍下Nginx有关内容&#xff0c;Nginx是一个开源且高性能、可靠的Http Web服务、代理服务。 开源&#xff1a;直接获取源代码&#xff0c;高性能&#xff1a;支持海量并发&#xff0c;可靠&#xff1a;服务稳定 Web服务有很多&#xff0c;选择Nginx是因为他的轻量化…

【软件测试】测试开发的一生之敌-BUG

文章目录 1.前言2.如何描述/创建一个BUG3.BUG的级别4.BUG的生命周期5.跟开发产生争执怎么办 1.前言 BUG相比大家都知道,程序运行出错或者与预期不符就是BUG.现在我们来用测试人员的角度来看待BUG. 2.如何描述/创建一个BUG 测试人员要测试开发人员的代码,找出开发人员可能忽略…

敏态开发在大兴机场数字化转型中的实践

一、最大事业是育人 大兴机场是一个年轻的企业&#xff0c;作为一个企业来讲&#xff0c;最宝贵的就是人才。我们在2017年开始社招大学生&#xff0c;到目前为止&#xff0c;公司有一半都是30岁左右的年轻人&#xff0c;并且每年都会招几十个。年轻人特别想做事&#xff0c;而…

C++之内存管理及函数模版

C中的内存管理机制和C语言是一样的&#xff0c;但在具体内存管理函数上&#xff0c;C语言的malloc已经无法满足C面向对象销毁的需求&#xff0c;于是祖师爷在C中新增了一系列内存管理函数&#xff0c;即 new 和 delete 著名段子&#xff1a;如果你还没没有对象&#xff0c;那就…