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

本文介绍了如何使用同步代码块解决Java中通过Thread类和Runnable接口创建线程时可能出现的线程不安全问题。通过在关键代码段使用`synchronized`关键字,确保了线程同步,避免了资源竞争。示例代码详细展示了在Thread和Runnable中应用同步代码块的方法。
摘要由CSDN通过智能技术生成

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 接口的线程不安全问题,本文给出了通过同步代码块解决线程同步问题的两个例子。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值