- 臨近秋招,備戰暑期實習,祝大家每天進步億點點!Day21
- 推薦一篇囧輝大佬的 Spring 相關面試題:面試必問的 Spring,你懂了嗎?

1、JDK 動態代理和 CGLIB 代理有什么區別?
-
JDK 動態代理主要是針對類實作了某個介面,AOP 則會使用 JDK 動態代理,他基于反射的機制實作,生成一個實作同樣介面的一個代理類,然后通過重寫方法的方式,實作對代碼的增強,
-
而如果某個類沒有實作介面,AOP 則會使用 CGLIB 代理,他的底層原理是基于 ASM 第三方框架,通過修改位元組碼生成一個子類,然后重寫父類的方法,實作對代碼的增強,
詳細分析參考我的博客:【Java萌新】面試常問設計模式——代理模式
2、FactoryBean、BeanFactory、ApplicationContext 有什么區別?
- BeanFactory:是一個 Bean 工廠,使用簡單工廠模式,是 Spring IoC 容器頂級介面,是用于管理 Bean 的工廠,最核心的功能是通過
getBean()方法加載 Bean 物件,通常我們不會直接使用該介面,而是使用其子介面 ApplicationContext, - FactoryBean:是一個工廠 Bean,使用了工廠方法模式,實作該介面的類可以自己定義要創建的 Bean 實體,只需要實作它的
getObject()方法即可, - ApplicationConext:是 BeanFactory 的子介面,擴展了 BeanFactory 的功能(高級 IOC 容器),
3、說一說Spring Bean 的生命周期?
Spring Bean 生命周期簡單概括為 5 個階段:
- Bean 的實體化階段:創建一個 Bean 物件,
- Bean 實體的屬性填充階段:為 Bean 實體的屬性賦值,
- Bean 實體的初始化階段:對 Bean 實體進行初始化,
- Bean 實體的正常使用階段,
- Bean 實體的銷毀階段:容器關閉后,將 Bean 實體銷毀,
4、依賴注入的實作方法,以及相關注解?
- 構造方法注入、Setter 方法注入、介面注入 三種,
依賴注入的相關注解
@Autowired:自動按型別注入,如果有多個匹配則按照指定 Bean 的 id 查找,查找不到會報錯,@Qualifier:在自動按照型別注入的基礎上再按照 Bean 的 id 注入,給變數注入時必須搭配@Autowired,給方法注入時可單獨使用,@Resource:直接按照 Bean 的 id 注入,只能注入 Bean 型別,@Value:用于注入基本資料型別和 String 型別,
5、什么是 Spring IOC ?
IOC 即控制反轉,簡單來說就是把原來代碼里需要實作的物件創建、依賴反轉給容器來幫忙實作,Spring 中管理物件及其依賴關系都是通過 Spring 的 IOC 容器實作的,
IOC 的實作方式有依賴注入和依賴查找,由于依賴查找使用的很少,因此 IOC 也叫做依賴注入,
我們之前在創建一個物件的時候都是直接 new 一個物件實體,而有了 IOC ,物件實體的創建都交給容器去實作即可,
6、Spring IOC 容器的構建流程(初始化程序)
我們以 XML 方式的容器初始化為例:
-
通過
ClassPathXmlApplicationContext,去創建ApplicationContext容器物件:ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");ClassPathXmlApplicationContext創建容器物件時,構造方法做了如下兩件事:- ① 呼叫父容器的構造方法為容器先設定好 Bean 資源加載器,
- ② 呼叫父類的
setConfigLocations()方法設定 Bean 配置資訊的定位路徑, - ③ 呼叫父類 AbstractApplicationContext 的
refresh()方法啟動整個 IOC 容器對 Bean 的載入,在創建 IOC 容器前如果已有容器存在,需要把已有的容器銷毀,保證在refresh()方法后使用的是新創建的 IOC 容器,
-
容器創建完成后,通過
loadBeanDefinitions()方法加載 Bean 配置資源,該方法在加載資源時,首先決議組態檔路徑,讀取組態檔的內容,然后通過 XML 決議器將 Bean 的配置資訊轉換成檔案物件,之后按照 Spring Bean 的定義規則將檔案物件決議為 BeanDefinition 物件, -
接下來,將決議得到的 BeanDefinition 物件存入本地快取(一個 HashMap 集合,key 是字串,值是 BeanDefinition)中,
-
最后,實體化所有的 Bean 實體(非懶加載):包括實體的創建,實體的屬性填充,實體的初始化,
7、依賴注入的程序(Bean 的加載流程)?
原始碼分析可以參考我的文章:Spring原始碼分析——Bean的加載
先來看下面幾行代碼:
public class BeanFactoryTest {
public static void main(String[] args) {
// 加載與決議XML組態檔,獲得BeanFactory:
BeanFactory beanFactory = new XmlBeanFactory(new ClassPathResource("spring-bf.xml"));
// 從BeanFactory中加載Bean物件
Object a = beanFactory.getBean("componentA");
Object b = beanFactory.getBean("componentB");
System.out.println(a);// com.myspring.test.xmltest.ComponentA@1c93084c
System.out.println(b);// com.myspring.test.xmltest.ComponentB@6ef888f6
}
}
- 首先通過 BeanFactory/ApplicationContext 呼叫
getBean()方法,來獲取 Bean 實體,該方法中,真正獲取 Bean 實體的是其內層方法doGetBean()方法(真正實作從 IOC 容器獲取 Bean ,也是觸發依賴注入的地方), - 在
doGetBean()方法中,主要做了以下幾件事:- ① beanName 的轉換方法
transformedBeanName(name),該方法的作用是,根據傳入的 name 引數,獲取真正的 Bean 對應的 beanName,該方法的 name 引數,有可能是一個別名(alias 屬性設定的別名),也有可能是一個&開頭的 name (工廠 Bean 物件), - ② 嘗試從快取中加載 Bean 的單實體,根據上面
transformedBeanName方法轉換 name 后得到的真實 beanName,getSingleton(beanName)方法直接嘗試從快取中獲取 Bean 的共享單實體,這時候獲取的是初始狀態,尚未實體化,(從快取中加載的流程就是,根據 beanName 依次從一級快取、二級快取、三級快取中嘗試獲取,通過三級快取機制也可以有效避免回圈依賴) - ③ Bean 的實體化,
getSingleton(beanName)方法執行后,從快取中得到了 Bean 的原始狀態,接下來需要對該 Bean 進行實體化, - ④ Bean 的初始化:尋找依賴(回圈依賴檢查、依賴注入),因為 Bean 的初始化程序中很可能會用到某些屬性,而某些屬性很可能是動態配置的,并且配置的成依賴于其他的 Bean,那么此時應該先加載依賴的 Bean,所以在流程中,Spring初始化一個 Bean,會先初始化其依賴的所有的其他 Bean,
- ⑤ 根據不同的 scope 作用域創建 Bean,呼叫
doCreateBean()方法創建 Bean, - ⑥ 型別轉換,根據 scope 創建完 Bean 成功后,一般可以直接回傳即可,但當傳入
doGetBean方法中的requireType引數不為空時,意味著我們對最后回傳的 Bean 有著型別上的要求,Spring 通過 型別轉換器 將第 ⑤ 步創建完成的 Bean 轉換為requireType指定的型別,
- ① beanName 的轉換方法
8、Bean 的作用范圍?
通過 scope 屬性指定 Bean 的作用范圍,包括:
- ① singleton:單例模式,是默認作用域,不管收到多少 Bean 請求每個容器中只有一個唯一的 Bean 實體,
- ② prototype:原型模式,和 singleton 相反,每次 Bean 請求都會創建一個新的實體,
- ③ request:每次 HTTP 請求都會創建一個新的 Bean 并把它放到 request 域中,在請求完成后 Bean 會失效并被垃圾收集器回收,
- ④ session:和 request 類似,確保每個 session 中有一個 Bean 實體,session 過期后 bean 會隨之失效,
- ⑤ global session:當應用部署在 Portlet 容器時,如果想讓所有 Portlet 共用全域存盤變數,那么該變數需要存盤在 global session 中,
9、Spring事務傳播機制有哪些?
- ① REQUIRED:Spring 默認的事務傳播級別,如果背景關系中已經存在事務,那么就加入到事務中執行,如果當前背景關系中不存在事務,則新建事務執行,
- ② REQUIRES_NEW:每次都會新建一個事務,如果背景關系中有事務,則將背景關系的事務掛起,當新建事務執行完成以后,背景關系事務再恢復執行,
- ③ SUPPORTS:如果背景關系存在事務,則加入到事務執行,如果沒有事務,則使用非事務的方式執行,
- ④ MANDATORY:背景關系中必須要存在事務,否則就會拋出例外,
- ⑤ NOT_SUPPORTED :如果背景關系中存在事務,則掛起事務,執行當前邏輯,結束后恢復背景關系的事務,
- ⑥ NEVER:背景關系中不能存在事務,否則就會拋出例外,
- ⑦ ESTED:嵌套事務,如果背景關系中存在事務,則嵌套事務執行,如果不存在事務,則新建事務,
10、Spring 的事務隔離級別有哪些?
Spring 的事務隔離級別底層其實是基于資料庫的,Spring 并沒有自己的一套隔離級別,
- DEFAULT:使用資料庫的默認隔離級別,
- READ_UNCOMMITTED:讀未提交,最低的隔離級別,會讀取到其他事務還未提交的內容,存在臟讀,
- READ_COMMITTED:讀已提交,讀取到的內容都是已經提交的,可以解決臟讀,但是存在不可重復讀,
- REPEATABLE_READ:可重復讀,在一個事務中多次讀取時看到相同的內容,可以解決不可重復讀,但是存在幻讀,
- SERIALIZABLE:串行化,最高的隔離級別,對于同一行記錄,寫會加寫鎖,讀會加讀鎖,在這種情況下,只有讀讀能并發執行,其他并行的讀寫、寫讀、寫寫操作都是沖突的,需要串行執行,可以防止臟讀、不可重復度、幻讀,沒有并發事務問題,
11、AOP 是什么?AOP有哪些應用場景?
AOP 概念: 即面向切面編程,使用動態代理技術,在不修改原始碼的基礎上對目標方法進行增強,
Spring 中的 AOP 目前支持 JDK 動態代理和 Cglib 代理,如果被代理物件實作了介面,則使用 JDK 動態代理,否則使用 Cglib 代理,另外,也可以通過指定 proxyTargetClass=true 來實作強制走 Cglib 代理,
應用場景:
- 權限認證
- 日志列印
- 事務
- …
12、AOP 的相關注解有哪些?
-
@Aspect:切面,宣告被注解標注的類是一個切面 Bean,@Aspect @Component public class LogAspect { ... } -
@Pointcut:切入點,可以通過@Pointcut("execution(* top.csp1999.service.impl.*.*(..))")去指定要切入的目標物件,并對其符合運算式要求的方法進行增強,@Pointcut("execution(* top.csp1999.service.impl.*.*(..))") public void operationLog(){} -
@Before:前置通知,指在某個連接點之前執行的通知,@Before("operationLog()") public void doBeforeAdvice(JoinPoint joinPoint){ System.out.println("進入方法前執行....."); } -
@After:后置通知,指某個連接點退出時執行的通知(不論正常回傳還是例外退出),@After("operationLog()") public void after(JoinPoint jp){ System.out.println("方法最后執行....."); } -
@AfterReturning:后置回傳通知,指某連接點正常完成之后執行的通知,回傳值可以在回傳后通知方法里接收,@AfterReturning(returning = "ret", pointcut = "operationLog()") public void doAfterReturning(Object ret) { System.out.println("方法的回傳值 : " + ret); } -
@AfterThrowing:后置例外通知,指方法拋出例外導致退出時執行的通知,和@AfterReturning只會有一個執行,例外使用 throwing 屬性接收,@AfterThrowing(throwing = "jp", pointcut = "operationLog()") public void throwss(JoinPoint jp){ System.out.println("方法例外時執行....."); } -
@Around:環繞通知,可以用來在呼叫一個具體方法前和呼叫后來完成一些具體的任務,@Around("operationLog()") public Object run2(ProceedingJoinPoint joinPoint) throws Throwable { // 獲取方法引數值陣列 Object[] args = joinPoint.getArgs(); // 得到其方法簽名 MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature(); // 獲取方法引數型別陣列 Class[] paramTypeArray = methodSignature.getParameterTypes(); if (EntityManager.class.isAssignableFrom(paramTypeArray[paramTypeArray.length - 1])) { // 如果方法的引數串列最后一個引數是entityManager型別,則給其賦值 args[args.length - 1] = entityManager; } logger.info("請求引數為{}",args); // 動態修改其引數 // 注意,如果呼叫joinPoint.proceed()方法,則修改的引數值不會生效,必須呼叫joinPoint.proceed(Object[] args) Object result = joinPoint.proceed(args); logger.info("回應結果為{}",result); // 如果這里不回傳result,則目標物件實際回傳值會被置為null return result; }
13、AOP 的相關術語有什么?
Aspect:切面,一個關注點的模塊化,這個關注點可能會橫切多個物件,
Joinpoint:連接點,程式執行程序中的某一行為,即業務層中的所有方法,,
Advice:通知,指切面對于某個連接點所產生的動作,包括前置通知、后置通知、回傳后通知、例外通知和環繞通知,
Pointcut:切入點,指被攔截的連接點,切入點一定是連接點,但連接點不一定是切入點,
Proxy:代理,Spring AOP 中有 JDK 動態代理和 CGLib 代理,目標物件實作了介面時采用 JDK 動態代理,反之采用 CGLib 代理,
Target:代理的目標物件,指一個或多個切面所通知的物件,
Weaving :織入,指把增強應用到目標物件來創建代理物件的程序,
總結的面試題也挺費時間的,文章會不定時更新,有時候一天多更新幾篇,如果幫助您復習鞏固了知識點,還請三連支持一下,后續會億點點的更新!

為了幫助更多小白從零進階 Java 工程師,從CSDN官方那邊搞來了一套 《Java 工程師學習成長知識圖譜》,尺寸 870mm x 560mm,展開后有一張辦公桌大小,也可以折疊成一本書的尺寸,有興趣的小伙伴可以了解一下,當然,不管怎樣博主的文章一直都是免費的~

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/287148.html
標籤:java
