Berbagi teknologi

Keamanan thread (2) Prinsip implementasi yang mendasari sinkronisasi, peningkatan kunci, dan struktur memori objek

2024-07-12

한어Русский языкEnglishFrançaisIndonesianSanskrit日本語DeutschPortuguêsΕλληνικάespañolItalianoSuomalainenLatina

1. Penggunaan dasar

  • Misalkan ada skenario seperti ini:20 pengguna mengambil 10 tiket bersama-sama

1.1 Implementasi kode tanpa penguncian

public class TicketDemo {
   

    // 票总数
    private int ticketNum = 10;

    /**
     * 抢票
     */
    public void getTicket() {
   
        if (ticketNum <= 0) {
   
            return;
        }
        System.out.println(Thread.currentThread().getName() + " 抢到一张票,剩余:" + ticketNum);
        // 非原子性操作
        ticketNum--;
    }

    /**
     * 测试:20个人抢一张票
     */
    public static void main(String[] args) {
   
        TicketDemo ticketDemo = new TicketDemo();
        for (int i = 0; i < 20; i++) {
   
            new Thread(ticketDemo::getTicket).start();
        }
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32

Hasil dari:

Anda bisa melihatnya muncul masalah penjualan berlebih , total ada 10 tiket. Ketika 20 thread mengambil tiket bersama-sama, 11 orang tampaknya telah mengambil tiket tersebut. Ini karena jika kedua thread melewati pemeriksaan if secara bersamaan.

Oleh karena itu, kita perlu mengunci operasi tiket untuk memastikan bahwa hanya satu thread yang dapat memeriksa dan mengoperasikan pengurangan tiket pada saat yang bersamaan.

1.2 Implementasi kode penguncian

public class TicketDemo {
   

    // 锁
    private static Object lock = new Object();

    // 票总数
    private int ticketNum = 10;

    /**
     * 抢票
     */
    public void getTicket() {
   
        synchronized (lock) {
   
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16