4-3 執行緒安全性-原子性-synchronized
阿新 • • 發佈:2018-12-14
原子性 - Synchronize
- 修飾一個程式碼塊:作用區域是呼叫方法的物件,不同物件之間互不影響。
- 修飾一個方法:作用區域是呼叫方法的物件,不同物件之間互不影響
- 修飾一個類:作用區域是是所有物件
- 修飾一個靜態方法:作用區域是是所有物件
程式碼示例 Synchronize修飾一個程式碼塊和修飾一個方法,表現一樣
package com.mmall.concurrency.example.sync; import lombok.extern.slf4j.Slf4j; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; @Slf4j /* Synchronize修飾一個程式碼塊和修飾一個方法,表現一樣 * */ public class SynchronizedExample1 { // 修飾一個程式碼塊:作用區域是呼叫方法的物件,不同物件之間互不影響 public void test1(int j) { synchronized (this) { for (int i = 0; i < 10; i++) { log.info("test1 {} - {}", j, i); } } } // 修飾一個方法:作用區域是呼叫方法的物件,不同物件之間互不影響 // 如果此類是父類,子類要呼叫父類的時候,是不包含synchronized的,原因是synchronized不是方法宣告的一部分。 如果子類也需要synchronized的話,需要宣告synchronized。 public synchronized void test2(int j) { for (int i = 0; i < 10; i++) { log.info("test2 {} - {}", j, i); } } public static void main(String[] args) { SynchronizedExample1 example1 = new SynchronizedExample1(); SynchronizedExample1 example2 = new SynchronizedExample1(); ExecutorService executorService = Executors.newCachedThreadPool(); //2個物件呼叫方法1 // executorService.execute(() -> { // // 執行結果是:test1 1和test1 2從0-9交叉出現 // example1.test1(1); // }); // executorService.execute(() -> { // example1.test1(2); // }); //2個物件呼叫方法2 executorService.execute(() -> { // 執行結果也是:test1 1和test1 2從0-9交叉出現 example1.test2(1); }); executorService.execute(() -> { example2.test2(2); }); } }
Synchronize修飾一個類和修飾一個靜態方法,表現一樣
package com.mmall.concurrency.example.sync; import lombok.extern.slf4j.Slf4j; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; @Slf4j /* Synchronize修飾一個類和修飾一個靜態方法,表現一樣 * */ public class SynchronizedExample2 { // 修飾一個類:作用區域是是所有物件 public static void test1(int j) { synchronized (SynchronizedExample2.class) { for (int i = 0; i < 10; i++) { log.info("test1 {} - {}", j, i); } } } // 修飾一個靜態方法:作用區域是是所有物件 public static synchronized void test2(int j) { for (int i = 0; i < 10; i++) { log.info("test2 {} - {}", j, i); } } public static void main(String[] args) { SynchronizedExample2 example1 = new SynchronizedExample2(); SynchronizedExample2 example2 = new SynchronizedExample2(); ExecutorService executorService = Executors.newCachedThreadPool(); //2個物件呼叫方法2 // executorService.execute(() -> { // // 執行結果是:test1 1和test1 2分別從0-9出現(不交叉) // example1.test2(1); // }); // executorService.execute(() -> { // example2.test2(2); // }); //2個物件呼叫方法1 executorService.execute(() -> { // 執行結果也是:test1 1和test1 2分別從0-9出現(不交叉) example1.test1(1); }); executorService.execute(() -> { example2.test1(2); }); } }
Synchronize修飾一個方法時,如果此類是父類,子類要呼叫父類的時候,是不包含synchronized的,原因是synchronized不是方法宣告的一部分。 如果子類也需要synchronized的話,需要宣告synchronized。