1.什么是Java虛擬機(jī)?為什么Java被稱作是“平臺無關(guān)的編程語言”?
Java虛擬機(jī)是一個可以執(zhí)行Java字節(jié)碼的虛擬機(jī)進(jìn)程。Java源文件被編譯成能被Java虛擬機(jī)執(zhí)行的字節(jié)碼文件。
Java被設(shè)計成允許應(yīng)用程序可以運行在任意的平臺,而不需要程序員為每一個平臺單獨重寫或者是重新編譯。Java虛擬機(jī)讓這個變?yōu)榭赡埽驗樗赖讓佑布脚_的指令長度和其他特性。
2.JDK和JRE的區(qū)別是什么?
Java運行時環(huán)境(JRE)是將要執(zhí)行Java程序的Java虛擬機(jī)。它同時也包含了執(zhí)行applet需要的瀏覽器插件。
Java開發(fā)工具包(JDK)是完整的Java軟件開發(fā)包,包含了JRE,編譯器和其他的工具(比如:JavaDoc,Java調(diào)試器),可以讓開發(fā)者開發(fā)、編譯、執(zhí)行Java應(yīng)用程序。
3.”static”關(guān)鍵字是什么意思?Java中是否可以覆蓋(override)一個private或者是static的方法?
“static”關(guān)鍵字表明一個成員變量或者是成員方法可以在沒有所屬的類的實例變量的情況下被訪問。
Java中static方法不能被覆蓋,因為方法覆蓋是基于運行時動態(tài)綁定的,而static方法是編譯時靜態(tài)綁定的。static方法跟類的任何實例都不相關(guān),所以概念上不適用。private修飾不支持繼承,private修飾的方法同樣不能被覆蓋。
4.是否可以在static環(huán)境中訪問非static變量?
static變量在Java中是屬于類的,它在所有的實例中的值是一樣的。當(dāng)類被Java虛擬機(jī)載入的時候,會對static變量進(jìn)行初始化。
因為靜態(tài)的成員屬于類,隨著類的加載而加載到靜態(tài)方法區(qū)內(nèi)存,當(dāng)類加載時,此時不一定有實例創(chuàng)建,沒有實例,就不可以訪問非靜態(tài)的成員。類的加載先于實例的創(chuàng)建,因此靜態(tài)環(huán)境中,不可以訪問非靜態(tài)!
5.Java支持的數(shù)據(jù)類型有哪些?什么是自動拆裝箱?
Java語言支持的8種基本數(shù)據(jù)類型是:
byte
short
int
long
float
double
boolean
char
自動裝箱是Java編譯器在基本數(shù)據(jù)類型和對應(yīng)的對象包裝類型之間做的一個轉(zhuǎn)化。比如:把int轉(zhuǎn)化成Integer,double轉(zhuǎn)化成Double,等等。反之就是自動拆箱。
6.Java中的方法覆蓋(Overriding)和方法重載(Overloading)是什么意思?
Java中的方法重載發(fā)生在同一個類里面兩個或者是多個方法的方法名相同但是參數(shù)不同的情況。
與此相對,方法覆蓋是說子類重新定義了父類的方法。方法覆蓋必須有相同的方法名,參數(shù)列表和返回類型。覆蓋者可能不會限制它所覆蓋的方法的訪問。
7.Java中,什么是構(gòu)造函數(shù)?什么是構(gòu)造函數(shù)重載?什么是復(fù)制構(gòu)造函數(shù)?
當(dāng)新對象被創(chuàng)建的時候,構(gòu)造函數(shù)會被調(diào)用。每一個類都有構(gòu)造函數(shù)。在程序員沒有給類提供構(gòu)造函數(shù)的情況下,Java編譯器會為這個類創(chuàng)建一個默認(rèn)的構(gòu)造函數(shù)。
Java中構(gòu)造函數(shù)重載和方法重載很相似。可以為一個類創(chuàng)建多個構(gòu)造函數(shù)。每一個構(gòu)造函數(shù)必須有它自己唯一的參數(shù)列表。
Java不支持像C++中那樣的復(fù)制構(gòu)造函數(shù),這個不同點是因為如果你不自己寫構(gòu)造函數(shù)的情況下,Java不會創(chuàng)建默認(rèn)的復(fù)制構(gòu)造函數(shù)。( java中不支持構(gòu)造函數(shù)的復(fù)制。構(gòu)造函數(shù)的復(fù)制屬于C++的內(nèi)容。)
8.Java支持多繼承么?
Java中類不支持多繼承,只支持單繼承(即一個類只有一個父類)。但是java中的接口支持多繼承,即一個子接口可以有多個父接口。(接口的作用是用來擴(kuò)展對象的功能,一個子接口繼承多個父接口,說明子接口擴(kuò)展了多個功能,當(dāng)類實現(xiàn)接口時,類就擴(kuò)展了相應(yīng)的功能)。
9.接口和抽象類的區(qū)別是什么?
Java提供和支持創(chuàng)建抽象類和接口。它們的實現(xiàn)有共同點,不同點在于:
接口中所有的方法隱含的都是抽象的。而抽象類則可以同時包含抽象和非抽象的方法;
接口中的成員方法默認(rèn)是public的。抽象類的成員函數(shù)可以是private,protected或者是public;
接口中聲明的變量默認(rèn)都是final的。抽象類可以包含非final的變量;
類可以實現(xiàn)很多個接口,但是只能繼承一個抽象類;
類可以不實現(xiàn)抽象類和接口聲明的所有方法,當(dāng)然,在這種情況下,類也必須得聲明成是抽象的;
抽象類可以在不提供接口方法實現(xiàn)的情況下實現(xiàn)接口;
接口是絕對抽象的,不可以被實例化。抽象類也不可以被實例化,但是,如果它包含main方法的話是可以被調(diào)用的;
10.什么是值傳遞和引用傳遞?
值傳遞是對基本型變量而言的,傳遞的是該變量的一個副本,改變副本不影響原變量.
引用傳遞一般是對于對象型變量而言的,傳遞的是該對象地址的一個副本, 并不是原對象本身,所以對引用對象進(jìn)行操作會同時改變原對象。
一般認(rèn)為,java內(nèi)的傳遞都是值傳遞。
11.進(jìn)程和線程的區(qū)別是什么?
進(jìn)程是執(zhí)行著的應(yīng)用程序,而線程是進(jìn)程內(nèi)部的一個執(zhí)行序列。一個進(jìn)程可以有多個線程。線程又叫做輕量級進(jìn)程。
12.創(chuàng)建線程有幾種不同的方式?你喜歡哪一種?為什么?
有三種方式可以用來創(chuàng)建線程:
繼承Thread類
實現(xiàn)Runnable接口
應(yīng)用程序可以使用Executor框架來創(chuàng)建線程池實現(xiàn)Runnable接口這種方式更受歡迎,因為這不需要繼承Thread類。在應(yīng)用設(shè)計中已經(jīng)繼承了別的對象的情況下,這需要多繼承(而Java不支持多繼承),只能實現(xiàn)接口。同時,線程池也是非常高效的,很容易實現(xiàn)和使用。
13.概括的解釋下線程的幾種可用狀態(tài)。
新建( new ):新創(chuàng)建了一個線程對象;
可運行( runnable ):線程對象創(chuàng)建后,其他線程(比如 main 線程)調(diào)用了該對象的 start ()方法。該狀態(tài)的線程位于可運行線程池中,等待被線程調(diào)度選中,獲 取CPU的使用權(quán);
運行( running ):可運行狀態(tài)( runnable )的線程獲得了CPU時間片( timeslice ) ,執(zhí)行程序代碼;
阻塞( block ):阻塞狀態(tài)是指線程因為某種原因放棄了CPU 使用權(quán),也即讓出了 CPU timeslice ,暫時停止運行。直到線程進(jìn)入可運行( runnable )狀態(tài),才有 機(jī)會再次獲得 cpu timeslice 轉(zhuǎn)到運行( running )狀態(tài)。
阻塞的情況分三種:
等待阻塞:運行( running )的線程執(zhí)行 o . wait ()方法, JVM 會把該線程放 入等待隊列( waitting queue )中。
同步阻塞:運行( running )的線程在獲取對象的同步鎖時,若該同步鎖被別的線程占用,則 JVM 會把該線程放入鎖池( lock pool )中。
其他阻塞: 運行( running )的線程執(zhí)行 Thread . sleep ( long ms )或 t . join ()方法,或者發(fā)出了 I / O 請求時, JVM 會把該線程置為阻塞狀態(tài)。當(dāng) sleep ()狀態(tài)超時、 join ()等待線程終止或者超時、或者 I / O 處理完畢時,線程重新轉(zhuǎn)入可運行( runnable )狀態(tài)。
死亡( dead ):線程 run ()、 main () 方法執(zhí)行結(jié)束,或者因異常退出了 run ()方法,則該線程結(jié)束生命周期。死亡的線程不可再次復(fù)生。
14.同步方法和同步代碼塊的區(qū)別是什么?
區(qū)別:
同步方法默認(rèn)用this或者當(dāng)前類class對象作為鎖;
同步代碼塊可以選擇以什么來加鎖,比同步方法要更細(xì)顆粒度,我們可以選擇只同步會發(fā)生同步問題的部分代碼而不是整個方法;
15.在監(jiān)視器(Monitor)內(nèi)部,是如何做線程同步的?程序應(yīng)該做哪種級別的同步?
監(jiān)視器和鎖在Java虛擬機(jī)中是一塊使用的。監(jiān)視器監(jiān)視一塊同步代碼塊,確保一次只有一個線程執(zhí)行同步代碼塊。每一個監(jiān)視器都和一個對象引用相關(guān)聯(lián)。線程在獲取鎖之前不允許執(zhí)行同步代碼。
java 還提供了 顯式監(jiān)視器( Lock ) 和 隱式監(jiān)視器( synchronized ) 兩種鎖方案。
16.什么是死鎖(deadlock)?
兩個線程或兩個以上線程都在等待對方執(zhí)行完畢才能繼續(xù)往下執(zhí)行的時候就發(fā)生了死鎖。結(jié)果就是這些線程都陷入了無限的等待中。
17.如何確保N個線程可以訪問N個資源同時又不導(dǎo)致死鎖?
多線程產(chǎn)生死鎖的四個必要條件:
互斥條件:一個資源每次只能被一個進(jìn)程使用。
保持和請求條件:一個進(jìn)程因請求資源而阻塞時,對已獲得資源保持不放。
不可剝奪性:進(jìn)程已獲得資源,在未使用完成前,不能被剝奪。
循環(huán)等待條件(閉環(huán)):若干進(jìn)程之間形成一種頭尾相接的循環(huán)等待資源關(guān)系。
只要破壞其中任意一個條件,就可以避免死鎖
一種非常簡單的避免死鎖的方式就是:指定獲取鎖的順序,并強制線程按照指定的順序獲取鎖。因此,如果所有的線程都是以同樣的順序加鎖和釋放鎖,就不會出現(xiàn)死鎖了。
18.Java集合類框架的基本接口有哪些?
集合類接口指定了一組叫做元素的對象。集合類接口的每一種具體的實現(xiàn)類都可以選擇以它自己的方式對元素進(jìn)行保存和排序。有的集合類允許重復(fù)的鍵,有些不允許。
Java集合類提供了一套設(shè)計良好的支持對一組對象進(jìn)行操作的接口和類。Java集合類里面基本的接口有:
Collection:代表一組對象,每一個對象都是它的子元素。
Set:不包含重復(fù)元素的Collection。
List:有順序的collection,并且可以包含重復(fù)元素。
Map:可以把鍵(key)映射到值(value)的對象,鍵不能重復(fù)。