* 例子:建立三個視窗賣票,總票數為100張,使用實作Runnable接口的方式。
* 1.問題:賣票過程中出現了重票和錯票----->線程的安全問題
* 2.問題出現的原因:當某個線程操作車票的過程中,尚未操作完成時,其他線程參與進來,也操作車票。
* 3.如何解決:當一個線程a在操作ticket時,其他線程不能參與進來,直到線程a操作完ticket時,其他線程才能進來操作ticket,
* 即使線程a出現阻塞,也不能改變。
* 4.在java中,我們通過同步機制,來解決線程的安全問題。
* 方式一:同步代碼塊
* synchronized(同步螢幕){
* 需要被同步的代碼
* }
* 說明:1.操作共享資料的代碼,即為需要被同步的代碼 ------->不能包含代碼多了,也不能包含少了。
* 2.共享資料:多個線程共同操作的變量。比如:ticket就是共享資料。
* 3.同步螢幕:俗稱:鎖。任何一個類的對象都可以充當鎖。
* 要求:多個線程必須共用同一把鎖。
* 補充:在實作Runnable接口建立多線程的方式中,我們可以考慮使用this來充當同步螢幕。
*
* 方式二:同步方法
* 如果操作共享資料的代碼完整的聲明在一個方法中,我們不妨将此方法聲明為同步的。
* 關于同步方法的總結:
* 1.同步方法任然涉及到同步螢幕,隻是不需要我們顯示的聲明。
* 2.非靜态的同步方法,同步螢幕:this
* 3.靜态的同步方法,同步螢幕:目前類本身
*
* 5.同步的方式:解決了線程的安全問題----->好處
* 操作同步代碼時,隻能有一個線程參與,其他線程等待,相當于一個單線程的過程,效率低。------>局限性
*
1 package com.atfu.java01;
2
3 /**
4 * 例子:建立三個視窗賣票,總票數為100張,使用實作Runnable接口的方式。
5 * 1.問題:賣票過程中出現了重票和錯票----->線程的安全問題
6 * 2.問題出現的原因:當某個線程操作車票的過程中,尚未操作完成時,其他線程參與進來,也操作車票。
7 * 3.如何解決:當一個線程a在操作ticket時,其他線程不能參與進來,直到線程a操作完ticket時,其他線程才能進來操作ticket,
8 *
9 * @author fu jingchao
10 * @creat 2021/10/15-17:09
11 */
12 class Window1 implements Runnable{
13 private int ticket = 100;//這裡不用加static也可以,因為後面隻造了一個Window1的對象w,三個線程共用一個對象;
14 // 如果不是共用一個對象,而是每個線程單獨造了一個對象,則要加static
15 // Object obj = new Object();
16 @Override
17 public void run() {
18 while (true){
19 synchronized(this) {// 此時的this:唯一的window1的對象 //方式二:synchronized(obj){
20 if (ticket > 0) {
21 try {
22 Thread.sleep(100);
23 } catch (InterruptedException e) {
24 e.printStackTrace();
25 }
26 System.out.println(Thread.currentThread().getName() + ":" + ticket);
27 ticket--;
28 } else {
29 break;
30 }
31 }
32 }
33 }
34 }
35
36 public class WindowTest1 {
37 public static void main(String[] args) {
38 Window1 w = new Window1();
39
40 Thread t1 = new Thread(w);
41 Thread t2 = new Thread(w);
42 Thread t3 = new Thread(w);
43
44 t1.setName("線程1");
45 t2.setName("線程2");
46 t3.setName("線程3");
47
48 t1.start();
49 t2.start();
50 t3.start();
51
52 }
53 }
此為本人學習筆記,若有錯誤,請不吝賜教