- 相關(guān)推薦
.net面試題及詳解
1.什么是CLR
公共語言運行時(Comman language Runtime),是一個可由多種編程語言使用的“運行時”。CLR的核心功能:程序集加載,異常處理,線程同步,內(nèi)存管理等可由CLR
的所有語言使用.
2.什么是IL
Intermediate language
中間語言,.net程序在經(jīng)過編譯后就成為IL代碼。運行時CLR將IL語言編譯成CPU能識別的CRU指令。IL也可以叫做托管代碼,IL可以訪問CLR所提供的所有功能。
3.什么是JIT,它是如何工作的?
即時編譯器,由CLR調(diào)用,負責將IL語言編譯成本地CPU指令。
工作原理:
當程序被第一次調(diào)用的時候,CLR會指向包含在CLR內(nèi)部定義的特殊函數(shù),這個函數(shù)就是JITCompiler。JITComliler負責將IL代碼編譯成本地指令。
JITCompiler知道實際調(diào)用了哪個方法,以及該方法是哪些類型定義的。JITCompiler會在定義該類型的程序集的元數(shù)據(jù)中查找該方法的IL代碼。
并將IL編譯成本地CPU指令。編譯好的結(jié)果被放在一個內(nèi)存塊中,JITCompiler返回CLR為類型定義的內(nèi)部數(shù)據(jù)結(jié)構(gòu)。找到被調(diào)用方法對應(yīng)的那條記錄,
并修改最初對于JITCompiler的引用。讓其指向內(nèi)存塊中被調(diào)用方法剛剛編譯好的CPU指令地址,最后執(zhí)行被調(diào)用方法的CPU指令。
4.GC是什么,簡述一下GC的工作方式?
垃圾回收(garbage collection)
Dot Net的垃圾回收可以分為兩個步驟,第一步進行“標記”,垃圾回收器假設(shè)所有的對象都是垃圾,然后開始遍歷每一個“根”(根包含指向引用類型對象的一個指針,值類型對象永遠不會被認為是一個根),如果發(fā)現(xiàn)一個根引用了一個對象(非NULL),就對對象進行標記。沒有被標記的對象被認為是垃圾。第二個階段就是“壓縮”,其實就是將后面的對象移動到已經(jīng)成為垃圾的對象位置,使得原來的托管堆更為緊湊。從而釋放了托管堆。
GC類中的方法影響何時對對象進行垃圾回收以及何時釋放對象所分配的資源。此類中的屬性提供以下信息:系統(tǒng)可用內(nèi)存總量、分配給對象的內(nèi)存的周期類別(代)。
GC跟蹤并回收托管內(nèi)存中分配的對象。垃圾回收器定期執(zhí)行垃圾回收以回收分配給沒有有效引用的對象的內(nèi)存。當使用可用內(nèi)存不能滿足內(nèi)存請求時,垃圾回收會自動進行;蛘,應(yīng)用程序可以使用 Collect 方法強制進行垃圾回收。
垃圾回收由以下步驟組成:
GC搜索托管代碼中引用的托管對象。
GC嘗試完成沒有被引用的對象。
GC釋放沒有被引用的對象并回收它們的內(nèi)存。
在回收期間,如果GC在托管代碼中找到對某對象的一個或多個引用,則不會釋放該對象。然而,GC不識別非托管代碼中對對象的引用,因此,除非明確禁止,否則它有可能釋放非托管代碼中以獨占方式使用的對象。KeepAlive 方法提供一種機制,該機制可防止垃圾回收器回收在非托管代碼中仍使用的對象。
5.在.NET程序運行過程中,什么是堆,什么是棧?什么情況下會在堆(棧)上分配數(shù)據(jù)?它們有性能上的區(qū)別嗎?“結(jié)構(gòu)”對象可能分配在堆上嗎?什么情況下會發(fā)生,有什么需要注意的?
.NET進程被創(chuàng)建時就會有一個堆隨之被創(chuàng)建, 用來保存該進程在運行中需要使用的對象/數(shù)據(jù); 當一個線程被創(chuàng)建時, 會有一個棧被創(chuàng)建,用來保存方法調(diào)用參數(shù), 局部變量等輕量型數(shù)據(jù).
當一個類里面包含一個結(jié)構(gòu)體類型的變量時, 該結(jié)構(gòu)體類型會被分配在堆上.
>泛型的作用是什么?它有什么優(yōu)勢?它對性能有影響嗎?它在執(zhí)行時的行為是什么?.NET BCL中有哪些泛型類型?舉例說明平時編程中您定義的泛型類型
泛型的作用是什么?
泛型的作用在于“算法的重用”。(這點其實很好理解,原來的ArrayList只能接受Object,現(xiàn)在通過List可以接受任何類型,也就是說ArrayList的方法都被各個類型重用了。但是Dot Net的泛型有個比較制肘地方,就是你很難對數(shù)值類型(值類型)進行算法抽象,因為這牽涉到運算符重載的問題,同時Dot Net的泛型的類型參數(shù)也不能約束成一個基元值類型(如int、double、float) 。)
它有什么優(yōu)勢?
第一:源代碼保護。(如果你知道C++模板對泛型的實現(xiàn)機制,就會知道C++在編譯的時候根據(jù)對泛型的調(diào)用,自動“內(nèi)聯(lián)”了一個實現(xiàn),這樣泛型的內(nèi)容就暴露了,爾DotNet的實現(xiàn)方式就不同了,泛型類和方法會被編譯成IL,在執(zhí)行的時候由JIT負責將IL變化為指定類型參數(shù)的本地代碼,從而保護了源代碼)
第二:類型安全。(這點是最顯而易見的,拋棄了使用ArrayList時各種丑陋的強制類型轉(zhuǎn)換)
第三:更清晰地代碼。因為沒有了強制類型轉(zhuǎn)換,所以代碼自然顯得更清晰,但是使用泛型時候帶來的<>有時候確實也會讓人搞糊涂,幸好泛型方法可以用類型推斷或者using語句來進一步簡化寫法。
第四:更好的性能,因為值類型可以避免裝箱和拆箱所帶來的損耗(垃圾回收的次數(shù)也會減少)。(這點正是泛型神奇的地方,開發(fā)歷史上抽象能力的上升往往意味著性能的下降,但是泛型卻不是!泛型抽象了算法,但是C++和DotNet對泛型的實現(xiàn)能夠讓性能無損,并且更快。Java的擦除法泛型就沒有這種性能上的好處。)
它對性能有影響嗎?
對性能有積極的影響,因為值類型可以避免裝箱和拆箱所帶來的負面影響,避免了垃圾回收,使得性能顯著提高。但是對引用類型這種影響就不明顯了。但是需要注意的是首次為一個特定數(shù)據(jù)類型調(diào)用方法時,CLR都會為這個方法生成本地代碼。這會增大應(yīng)用程序的工作集大小,從而影響性能。
它在執(zhí)行時的行為是什么?
使用泛型類型參數(shù)的一個方法在進行JIT編譯時,CLR獲取IL,用指定的類型實參進行替換,然后創(chuàng)建本地代碼。需要特別注意的是引用類型是共享代碼的,而對值類型就會為每一種生成獨立的一份類型代碼。但是需要指出的是引用類型的這種代碼共享并不會造成封閉類型只執(zhí)行一次構(gòu)造函數(shù)(就算是靜態(tài)構(gòu)造函數(shù)也是這樣的)。
.NET BCL中有哪些泛型類型?
List、Dictionary、Queue、Stack、SortedList和SortedDictionary、LinkedList等等。
舉例說明平時編程中您定義的泛型類型。
泛型的出現(xiàn)會替換原來一部分使用多態(tài)的地方從而提高性能和帶來更好的編譯時檢查,這樣就不需要在子類和超類(接口)間頻繁轉(zhuǎn)換了。比如你要根據(jù)情況打出各種報表,那么先把報表類定義成泛型類從而可以共享報表一系列的算法。
> 異常的作用是什么?.NET BCL中有哪些常見的異常?在代碼中您是如何捕獲/處理異常的?在“catch (ex)”中,“throw”和“throw ex”有什么區(qū)別?您會如何設(shè)計異常的結(jié)構(gòu),什么情況下您會拋出異常?
異常的作用是什么?
異常用于處理系統(tǒng)級或者應(yīng)用程序級的錯誤狀態(tài)。這就會引發(fā)另外幾個問題,異常相比原來使用的返回錯誤代碼的優(yōu)點在哪里?異常處理是一種結(jié)構(gòu)化的處理過程,個人認為他最大的優(yōu)點就在于將“成功場景”剝離出來,使得代碼更加清晰自然。但是異常處理相對于返回錯誤碼有一個缺點,那就是他會失去發(fā)生異常的位置。不過異常本身提供了很多幫助調(diào)試問題的工具,一般都帶有棧跟蹤,這樣位置的問題就得到一定程度的解決。還有就是IF和異常之間的選擇,我記得以前有人討論過在各種分支下是使用異常來處理各種“失敗場景”的分支還是使用IF或者SWITCH來處理呢?這其實是一個假問題,因為異常和錯誤是有概念上的不同的,這里的錯誤是指有違“主成功場景”的“異常場景”,爾異常是指當程序不能完成其名字所表示功能時的錯誤。所以需要強調(diào)不要使用異常來區(qū)分各種失敗場景,異常壓根就不是用來干這件事情的!
.NET BCL中有哪些常見的異常?
隨便說幾個,最著名的恐怕就是那句像繞口令一樣的“未將對象引用設(shè)置到對象實例”了,還有那些基本一出現(xiàn)整個應(yīng)用程序就被判死刑的“堆棧溢出”、“內(nèi)存無法分配”異常了。
在代碼中您是如何捕獲/處理異常的?
這道題的回答可以體現(xiàn)你是什么“級別”的程序員,這個級別倒不是說水平的高低,是指經(jīng)常寫哪一類的程序,如果對異常的捕獲比較“激進”(經(jīng)常捕獲異常)那么這個人應(yīng)該是一個應(yīng)用級的程序員。如果對捕獲異常比較謹慎那么應(yīng)該是框架級別的程序員,這些人經(jīng)常寫給別人使用的代碼,如果無故的使用異常處理來越俎代庖,那后果很嚴重了,這里說一個我經(jīng)歷的事,剛畢業(yè)的時候我和同事做一個WEB的項目,項目里用第三方的Grid,那個Grid在發(fā)生異常的時候會自己報一個錯誤,你知道我們有多傻眼了吧,我們需要的是我們來抓住異常,然后報出一句對用戶友好的錯誤,但是那個控件卻干了這么個蠢事。
我覺得普通程序員用的最多的CATCH就是抓住數(shù)據(jù)的異常,然后回滾數(shù)據(jù)庫來事務(wù)處理。這是一個典型的場景,因為你明確并且能夠很好的恢復(fù)狀態(tài)。
在“catch (ex)”中,“throw”和“throw ex”有什么區(qū)別?
throw 重新拋出異常但是不破壞異常發(fā)生的調(diào)用棧爾“throw ex”會重置調(diào)用棧這樣捕獲異常的人會以為代碼出錯在這里。
您會如何設(shè)計異常的結(jié)構(gòu),什么情況下您會拋出異常?
首先我會盡量的使用系統(tǒng)定義的那些異常,如果我需要處理某一特定類別的異常,而且處理方式和通常處理方式不同那么就考慮自定義異常,還有如果需要調(diào)用方用一種統(tǒng)一的方式來處理異常那么自定義異常就是一個好的選擇。結(jié)構(gòu)的話當然基類是Sysytem.Exception,盡量使用扁平化異常的層次?梢钥紤]用泛型類來定義異常。
我寫的代碼不能完成名字所說明的功能,那么我就會拋出異常。
> List和T[]的區(qū)別是什么,平時你如何進行選擇?Dictionary是做什么的?.NET BCL中還有哪些常用的容器?它們分別是如何實現(xiàn)的(哪種數(shù)據(jù)結(jié)構(gòu))?分別是適用于哪些場景?
List和T[]的區(qū)別明顯有本質(zhì)的區(qū)別,List 是動態(tài)分配內(nèi)存的鏈表,
【.net面試題及詳解】相關(guān)文章:
2011最新asp.net面試題與答案07-11
資深HR詳解面試題的潛臺詞07-13
WAP網(wǎng)絡(luò)與NET網(wǎng)絡(luò)的區(qū)別于定義!07-10
京東為什么在用.net07-11
面試題07-13
《濟南的秋天》詳解07-04
除夕的由來詳解07-04
房屋保險詳解07-13