1、Java 中能創建 volatile 陣列嗎?
能,Java 中可以創建 volatile 型別陣列,不過只是一個指向陣列的參考,而不是整個陣列,我的意思是,如果改變參考指向的陣列,將會受到 volatile 的保護,但是如果多個執行緒同時改變陣列的元素,volatile 標示符就不能起到之前的保護作用了,
2、volatile 能使得一個非原子操作變成原子操作嗎?
一個典型的例子是在類中有一個 long 型別的成員變數,如果你知道該成員變數會被多個執行緒訪問,如計數器、價格等,你最好是將其設定為 volatile,為什么?因為 Java 中讀取 long 型別變數不是原子的,需要分成兩步,如果一個執行緒正在修改該 long 變數的值,另一個執行緒可能只能看到該值的一半(前 32 位),但是對一個 volatile 型的 long 或 double 變數的讀寫是原子,
3、volatile 修飾符的有過什么實踐?
一種實踐是用 volatile 修飾 long 和 double 變數,使其能按原子型別來讀寫,double 和 long 都是 64 位寬,因此對這兩種型別的讀是分為兩部分的,第一次讀取第一個 32 位,然后再讀剩下的 32 位,這個程序不是原子的,但 Java 中volatile 型的 long 或 double 變數的讀寫是原子的,volatile 修復符的另一個作用是提供記憶體屏障(memory barrier),例如在分布式框架中的應用,簡單的說,就是當你寫一個 volatile 變數之前,Java 記憶體模型會插入一個寫屏障(writebarrier),讀一個 volatile 變數之前,會插入一個讀屏障(read barrier),意思就是說,在你寫一個 volatile 域時,能保證任何執行緒都能看到你寫的值,同時,在寫之前,也能保證任何數值的更新對所有執行緒是可見的,因為記憶體屏障會將其他所有寫的值更新到快取,
4、volatile 型別變數提供什么保證?
volatile 變數提供順序和可見性保證,例如,JVM 或者 JIT 為了獲得更好的性能會對陳述句重排序,但是 volatile 型別變數即使在沒有同步塊的情況下賦值也不會與其他陳述句重排序,volatile 提供 happens-before 的保證,確保一個執行緒的修改能對其他執行緒是可見的,某些情況下,volatile 還能提供原子性,如讀 64 位資料型別,像 long 和 double 都不是原子的,但 volatile 型別的 double 和long 就是原子的,
5、10 個執行緒和 2 個執行緒的同步代碼,哪個更容易寫?
從寫代碼的角度來說,兩者的復雜度是相同的,因為同步代碼與執行緒數量是相互獨立的,但是同步策略的選擇依賴于執行緒的數量,因為越多的執行緒意味著更大的競爭,所以你需要利用同步技術,如鎖分離,這要求更復雜的代碼和專業知識,
6、你是如何呼叫 wait()方法的?使用 if 塊還是回圈?為什么?
wait() 方法應該在回圈呼叫,因為當執行緒獲取到 CPU 開始執行的時候,其他條件可能還沒有滿足,所以在處理前,回圈檢測條件是否滿足會更好,下面是一段標準的使用 wait 和 notify 方法的代碼:
// The standard idiom for using the wait methodsynchronized (obj) { while (condition does not hold) obj.wait(); // (Releases lock, and reacquires on wakeup) ... // Perform action appropriate to condition}
7、什么是多執行緒環境下的偽共享(false sharing)?
偽共享是多執行緒系統(每個處理器有自己的區域快取)中一個眾所周知的性能問題,偽共享發生在不同處理器的上的執行緒對變數的修改依賴于相同的快取行,
8、什么是 Busy spin?我們為什么要使用它?
Busy spin 是一種在不釋放 CPU 的基礎上等待事件的技術,它經常用于避免丟失 CPU 快取中的資料(如果執行緒先暫停,之后在其他 CPU 上運行就會丟失),所以,如果你的作業要求低延遲,并且你的執行緒目前沒有任何順序,這樣你就可以通過回圈檢測佇列中的新訊息來代替呼叫 sleep() 或 wait() 方法,它唯一的好處就是你只需等待很短的時間,如幾微秒或幾納秒,LMAX 分布式框架是一個高性能執行緒間通信的庫,該庫有一個 BusySpinWaitStrategy 類就是基于這個概念實作的,使用 busy spin 回圈 EventProcessors 等待屏障,
9、Java 中怎么獲取一份執行緒 dump 檔案?
在 Linux 下,你可以通過命令 kill -3 PID (Java 行程的行程 ID)來獲取 Java應用的 dump 檔案,在 Windows 下,你可以按下 Ctrl + Break 來獲取,這樣 JVM 就會將執行緒的 dump 檔案列印到標準輸出或錯誤檔案中,它可能列印在控制臺或者日志檔案中,具體位置依賴應用的配置,如果你使用 Tomcat,
10、Swing 是執行緒安全的?
不是,Swing 不是執行緒安全的,你不能通過任何執行緒來更新 Swing 組件,如JTable、JList 或 JPanel,事實上,它們只能通過 GUI 或 AWT 執行緒來更新,這就是為什么 Swing供 invokeAndWait() 和 invokeLater() 方法來獲取其他執行緒的 GUI 更新請求,這些方法將更新請求放入 AWT 的執行緒佇列中,可以一直等待,也可以通過異步更新直接回傳結果,你也可以在參考答案中查看和學習到更詳細的內容,
11、什么是執行緒區域變數?
執行緒區域變數是局限于執行緒內部的變數,屬于執行緒自身所有,不在多個執行緒間共享,Java 提供 ThreadLocal 類來支持執行緒區域變數,是一種實作執行緒安全的方式,但是在管理環境下(如 web 服務器)使用執行緒區域變數的時候要特別小心,在這種情況下,作業執行緒的生命周期比任何應用變數的生命周期都要長,任何執行緒區域變數一旦在作業完成后沒有釋放,Java 應用就存在記憶體泄露的風險,
12、用 wait-notify 寫一段代碼來解決生產者-消費者問題?
只要記住在同步塊中呼叫 wait() 和 notify()方 法 ,如果阻塞,通過回圈來測驗等待條件,
13、用 Java 寫一個執行緒安全的單例模式(Singleton)?
一步一步創建一個執行緒安全的 Java 單例類,當我們說執行緒安全時,意思是即使初始化是在多執行緒環境中,仍然能保證單個實體,Java 中,使用列舉作為單例類是最簡單的方式來創建執行緒安全單例模式的方式,
14、Java 中 sleep 方法和 wait 方法的區別?
雖然兩者都是用來暫停當前運行的執行緒,但是 sleep() 實際上只是短暫停頓,因為它不會釋放鎖,而 wait() 意味著條件等待,這就是為什么該方法要釋放鎖,因為只有這樣,其他等待的執行緒才能在滿足條件時獲取到該鎖,
15、什么是不可變物件(immutable object)?Java 中怎么創建一個不可變物件?
不可變物件指物件一旦被創建,狀態就不能再改變,任何修改都會創建一個新的物件,如 String、Integer 及其它包裝類,詳情參見答案,一步一步指導你在 Java中創建一個不可變的類,
16、我們能創建一個包含可變物件的不可變物件嗎?
是的,我們是可以創建一個包含可變物件的不可變物件的,你只需要謹慎一點,不要共享可變物件的參考就可以了,如果需要變化時,就回傳原物件的一個拷貝,最常見的例子就是物件中包含一個日期物件的參考,資料型別和 Java 基礎面試問題
17、Java 中應該使用什么資料型別來代表價格?
如果不是特別關心記憶體和性能的話,使用 BigDecimal,否則使用預定義精度的double 型別,
18、怎么將 byte 轉換為 String?
可以使用 String 接收 byte[] 引數的構造器來進行轉換,需要注意的點是要使用的正確的編碼,否則會使用平臺默認編碼,這個編碼可能跟原來的編碼相同,也可能不同,
19、Java 中怎樣將bytes[] 轉換為 long 型別?
bytes[] 到數字型別的轉換是個經常用到的代碼,解決方式也不止一種,
java代碼實作,如果不想借助任何已經有的類,完全可以自己實作這段代碼,如下:
/** * 將位元組陣列轉為long<br> * 如果input為null,或offset指定的剩余陣列長度不足8位元組則拋出例外 * @param input * @param offset 起始偏移量 * @param littleEndian 輸入陣列是否小端模式 * @return */public static long longFrom8Bytes(byte[] input, int offset, Boolean littleEndian){ long value=https://www.cnblogs.com/hwtnet/p/0; // 回圈讀取每個位元組通過移位運算完成long的8個位元組拼裝 for (int count=0;count<8;++count){ int shift=(littleEndian?count:(7-count))<<3; value |=((long)0xff<< shift) & ((long)input[offset+count] << shift); } return value;}
借助java.nio.ByteBuffer實作
java.nio.ByteBuffer 本身就有getLong,getInt,getFloat….方法,只要將byte[]轉換為ByteBuffer就可以實作所有primitive型別的資料讀取,參見javadoc,
/** * 利用 {@link java.nio.ByteBuffer}實作byte[]轉long * @param input * @param offset * @param littleEndian 輸入陣列是否小端模式 * @return */public static long bytesTolong(byte[] input, int offset, Boolean littleEndian) { // 將byte[] 封裝為 ByteBuffer ByteBuffer buffer = ByteBuffer.wrap(input,offset,8); if(littleEndian){ // ByteBuffer.order(ByteOrder) 方法指定位元組序,即大小端模式(BIG_ENDIAN/LITTLE_ENDIAN) // ByteBuffer 默認為大端(BIG_ENDIAN)模式 buffer.order(ByteOrder.LITTLE_ENDIAN); } return buffer.getlong();}
借助java.io.DataInputStream實作
java.io.DataInputStream 同樣提供了readLong,readLong,readLong….方法,只要將byte[]轉換為DataInputStream就可以實作所有primitive型別的資料讀取,參見javadoc,
20、我們能將 int 強制轉換為 byte 型別的變數嗎?如果該值大于 byte 型別的范圍,將會出現什么現象?
是的,我們可以做強制轉換,但是 Java 中 int 是 32 位的,而 byte 是 8 位的,所以,如果強制轉化是,int 型別的高 24 位將會被丟棄,byte 型別的范圍是從 -128 到 127,
轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/5234.html
標籤:其他
