Partage de technologie

Sécurité des threads (2) Principe de mise en œuvre sous-jacent de la structure synchronisée, de mise à niveau du verrouillage et de mémoire d'objet

2024-07-12

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

1. Utilisation de base

  • Supposons qu'il existe un tel scénario :20 utilisateurs récupèrent 10 billets ensemble

1.1 Implémentation du code sans verrouillage

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

Résultats du :

Vous pouvez voir qu'il apparaît problème de survente , il y a 10 tickets au total. Lorsque 20 fils récupèrent des tickets ensemble, 11 personnes semblent avoir récupéré les tickets. En effet, si les deux threads réussissent la vérification if en même temps.

Par conséquent, nous devons verrouiller l'opération du ticket pour garantir qu'un seul thread peut vérifier et effectuer la déduction du ticket en même temps.

1.2 Implémentation du code de verrouillage

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