對于大部分人而言,在找工作之前少不了準備工作,有一份全面細致面試題將幫助大家減少許多麻煩。所以準備找工作的小伙伴可以看參考一下哦!
ThreadLocal(線程變量副本)
Synchronized實現內存共享,ThreadLocal為每個線程維護一個本地變量。采用空間換時間,它用于線程間的數據隔離,為每一個使用該變量的線程提供一個副本,每個線程都可以獨立地改變自己的副本,而不會和其他線程的副本沖突。ThreadLocal類中維護一個Map,用于存儲每一個線程的變量副本,Map中元素的鍵為線程對象,而值為對應線程的變量副本。ThreadLocal在Spring中發揮著巨大的作用,在管理Request作用域中的Bean、事務管理、任務調度、AOP等模塊都出現了它的身影。Spring中絕大部分Bean都可以聲明成Singleton作用域,采用ThreadLocal進行封裝,因此有狀態的Bean就能夠以singleton的方式在多線程中正常工作了。
“你能不能談談,java GC是在什么時候,對什么東西,做了什么事情?”
在什么時候:1、新生代有一個Eden區和兩個survivor區,首先將對象放入Eden區,如果空間不足就向其中的一個survivor區上放,如果仍然放不下就會引發一次發生在新生代的minor GC,將存活的對象放入另一個survivor區中,然后清空Eden和之前的那個survivor區的內存。在某次GC過程中,如果發現仍然又放不下的對象,就將這些對象放入老年代內存里去。2、大對象以及長期存活的對象直接進入老年區。3、當每次執行minor GC的時候應該對要晉升到老年代的對象進行分析,如果這些馬上要到老年區的老年對象的大小超過了老年區的剩余大小,那么執行一次Full GC以盡可能地獲得老年區的空間。
對什么東西:
從GC Roots搜索不到,而且經過一次標記清理之后仍沒有復活的對象。
做什么:
新生代:復制清理; 老年代:標記-清除和標記-壓縮算法; 永久代:存放Java中的類和加載類的類加載器本身。GC Roots都有哪些:1. 虛擬機棧中的引用的對象 2. 方法區中靜態屬性引用的對象,常量引用的對象 3. 本地方法棧中JNI(即一般說的Native方法)引用的對象。
Synchronized 與Loc
Synchronized 與Lock都是可重入鎖,同一個線程再次進入同步代碼的時候??梢允褂米约阂呀洬@取到的鎖。
Synchronized是悲觀鎖機制,獨占鎖。而Locks.ReentrantLock是,每次不加鎖而是假設沒有沖突而去完成某項操作,如果因為沖突失敗就重試,直到成功為止。
ReentrantLock適用場景
1、某個線程在等待一個鎖的控制權的這段時間需要中斷;
2、需要分開處理一些wait-notify,ReentrantLock里面的Condition應用,能夠控制notify哪個線程,鎖可以綁定多個條件;
3、具有公平鎖功能,每個到來的線程都將排隊等候。
StringBuffer是線程安全的,每次操作字符串,String會生成一個新的對象,而StringBuffer不會;StringBuilder是非線程安全的。
fail-fast是什么?
fail-fast:機制是java集合(Collection)中的一種錯誤機制。當多個線程對同一個集合的內容進行操作時,就可能會產生fail-fast事件。例如:當某一個線程A通過iterator去遍歷某集合的過程中,若該集合的內容被其他線程所改變了;那么線程A訪問集合時,就會拋出ConcurrentModificationException異常,產生fail-fast事件。
happens-before
happens-before:如果兩個操作之間具有 happens-before 關系,那么前一個操作的結果就會對后面一個操作可見。1、程序順序規則:一個線程中的每個操作,happens- before 于該線程中的任意后續操作。2、監視器鎖規則:對一個監視器鎖的解鎖,happens- before 于隨后對這個監視器鎖的加鎖。3、volatile變量規則:對一個volatile域的寫,happens- before于任意后續對這個volatile域的讀。4、傳遞性:如果A happens- before B,且B happens- before C,那么A happens- before C。5、線程啟動規則:Thread對象的start()方法happens- before于此線程的每一個動作。
Volatile和Synchronized的不同點
Volatile和Synchronized四個不同點:1、粒度不同,前者針對變量 ,后者鎖對象和類;2、syn阻塞,volatile線程不阻塞;3、syn保證三大特性,volatile不保證原子性;
4、syn編譯器優化,volatile不優化 volatile具備兩種特性:
保證此變量對所有線程的可見性,指一條線程修改了這個變量的值,新值對于其他線程來說是可見的,但并不是多線程安全的;禁止指令重排序優化。Volatile如何保證內存可見性:
1、當寫一個volatile變量時,JMM會把該線程對應的本地內存中的共享變量刷新到主內存。
2、當讀一個volatile變量時,JMM會把該線程對應的本地內存置為無效。線程接下來將從主內存中讀取共享變量。
同步:就是一個任務的完成需要依賴另外一個任務,只有等待被依賴的任務完成后,依賴任務才能完成。異步:不需要等待被依賴的任務完成,只是通知被依賴的任務要完成什么工作,只要自己任務完成了就算完成了,被依賴的任務是否完成會通知回來。(異步的特點就是通知)。 打電話和發短信來比喻同步和異步操作。阻塞:CPU停下來等一個慢的操作完成以后,才會接著完成其他的工作。非阻塞:非阻塞就是在這個慢的執行時,CPU去做其他工作,等這個慢的完成后,CPU才會接著完成后續的操作。非阻塞會造成線程切換增加,增加CPU的使用時間能不能補償系統的切換成本需要考慮。
CAS
CAS(Compare And Swap) 無鎖算法: CAS是樂觀鎖技術,當多個線程嘗試使用CAS同時更新同一個變量時,只有其中一個線程能更新變量的值,而其它線程都失敗,失敗的線程并不會被掛起,而是被告知這次競爭中失敗,并可以再次嘗試。
CAS有3個操作數,內存值V,舊的預期值A,要修改的新值B。當且僅當預期值A和內存值V相同時,將內存值V修改為B,否則什么都不做。
線程池的作用
線程池的作用:在程序啟動的時候就創建若干線程來響應處理,它們被稱為線程池,里面的線程叫工作線程。第一:降低資源消耗。通過重復利用已創建的線程降低線程創建和銷毀造成的消耗。第二:提高響應速度。當任務到達時,任務可以不需要等到線程創建就能立即執行。第三:提高線程的可管理性。常用線程池:ExecutorService 是主要的實現類,其中常用的有 Executors.newSingleThreadPool()、newFixedThreadPool()、newcachedTheadPool()、newScheduledThreadPool()。
類加載器工作機制
裝載:將Java二進制代碼導入jvm中,生成Class文件。連接:
a)校驗:檢查載入Class文件數據的正確性;
b)準備:給類的靜態變量分配存儲空間;
c)解析:將符號引用轉成直接引用。初始化:對類的靜態變量,靜態方法和靜態代碼塊執行初始化工作。雙親委派模型:類加載器收到類加載請求,首先將請求委派給父類加載器完成,用戶自定義加載器-應用程序加載器-擴展類加載器-啟動類加載器。
Redis數據結構
String—字符串(key-value 類型)
Hash—字典(hashmap) Redis的哈希結構可以使你像在數據庫中更新一個屬性一樣只修改某一項屬性值List—列表 實現消息隊列Set—集合 利用唯一性Sorted Set—有序集合 可以進行排序 可以實現數據持久化
索引:B+,B-,全文索引
Mysql的索引是一個數據結構,旨在使數據庫高效的查找數據。常用的數據結構是B+Tree,每個葉子節點不但存放了索引鍵的相關信息還增加了指向相鄰葉子節點的指針,這樣就形成了帶有順序訪問指針的B+Tree,做這個優化的目的是提高不同區間訪問的性能。什么時候使用索引:
1、經常出現在group by,order by和distinc關鍵字后面的字段。
2、經常與其他表進行連接的表,在連接字段上應該建立索引。
3、經常出現在Where子句中的字段。
4、經常出現用作查詢選擇的字段。
Spring IOC
Spring支持三種依賴注入方式,分別是屬性(Setter方法)注入,構造注入和接口注入。在Spring中,那些組成應用的主體及由Spring IOC容器所管理的對象被稱之為Bean。Spring的IOC容器通過反射的機制實例化Bean并建立Bean之間的依賴關系。簡單地講,Bean就是由Spring IOC容器初始化、裝配及被管理的對象。獲取Bean對象的過程,首先通過Resource加載配置文件并啟動IOC容器,然后通過getBean方法獲取bean對象,就可以調用他的方法。Spring Bean的作用域:Singleton:Spring IOC容器中只有一個共享的Bean實例,一般都是Singleton作用域。Prototype:每一個請求,會產生一個新的Bean實例。Request:每一次http請求會產生一個新的Bean實例。
代理都有哪些?
代理的共有優點:業務類只需要關注業務邏輯本身,保證了業務類的重用性。Java靜態代理:代理對象和目標對象實現了相同的接口,目標對象作為代理對象的一個屬性,具體接口實現中,代理對象可以在調用目標對象相應方法前后加上其他業務處理邏輯。缺點:一個代理類只能代理一個業務類。如果業務類增加方法時,相應的代理類也要增加方法。Java動態代理:Java動態代理是寫一個類實現InvocationHandler接口,重寫Invoke方法,在Invoke方法可以進行增強處理的邏輯的編寫,這個公共代理類在運行的時候才能明確自己要代理的對象,同時可以實現該被代理類的方法的實現,然后在實現類方法的時候可以進行增強處理。實際上:代理對象的方法 = 增強處理 + 被代理對象的方法
JDK和CGLIB生成動態代理類的區別:JDK動態代理只能針對實現了接口的類生成代理(實例化一個類)。此時代理對象和目標對象實現了相同的接口,目標對象作為代理對象的一個屬性,具體接口實現中,可以在調用目標對象相應方法前后加上其他業務處理邏輯。CGLIB是針對類實現代理,主要是對指定的類生成一個子類(沒有實例化一個類),覆蓋其中的方法 。
SpringMVC運行原理
1、客戶端請求提交到DispatcherServlet。
2、由DispatcherServlet控制器查詢HandlerMapping,找到并分發到指定的Controller中。
3、Controller調用業務邏輯處理后,返回ModelAndView。
4、DispatcherServlet查詢一個或多個ViewResoler視圖解析器,找到ModelAndView指定的視圖。
5、視圖負責將結果顯示到客戶端。
一個Http請求
DNS域名解析 – 發起TCP的三次握手 – 建立TCP連接后發起http請求 – 服務器響應http請求,瀏覽器得到html代碼 – 瀏覽器解析html代碼,并請求html代碼中的資源(如javascript、css、圖片等) – 瀏覽器對頁面進行渲染呈現給用戶。
設計存儲海量數據的存儲系統:設計一個叫“中間層”的一個邏輯層,在這個層,將數據庫的海量數據抓出來,做成緩存,運行在服務器的內存中,同理,當有新的數據到來,也先做成緩存,再想辦法,持久化到數據庫中,這是一個簡單的思路。
主要的步驟是負載均衡,將不同用戶的請求分發到不同的處理節點上,然后先存入緩存,定時向主數據庫更新數據。讀寫的過程采用類似樂觀鎖的機制,可以一直讀(在寫數據的時候也可以),但是每次讀的時候會有個版本的標記,如果本次讀的版本低于緩存的版本,會重新讀數據,這樣的情況并不多,可以忍受。
Session與Cookie
Session與Cookie:Cookie可以讓服務端跟蹤每個客戶端的訪問,但是每次客戶端的訪問都必須傳回這些Cookie,如果Cookie很多,則無形的增加了客戶端與服務端的數據傳輸量,而Session則很好地解決了這個問題,同一個客戶端每次和服務端交互時,將數據存儲通過Session到服務端,不需要每次都傳回所有的Cookie值,而是傳回一個ID,每個客戶端第一次訪問服務器生成的唯一的ID,客戶端只要傳回這個ID就行了,這個ID通常為NAME為JSESSIONID的一個Cookie。這樣服務端就可以通過這個ID,來將存儲到服務端的KV值取出了。
分布式Session框架
1、配置服務器,Zookeeper集群管理服務器可以統一管理所有服務器的配置文件;
2、共享這些Session存儲在一個分布式緩存中,可以隨時寫入和讀取,而且性能要很好,如Memcache,Tair;
3、封裝一個類繼承自HttpSession,將Session存入到這個類中然后再存入分布式緩存中;
4、由于Cookie不能跨域訪問,要實現Session同步,要同步SessionID寫到不同域名下。
InputStream
適配器模式:將一個接口適配到另一個接口,Java I/O中InputStreamReader將Reader類適配到InputStream,從而實現了字節流到字符流的準換。裝飾者模式:保持原來的接口,增強原來有的功能。FileInputStream 實現了InputStream的所有接口,BufferedInputStreams繼承自FileInputStream是具體的裝飾器實現者,將InputStream讀取的內容保存在內存中,而提高讀取的性能。
Spring事務配置方法
1、切點信息,用于定位實施事物切面的業務類方法。2、控制事務行為的事務屬性,這些屬性包括事物隔離級別,事務傳播行為,超時時間,回滾規則。Spring通過aop/tx Schema 命名空間和@Transaction注解技術來進行聲明式事物配置。
Mybatis
每一個Mybatis的應用程序都以一個SqlSessionFactory對象的實例為核心。首先用字節流通過Resource將配置文件讀入。
通過SqlSessionFactoryBuilder().build方法創建SqlSessionFactory,然后再通過SqlSessionFactory.openSession()方法創建一個SqlSession為每一個數據庫事務服務。經歷了Mybatis初始化 –創建SqlSession –運行SQL語句,返回結果三個過程。
Servlet和Filter的區別
整的流程是:Filter對用戶請求進行預處理,接著將請求交給Servlet進行處理并生成響應,最后Filter再對服務器響應進行后處理。
Filter有如下幾個用處:Filter可以進行對特定的url請求和相應做預處理和后處理。在HttpServletRequest到達Servlet之前,攔截客戶的HttpServletRequest。根據需要檢查HttpServletRequest,也可以修改HttpServletRequest頭和數據。在HttpServletResponse到達客戶端之前,攔截HttpServletResponse。根據需要檢查HttpServletResponse,也可以修改HttpServletResponse頭和數據。
實際上Filter和Servlet極其相似,區別只是Filter不能直接對用戶生成響應。實際上Filter里doFilter()方法里的代碼就是從多個Servlet的service()方法里抽取的通用代碼,通過使用Filter可以實現更好的復用。
Filter和Servlet的生命周期:1、Filter在web服務器啟動時初始化。2、如果某個Servlet配置了 1 ,該Servlet也是在Tomcat(Servlet容器)啟動時初始化。3、如果Servlet沒有配置1 ,該Servlet不會在Tomcat啟動時初始化,而是在請求到來時初始化。4、每次請求, Request都會被初始化,響應請求后,請求被銷毀。5、Servlet初始化后,將不會隨著請求的結束而注銷。6、關閉Tomcat時,Servlet、Filter依次被注銷。
HashMap與HashTable的區別
1、HashMap是非線程安全的,HashTable是線程安全的。2、HashMap的鍵和值都允許有null值存在,而HashTable則不行。3、因為線程安全的問題,HashMap效率比HashTable的要高。
HashMap的實現機制:
1、維護一個每個元素是一個鏈表的數組,而且鏈表中的每個節點是一個Entry[]鍵值對的數據結構。
2、實現了數組+鏈表的特性,查找快,插入刪除也快。
3、對于每個key,他對應的數組索引下標是 int i = hash(key.hashcode)(len-1);
4、每個新加入的節點放在鏈表首,然后該新加入的節點指向原鏈表首。
Linux常用命令
cd,cp,mv,rm,ps(進程),tar,cat(查看內容),chmod,vim,find,ls
死鎖的必要條件
1、互斥 至少有一個資源處于非共享狀態;
2、占有并等待;
3、非搶占;
4、循環等待。
解決死鎖,第一個是死鎖預防,就是不讓上面的四個條件同時成立。二是,合理分配資源。三是使用銀行家算法,如果該進程請求的資源操作系統剩余量可以滿足,那么就分配。
進程間的通信方式
管道( pipe ):管道是一種半雙工的通信方式,數據只能單向流動,而且只能在具有親緣關系的進程間使用。進程的親緣關系通常是指父子進程關系。
有名管道 (named pipe) :有名管道也是半雙工的通信方式,但是它允許無親緣關系進程間的通信。
信號量( semophore ) :信號量是一個計數器,可以用來控制多個進程對共享資源的訪問。它常作為一種鎖機制,防止某進程正在訪問共享資源時,其他進程也訪問該資源。因此,主要作為進程間以及同一進程內不同線程之間的同步手段。
消息隊列( message queue ) :消息隊列是由消息的鏈表,存放在內核中并由消息隊列標識符標識。消息隊列克服了信號傳遞信息少、管道只能承載無格式字節流以及緩沖區大小受限等缺點。
信號 ( sinal ) :信號是一種比較復雜的通信方式,用于通知接收進程某個事件已經發生。
共享內存( shared memory ) :共享內存就是映射一段能被其他進程所訪問的內存,這段共享內存由一個進程創建,但多個進程都可以訪問。共享內存是最快的 IPC 方式,它是針對其他進程間通信方式運行效率低而專門設計的。它往往與其他通信機制,如信號量,配合使用,來實現進程間的同步和通信。
套接字( socket ) :套解口也是一種進程間通信機制,與其他通信機制不同的是,它可用于不同機器間的進程通信。
Switch能否用string做參數?
在 Java 7 之前, switch 只能支持byte、short、char、int或者其對應的封裝類以及 Enum 類型。在Java 7中、String 支持被加上了。
Object有哪些公用方法?
a、方法equals測試的是兩個對象是否相等;
b、方法clone進行對象拷貝;
c、方法getClass返回和當前對象相關的Class對象;
d、方法notify,notifyall,wait都是用來對給定對象進行線程同步的。
Java的四種引用,強弱軟虛,以及用到的場景
利用軟引用和弱引用解決OOM問題:用一個HashMap來保存圖片的路徑和相應圖片對象關聯的軟引用之間的映射關系,在內存不足時,JVM會自動回收這些緩存圖片對象所占用的空間,從而有效地避免了OOM的問題。
通過軟可及對象重獲方法實現Java對象的高速緩存:比如我們創建了一Employee的類,如果每次需要查詢一個雇員的信息。哪怕是幾秒中之前剛剛查詢過的,都要重新構建一個實例,這是需要消耗很多時間的。我們可以通過軟引用和 HashMap 的結合,先是保存引用方面:以軟引用的方式對一個Employee對象的實例進行引用并保存該引用到HashMap 上,key 為此雇員的 id,value為這個對象的軟引用,另一方面是取出引用,緩存中是否有該Employee實例的軟引用,如果有,從軟引用中取得。如果沒有軟引用,或者從軟引用中得到的實例是null,重新構建一個實例,并保存對這個新建實例的軟引用。
強引用:如果一個對象具有強引用,它就不會被垃圾回收器回收。即使當前內存空間不足,JVM也不會回收它,而是拋出 OutOfMemoryError 錯誤,使程序異常終止。如果想中斷強引用和某個對象之間的關聯,可以顯式地將引用賦值為null,這樣一來的話,JVM在合適的時間就會回收該對象。
軟引用:在使用軟引用時,如果內存的空間足夠,軟引用就能繼續被使用,而不會被垃圾回收器回收,只有在內存不足時,軟引用才會被垃圾回收器回收。
弱引用:具有弱引用的對象擁有的生命周期更短暫。因為當 JVM 進行垃圾回收,一旦發現弱引用對象,無論當前內存空間是否充足,都會將弱引用回收。不過由于垃圾回收器是一個優先級較低的線程,所以并不一定能迅速發現弱引用對象。
虛引用:顧名思義,就是形同虛設,如果一個對象僅持有虛引用,那么它相當于沒有引用,在任何時候都可能被垃圾回收器回收。
Hashcode的作用,與 equal 有什么區別?
同樣用于鑒定2個對象是否相等的,java集合中有 list 和 set 兩類,其中 set不允許元素重復實現,那個這個不允許重復實現的方法,如果用 equal 去比較的話,如果存在1000個元素,你 new 一個新的元素出來,需要去調用1000次 equal 去逐個和他們比較是否是同一個對象,這樣會大大降低效率。
hashcode實際上是返回對象的存儲地址,如果這個位置上沒有元素,就把元素直接存儲在上面,如果這個位置上已經存在元素,這個時候才去調用equal方法與新元素進行比較,相同的話就不存了,散列到其他地址上。
Override和Overload的含義以及區別
a、Overload顧名思義是重新加載,它可以表現類的多態性,可以是函數里面可以有相同的函數名但是參數名、返回值、類型不能相同;或者說可以改變參數、類型、返回值但是函數名字依然不變。b、就是ride(重寫)的意思,在子類繼承父類的時候子類中可以定義某方法與其父類有相同的名稱和參數,當子類在調用這一函數時自動調用子類的方法,而父類相當于被覆蓋(重寫)了。具體可前往C++中重載、重寫(覆蓋)的區別實例分析查看。
抽象類和接口的區別
a、一個類只能繼承單個類,但是可以實現多個接口;
b、抽象類中可以有構造方法,接口中不能有構造方法;
c、抽象類中的所有方法并不一定要是抽象的,你可以選擇在抽象類中實現一些基本的方法。而接口要求所有的方法都必須是抽象的;
d、抽象類中可以包含靜態方法,接口中不可以;
e、抽象類中可以有普通成員變量,接口中不可以。
解析XML的幾種方式的原理與特點
DOM:消耗內存:先把xml文檔都讀到內存中,然后再用DOM API來訪問樹形結構,并獲取數據。這個寫起來很簡單,但是很消耗內存。要是數據過大,手機不夠牛逼,可能手機直接死機。
SAX:解析效率高,占用內存少,基于事件驅動的:更加簡單地說就是對文檔進行順序掃描,當掃描到文檔(document)開始與結束、元素(element)開始與結束、文檔(document)結束等地方時通知事件處理函數,由事件處理函數做相應動作,然后繼續同樣的掃描,直至文檔結束。
PULL:與 SAX 類似,也是基于事件驅動,我們可以調用它的next()方法,來獲取下一個解析事件(就是開始文檔,結束文檔,開始標簽,結束標簽),當處于某個元素時可以調用XmlPullParser的getAttributte()方法來獲取屬性的值,也可調用它的nextText()獲取本節點的值。
wait()和sleep()的區別
sleep來自Thread類,和wait來自Object類。
調用sleep()方法的過程中,線程不會釋放對象鎖。而 調用 wait 方法線程會釋放對象鎖。
sleep睡眠后不出讓系統資源,wait讓出系統資源其他線程可以占用CPU。
sleep(milliseconds)需要指定一個睡眠時間,時間一到會自動喚醒。
JAVA 中堆和棧的區別,說下java 的內存機制
基本數據類型比變量和對象的引用都是在棧分配的;
堆內存用來存放由new創建的對象和數組;
類變量(static修飾的變量):程序在一加載的時候就在堆中為類變量分配內存,堆中的內存地址存放在棧中;
實例變量:當你使用java關鍵字new的時候,系統在堆中開辟并不一定是連續的空間分配給變量,是根據零散的堆內存地址,通過哈希算法換算為一長串數字以表征這個變量在堆中的”物理位置”,實例變量的生命周期–當實例變量的引用丟失后,將被GC(垃圾回收器)列入可回收“名單”中,但并不是馬上就釋放堆中內存;
局部變量:由聲明在某方法,或某代碼段里(比如for循環),執行到它的時候在棧中開辟內存,當局部變量一但脫離作用域,內存立即釋放。
JAVA多態的實現原理
a、抽象的來講,多態的意思就是同一消息可以根據發送對象的不同而采用多種不同的行為方式。(發送消息就是函數調用)
b、實現的原理是動態綁定,程序調用的方法在運行期才動態綁定,追溯源碼可以發現,JVM 通過參數的自動轉型來找到合適的辦法。