摘要:對于方法,用戶一般不會去調用,相當于根據一定的規則將與對象相關的信息映射為一個數值,稱為散列值。一般在在覆蓋方法的同時也要覆蓋方法,否則將會違反的通用約定,從而導致該類無法與所有基于散列值的集合類結合在一起正常工作。
1. Java中如何比較兩個Long對象是否相等?
如果Long的值在[-127,128]之間,用“==”判斷是否相等是沒問題的,如果不在這個區間,是不能用“==”的,原因如下源碼解釋:
public static Long valueOf(long l) { final int offset = 128; if (l >= -128 && l <= 127) { // will cache return LongCache.cache[(int)l + offset]; } return new Long(l); }
如果不在[-127,128]之間,則會new一個新對象,自然“==”兩個不同的對象,其結果必然是false了。
解決辦法:
a. : 使用Long中的longValue()進行轉換
Long a = 128l; Long b = 128l; a.longValue() == b.longValue() //true
b. : Long中的equals()
public boolean equals(Object obj) { if (obj instanceof Long) { return value == ((Long)obj).longValue(); } return false; }2.Java中進程和線程的區別
https://blog.csdn.net/QQ1608731824/article/details/81461269
進程是資源分配的最小單位,線程是CPU調度的最小單位
地址空間和其它資源:進程間相互獨立,同一進程的各線程間共享。某進程內的線程在其它進程不可見。
通信:進程間通信IPC,線程間可以直接讀寫進程數據段(如全局變量)來進行通信——需要進程同步和互斥手段的輔助,以保證數據的一致性。
調度和切換:線程上下文切換比進程上下文切換要快得多。
在多線程OS中,進程不是一個可執行的實體。
線程之間如何實現資源共享:
將要共享的數據封裝成另外一個對象,對這個對象進行操作
將Runnable作為一個內部類,并在外部類中定義要共享的成員變量
3. 重寫equals為什么要重寫hashcode()?對于==:如果作用于基本數據類型的變量,則直接比較其存儲的 “值”是否相等;
如果作用于引用類型的變量,則比較的是所指向的對象的地址
對于equals方法:注意:equals方法不能作用于基本數據類型的變量
如果沒有對equals方法進行重寫,則比較的是引用類型的變量所指向的對象的地址;
諸如String、Date等類對equals方法進行了重寫的話,比較的是所指向的對象的內容。
hashCode()方法:Object類中的本地方法,也用來判斷兩個對象是否相等。
如果沒有重寫hashCode()方法,該方法返回的是對象在內存中的地址轉換成的int值,因此,任何對象的hashCode()方法值是不相等的。
equals()方法也是判斷兩個對象是否相等,但是與hashCode()方法有區別。一般來講,equals()方法是給用戶調用的,依據情況可以選擇重寫或者不重寫。對于hashCode()方法,用戶一般不會去調用,hashCode相當于根據一定的規則將與對象相關的信息映射為一個數值,稱為散列值。一般在在覆蓋equals()方法的同時也要覆蓋hashCode()方法,否則將會違反Object hashCode的通用約定,從而導致該類無法與所有基于散列值的集合類結合在一起正常工作。
Object hashCode()方法的通用約定:
如果x.equals(y)返回true,那么這兩個對象的hashCode()方法必須產生同樣的整數結果;
如果x.equals(y)返回false,那么這兩個對象的hashCode()方法產生的結果也可能相等,也可能不想等;
如果兩個對象的hashCode()方法返回值不相等,則x.equals(y)一定為false
如果兩個對象的hashCode()方法返回值相等,則x.equals(y)可能為true,可能為false。
4. Java的淺拷貝和深拷貝淺拷貝:使用一個已知實例對新創建實例的成員變量逐個賦值,這個方式被稱為淺拷貝。
深拷貝:當一個類的拷貝構造方法,不僅要復制對象的所有非引用成員變量值,還要為引用類型的成員變量創建新的實例,并且初始化為形式參數實例值。
也就是說淺拷貝只復制一個對象,傳遞引用,不能復制實例。而深拷貝對對象內部的引用均復制,它是創建一個新的實例,并且復制實例。對于淺拷貝當對象的成員變量是基本數據類型時,兩個對象的成員變量已有存儲空間,賦值運算傳遞值,所以淺拷貝能夠復制實例。但是當對象的成員變量是引用數據類型時,就不能實現對象的復制了
5. 為什么String被設計成不可變String 構成:
public final class String implements java.io.Serializable, Comparable, CharSequence{ /** The value is used for character storage. */ private final char value[]; /** Cache the hash code for the string */ private int hash; // Default to 0 ...
源碼我們可以知道String底層是由char 數組構成的,我們創建一個字符串對象的時候,其實是將字符串保存在char數組中,因為數組是引用對象,為了防止數組可變,jdk加了final修飾,加了final修飾的數組只是代表了引用不可變,不代表數組的內容不可變,因此jdk為了真正防止不可變,又加了一個private修飾符。
說到String 不得不提字符串常量池,這個常量池主要存儲在方法區中,當一個字符串被創建的時候,首先會去常量池中查找,如果找到了就返回對改字符串的引用,如果沒找到就創建這個字符串并塞到常量池中。
下面代碼只會在:
String s1 ="abc"; String s2 ="abc";
試想一下如果String是可變的,當兩個引用指向指向同一個字符串時,對其中一個做修改就會影響另外一個。5.1 什么是不可變?
對于Java而言,除了基本類型(即int, long, double等),其余的都是對象。
對于何為不可變對象,《java concurrency in practice》一書給出了一個粗略的定義:對象一旦創建后,其狀態不可修改,則該對象為不可變對象。
一般一個對象滿足以下三點,則可以稱為是不可變對象:
其狀態不能在創建后再修改;
所有域都是final類型;
其構造函數構造對象期間,this引用沒有泄露。
這里重點說明一下第2點,一個對象其所有域都是final類型,該對象也可能是可變對象。因為final關鍵字只是限制對象的域的引用不可變,但無法限制通過該引用去修改其對應域的內部狀態。因此,嚴格意義上的不可變對象,其final關鍵字修飾的域應該也是不可變對象和primitive type值。
從技術上講,不可變對象內部域并不一定全都聲明為final類型,String類型即是如此。在String對象的內部我們可以看到有一個名為hash的域并不是final類型,這是因為String類型惰性計算hashcode并存儲在hash域中(這是通過其他final類型域來保證每次的hashcode計算結果必定是相同的)。除此之外,String對象的不可變是由于對String類型的所有改變內部存儲結構的操作都會new出一個新的String對象。5.2 不可變帶來的好處(1):安全性
1、多線程安全性因為String是不可變的,因此在多線程操作下,它是安全的,我們看下如下代碼:
public String get(String str){ str +="aaa"; return str; }
試想一下如果String是可變的,那么get方法內部改變了str的值,方法外部str也會隨之改變。
2、類加載中體現的安全性類加載器要用到字符串,不可變性提供了安全性,以便正確的類被加載。
譬如你想加載java.sql. Connection類,而這個值被改成了hacked.Connection,那么會對你的數據庫造成不可知的破壞。
只有當字符串是不可變的,字符串池才有可能實現。字符串池的實現可以在運行時節約很多heap空間,因為不同的字符串變量都指向池中的同一個字符串。但如果字符串是可變的,那么String interning將不能實現(String interning是指對不同的字符串僅僅只保存一個,即不會保存多個相同的字符串),因為這樣的話,如果變量改變了它的值,那么其它指向這個值的變量的值也會一起改變
5.4 不可變帶來的好處(3): 緩存hashcode因為字符串是不可變的,所以在它創建的時候hashcode就被緩存了,不需要重新計算。這就使得字符串很適合作為Map中的鍵,字符串的處理速度要快過其它的鍵對象。這就是HashMap中的鍵往往都使用字符串。我們可以看到String中有如下代碼:
private int hash;//this is used to cache hash code.
以上代碼中hash變量中就保存了一個String對象的hashcode,因為String類不可變,所以一旦對象被創建,該hash值也無法改變。所以,每次想要使用該對象的hashcode的時候,直接返回即可。
5.5 不可變帶來的缺點不可變對象也有一個缺點就是會制造大量垃圾,由于他們不能被重用而且對于它們的使用就是”用“然后”扔“,字符串就是一個典型的例子,它會創造很多的垃圾,給垃圾收集帶來很大的麻煩。當然這只是個極端的例子,合理的使用不可變對象會創造很大的價值。
密碼應該存放在字符數組中而不是String中由于String在Java中是不可變的,如果你將密碼以明文的形式保存成字符串,那么它將一直留在內存中,直到垃圾收集器把它清除。而由于字符串被放在字符串緩沖池中以方便重復使用,所以它就可能在內存中被保留很長時間,而這將導致安全隱患,因為任何能夠訪問內存(memorydump內存轉儲)的人都能清晰的看到文本中的密碼,這也是為什么你應該總是使用加密的形式而不是明文來保存密碼。由于字符串是不可變的,所以沒有任何方式可以修改字符串的值,因為每次修改都將產生新的字符串,然而如果你使用char[]來保存密碼,你仍然可以將其中所有的元素都設置為空或者零。所以將密碼保存到字符數組中很明顯的降低了密碼被竊取的風險。當然只使用字符數組也是不夠的,為了更安全你需要將數組內容進行轉化。建議使用哈希的或者是加密過的密碼而不是明文,然后一旦完成驗證,就將它從內存中清除掉
6. String 字符串常量,StringBuffer 字符串變量(線程安全),StringBuilder 字符串變量(非線程安全)一旦一個string對象在內存(堆)中被創建出來,他就無法被修改。特別要注意的是,String類的所有方法都沒有改變字符串本身的值,都是返回了一個新的對象。
如果你需要一個可修改的字符串,應該使用StringBuffer或者StringBuilder。否則會有大量時間浪費在垃圾回收上,因為每次試圖修改都有新的string對象被創建出來。
查看API會發現,String、StringBuffer、StringBuilder都實現了 CharSequence接口,雖然它們都與字符串相關,但是其處理機制不同。
String:是不可改變的量,也就是創建后就不能在修改了。
StringBuffer:是一個可變字符串序列,它與String一樣,在內存中保存的都是一個有序的字符串序列(char類型的數組),不同點是StringBuffer對象的值是可變的。
StringBuilder:與StringBuffer類基本相同,都是可變字符串序列,不同點是StringBuffer是線程安全的,StringBuilder是線程不安全的。 在性能方面,由于String類的操作是產生新的String對象,而StringBuilder和StringBuffer只是一個字符數組的擴容而已,所以String類的操作要遠慢于StringBuffer和StringBuilder。
使用String類的場景:在字符串不經常變化的場景中可以使用String類,例如常量的聲明、少量的變量運算。共享的場合
使用StringBuffer類的場景:在頻繁進行字符串運算(如拼接、替換、刪除等),并且運行在多線程環境中,則可以考慮使用StringBuffer,例如XML解析、HTTP參數解析和封裝。
使用StringBuilder類的場景:在頻繁進行字符串運算(如拼接、替換、和刪除等),并且運行在單線程的環境中,則可以考慮使用StringBuilder,如SQL語句的拼裝、JSON封裝等。
簡要的說,String 類型和 StringBuffer 類型的主要性能區別其實在于 String 是不可變的對象, 因此在每次對 String 類型進行改變的時候其實都等同于生成了一個新的 String 對象,然后將指針指向新的 String 對象。所以經常改變內容的字符串最好不要用 String,因為每次生成對象都會對系統性能產生影響,特別當內存中無引用對象多了以后,JVM 的 GC 就會開始工作,那速度是一定會相當慢的。
而如果是使用StringBuffer類則結果就不一樣了,每次結果都會對 StringBuffer 對象本身進行操作,而不是生成新的對象,再改變對象引用。所以在一般情況下我們推薦使用 StringBuffer,特別是字符串對象經常改變的情況下。
在某些特別情況下, String 對象的字符串拼接其實是被 JVM 解釋成了 StringBuffer 對象的拼接,所以這些時候 String 對象的速度并不會比 StringBuffer 對象慢,而特別是以下的字符串對象生成中, String 效率是遠要比 StringBuffer 快的:
String S1 = "This is only a" + "simple" + "test"; StringBuffer Sb =new StringBuilder(“This is only a").append("simple").append(" test");
你會很驚訝的發現,生成 String S1 對象的速度簡直太快了,而這個時候 StringBuffer 居然速度上根本一點都不占優勢。其實這是 JVM 的一個把戲,在 JVM 眼里,這個
String S1 = “This is only a" + “ simple" + “test";
其實就是:String S1 = “This is only a simple test";
所以當然不需要太多的時間了。但大家這里要注意的是,如果你的字符串是來自另外的 String 對象的話,速度就沒那么快了,譬如:
String S2 = "This is only a"; String S3 = "simple"; String S4 = "test"; String S1 = S2 +S3 + S4;
這時候 JVM 會規規矩矩的按照原來的方式去做。
6.4 總結在大部分情況下 StringBuffer > String
Java.lang.StringBuffer是線程安全的可變字符序列。一個類似于 String 的字符串緩沖區,但不能修改。雖然在任意時間點上它都包含某種特定的字符序列,但通過某些方法調用可以改變該序列的長度和內容。在程序中可將字符串緩沖區安全地用于多線程。而且在必要時可以對這些方法進行同步,因此任意特定實例上的所有操作就好像是以串行順序發生的,該順序與所涉及的每個線程進行的方法調用順序一致。
StringBuffer 上的主要操作是 append 和 insert 方法,可重載這些方法,以接受任意類型的數據。每個方法都能有效地將給定的數據轉換成字符串,然后將該字符串的字符追加或插入到字符串緩沖區中。append 方法始終將這些字符添加到緩沖區的末端;而 insert 方法則在指定的點添加字符。
例如,如果 z 引用一個當前內容是“start”的字符串緩沖區對象,則此方法調用 z.append(“le”) 會使字符串緩沖區包含“startle”(累加);而 z.insert(4, “le”) 將更改字符串緩沖區,使之包含“starlet”。
在大部分情況下 StringBuilder > StringBuffer
java.lang.StringBuilder是一個可變的字符序列,是JAVA 5.0新增的。此類提供一個與 StringBuffer 兼容的 API,但不保證同步,所以使用場景是單線程。該類被設計用作 StringBuffer 的一個簡易替換,用在字符串緩沖區被單個線程使用的時候(這種情況很普遍)。如果可能,建議優先采用該類,因為在大多數實現中,它比 StringBuffer 要快。兩者的使用方法基本相同。
7. 序列化和反序列化 7.1 什么是序列化和反序列化Java序列化是指把Java對象保存為二進制字節碼的過程,Java反序列化是指把二進制碼重新轉換成Java對象的過程。
7.2 為什么要序列化?1.java對象的生命周期要比java虛擬機短,實際應用中希望虛擬機停止運行之后能夠持久化指定的對象,此時可以將對象序列化保存;
2.java對象通過網絡傳輸的時候,因為數據只能以二進制的形式在網絡中進行傳輸,因此當對象通過網絡發送出去之前,需要先序列化為二進制數據,在接收端收到二進制數據之后反序列化成二進制對象。
在序列化的時候使用默認的方式來進行序列化,這種序列化方式僅僅對對象的非transient的實例變量進行序列化,而不會序列化對象的transient的實例變量,也不會序列化靜態變量,所以我們對不想持久化的變量可以加上transient關鍵字。注意使用默認機制,在序列化對象時,不僅會序列化當前對象本身,還會對該對象引用的其它對象也進行序列化,同樣地,這些其它對象引用的另外對象也將被序列化,以此類推。所以,如果一個對象包含的成員變量是容器類對象,而這些容器所含有的元素也是容器類對象,那么這個序列化的過程就會較復雜,開銷也較大。如果需要實現自定義序列化和反序列化,那么需要重寫writeObject()方法和readObject()方法。在序列化過程中,如果被序列化的類中定義了writeObject 和 readObject 方法,將會使用反射的方式調用自定義的 writeObject 和 readObject 方法,進行用戶自定義的序列化和反序列化。
2.實現Externalizable接口也可以完成自定義序列化,必須實現writeExternal()方法和readExternal()方法。 7.4 Serializable 和 Externalizable 的區別Serializable既可以采用默認的序列化和反序列化方式,也可以使用用戶自定義的序列化和反序列化的方式。
Externalizable序列化,雖然Externalizable接口繼承自Serializable接口,但是需要序列化類繼承此接口的話,Serializable所有序列化機制全部失效。Externalizable序列化的過程:使用Externalizable序列化時,在進行反序列化的時候,會重新實例化一個對象,然后再將被反序列化的對象的狀態全部復制到這個新的實例化對象當中去,因此必須有一個無參構造方法供其調用,并且權限是public。
Java序列化和反序列化本質上是將對象信息生成一串二進制字節碼和從二進制字節碼解析的過程。序列化算法:
1)當前類的描述
2)當前類屬性的描述
3)父類描述
4)父類屬性描述
5)父類屬性值描述
6)子類屬性值描述
類描述是從下到上,類屬性描述是從上到下。8. 如何理解面向對象
面向對象的三個特征:封裝、繼承、多態
封裝:封裝是將客觀事物抽象成類,每個類都包含自身的數據以及操作,不必需要其他的類來完成操作。類內部的實現可以自由的修改;具有清晰的對外接口。良好的封裝能夠減少耦合;
繼承:繼承是從已有的類中派生出新的類稱為子類,子類繼承父類的屬性和行為,并能夠根據自己的需求擴展新的行為,提供了代碼的復用性。
多態:多態允許不同類的對象對同一消息做出響應。提供繼承關系,子類重寫父類的方法;父類的引用執行子類的對象;在調用父類的方法是實際上表現的是子類的狀態。
從內存角度考慮:
局部變量在棧內存中存在,當for循環語句結束,那么變量會及時被gc(垃圾回收器)及時的釋放掉,不浪費空間
如果使用循環之后還想去訪問循環語句中控制那個變量,使用while循環
從應用場景角度考慮:
如果一個需求明確循環的次數,那么使用for循環(開發中使用for循環的幾率大于while循環)
如果一個需求,不知道循環了多少次,使用while循環
10. Servlet的生命周期、Servlet是否線程安全在servlet容器啟動時初始化。在web.xml
servlet在第一次被訪問時初始化。即創建唯一的servlet實例。(單例多線程下面會說)
當有請求訪問該servlet是,servlet容器就會創建針對于這個請求的servletRequest于servletResponse,然后servlet的service方法被調用。當容器把servlet生成的響應結果發送給客戶,容器就會銷毀request和response對象
容器在銷毀該實例前調用servlet的destroy方法(釋放servlet所占用的資源,如關閉流和數據庫連接),此外還會銷毀與servlet對象關聯的ServletConfig對象。
servlet類只創建一個實例,對于可與客戶端的并發訪問,它是線程不安全的。
servlet的處理方式是,每次訪問時重新起一線程執行service方法。所以要想保證servlet的線程安全,不應該在servlet中定義實例變量。
當然完全可以通過加鎖保證線程安全,但對于成千上萬的并發訪問,性能下降。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/76706.html
摘要:作為面試官,我是如何甄別應聘者的包裝程度語言和等其他語言的對比分析和主從復制的原理詳解和持久化的原理是什么面試中經常被問到的持久化與恢復實現故障恢復自動化詳解哨兵技術查漏補缺最易錯過的技術要點大掃盲意外宕機不難解決,但你真的懂數據恢復嗎每秒 作為面試官,我是如何甄別應聘者的包裝程度Go語言和Java、python等其他語言的對比分析 Redis和MySQL Redis:主從復制的原理詳...
摘要:作為面試官,我是如何甄別應聘者的包裝程度語言和等其他語言的對比分析和主從復制的原理詳解和持久化的原理是什么面試中經常被問到的持久化與恢復實現故障恢復自動化詳解哨兵技術查漏補缺最易錯過的技術要點大掃盲意外宕機不難解決,但你真的懂數據恢復嗎每秒 作為面試官,我是如何甄別應聘者的包裝程度Go語言和Java、python等其他語言的對比分析 Redis和MySQL Redis:主從復制的原理詳...
摘要:好不容易在月號這天中午點左右接到了來自阿里的面試電話。這里會不斷收集和更新基礎相關的面試題,目前已收集題。面試重難點的和的打包過程多線程機制機制系統啟動過程,啟動過程等等掃清面試障礙最新面試經驗分享,此為第一篇,開篇。 2016 年末,騰訊,百度,華為,搜狗和滴滴面試題匯總 2016 年未,騰訊,百度,華為,搜狗和滴滴面試題匯總 各大公司 Java 后端開發面試題總結 各大公司 Jav...
摘要:今天整理了一下近大半年以來的一些文章,和我的預期一樣,很多文章我都忘記自己曾經寫過了,這個記錄的過程讓我也有了新的理解。希望大家,收藏,點贊,加轉發。 今天整理了一下近大半年以來的一些文章,和我的預期一樣,很多文章我都忘記自己曾經寫過了,這個記錄的過程讓我也有了新的理解。希望大家,收藏,點贊,加轉發。 面試必備 面試必備:深入Spring MVC DispatchServlet 源碼...
閱讀 2759·2021-10-09 09:44
閱讀 3561·2019-08-30 15:54
閱讀 2171·2019-08-30 14:16
閱讀 2803·2019-08-30 13:09
閱讀 835·2019-08-30 13:08
閱讀 1296·2019-08-29 16:29
閱讀 1683·2019-08-26 13:57
閱讀 1940·2019-08-26 13:53