1. 程式人生 > >4-3 執行緒安全性-原子性-synchronized

4-3 執行緒安全性-原子性-synchronized

原子性 - 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。