0
点赞
收藏
分享

微信扫一扫

Java 多线程线程的线程不安全问题 -- 通过同步代码块解决Java多线程的不安全问题

八怪不姓丑 2022-05-01 阅读 34

1. 前言:

Java 多线程线程的线程不安全问题 – 创建多线程的两种方式

在之前的文章中,我们简单总结了前两种Java 创建多线程的方式,并指出是线程不安全的。
之后我们给出了通过Lock 解决线程不安全的问题。本文中,继续总结,通过同步代码块的方式,解决
Thread and Runnable 的线程同步问题。

2. 通过同步代码块解决线程不安全问题:

2.1 同步代码块的使用方式:

synchronized(同步监视器){
// 需要被同步的代码
}

补充:
在实现runnable 接口创建多线程的方式中,我们可以考虑使用this充当同步监视器
在继承Thread类创建多线程的方式时,慎用this充当监视器,可考虑使用同步数据

2.2 通过同步代码块,解决Thread的线程同步问题

举例:

/**
 * 使用 synchronized方法解决 window1(继承) 线程不安全问题
 * @author JL
 * @create 2022-04-24 15:38
 */

class window3 extends Thread {

    private static int ticket = 100;
    @Override
    public void run() {
        Boolean isFlag = true;
        while (true) {
            // 因为run 是实现类,其形式由接口控制,
            // 因此我们需要引入额外synchronized方法
            // 应该不变
            if (ticket > 0) {
                show();
            } else {
                break;
            }
        }
    }

    public static synchronized void show(){// 继承,默认锁为当前类 this.class
        if (ticket > 0) {
//                    try { // 睡眠,让代码有更大机会异常
//                        Thread.sleep(100);
//                    } catch (InterruptedException e) {
//                        e.printStackTrace();
//                    }

            System.out.println(Thread.currentThread().getName() + ":卖票,票号为:" + ticket);
            ticket--;
        }
    }

}

public class window3Test {
    public static void main(String[] args) {

        window3Test win = new window3Test();
        window3 w1 = new window3();
        window3 w2 = new window3();
        window3 w3 = new window3();

        w1.setName("窗口一");
        w2.setName("窗口二");
        w3.setName("窗口三");

        w1.start();
        w2.start();
        w3.start();

    }
}

2.2 通过同步代码块,解决Runnable的线程同步问题

举例:

/**
 * 使用 同步代码块 解决 window2(实现多线程) 线程不安全问题
 * @author JL
 * @create 2022-04-24 17:06
 */
class window6 implements Runnable{


    private static int ticket = 100;//只有一个对象,也可以不加static
    @Override
    public void run() {
        while (true) {
            synchronized (this) {//使用this充当同步监视器
                if (ticket > 0) {
                    try { // 睡眠,让代码有更大机会异常
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + ":卖票,票号为:" + ticket);
                    ticket--;
                } else {
                    break;
                }
            }
        }
    }
}

public class window6Test {
    public static void main(String[] args) {
        window6Test window = new window6Test();

        window6 win = new window6();

        Thread t1 = new Thread(win);
        Thread t2 = new Thread(win);
        Thread t3 = new Thread(win);

        t1.setName("窗口1");
        t2.setName("窗口2");
        t3.setName("窗口3");

        t1.start();
        t2.start();
        t3.start();
    }
}

3. 总结

针对 继承Thread 和实现Runnable 接口的线程不安全问题,本文给出了通过同步代码块解决线程同步问题的两个例子。

举报

相关推荐

0 条评论