JAVA同步代码块 & 同步方法

JAVA同步代码块 & 同步方法

为了解决多线程操作共享数据时产生的安全问题

例如以下代码

if (ticket <= 0) {
    // 卖完了
    break;
} else {
    ticket--;
    System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticket + "张");
}

如果两个线程同时执行以上方法,同时访问或修改共享的变量ticket,就有可能会出现安全问题。

例如:

出现重复票
窗口三在卖票,还剩下98张
窗口一在卖票,还剩下98张
窗口一在卖票,还剩下96张
窗口三在卖票,还剩下95张
窗口二在卖票,还剩下95

原因:几个线程可能同时执行System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticket + "张");

出现负数票
窗口一在卖票,还剩下0张
窗口三在卖票,还剩下-1

原因:两个线程都在执行else语句前检查了ticket是大于0的。

JAVA中,可以使用同步代码块同步方法来解决以上问题。

同步代码块

为了解决以上问题,可以使用同步代码块实现锁定多条语句,以独立操作共享数据。

格式:

synchronized(任意对象) {
    多条语句操作共享数据的代码
}

需要对象作为一个锁,只要有一个线程进去执行代码了,这个对象就会被锁住。
当线程执行完出来了,才会自动为这个对象解锁。
若对象被锁住了,线程就无法继续执行,直到这个对象被解锁。

因此,为了使只能同时有一个线程执行语句,只能是同一个对象作为synchronized的参数。

相当于一个房间只有一个门,有一个线程进去了就会锁上门,其他线程进不去。完成后再出门解锁。
若多个线程synchronized的参数是几个不同的对象,就相当于一个房间有几个门,各自都可以通过自己的门进出,起不到隔离的作用。

修改后的代码如下

public class Ticket implements Runnable {
    // 票的对象
    private int ticket = 100;
    private Object obj = new Object();

    public void run() {
        while (true) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            synchronized (obj) {
                if (ticket <= 0) {

                    // 卖完了
                    break;
                } else {

                    ticket--;
                    System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticket + "张");
                }
            }
        }
    }
}
public class demo {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();

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

        t1.setName("窗口一");
        t2.setName("窗口二");
        t3.setName("窗口三");


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

    }
}

同步方法

与同步代码块不同的是,同步方法不需要指定锁对象,而且同步方法可以锁住方法中所有代码,故同时只能有一个线程运行同步方法中的代码。

格式:

修饰符 synchronized 返回值类型 方法名(方法参数) {

}

使用示例如下:

public class Ticket implements Runnable {
    // 票的对象
    private int ticket = 100;

    public void run() {
        while (true) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            if (soldout()) {
                break;
            }
        }
    }

    public synchronized boolean soldout() {
        if (ticket <= 0) {
            // 卖完了
            return true;
        } else {
            ticket--;
            System.out.println(Thread.currentThread().getName() + "在卖票,还剩下" + ticket + "张");
        }
        return false;
    }
}

运行结果:

/Library/Java/JavaVirtualMachines/jdk-19.jdk/Contents/Home/bin/java -javaagent:/Users/doubleq666/Library/Application Support/JetBrains/Toolbox/apps/IDEA-U/ch-0/223.8617.56/IntelliJ IDEA.app/Contents/lib/idea_rt.jar=61509:/Users/doubleq666/Library/Application Support/JetBrains/Toolbox/apps/IDEA-U/ch-0/223.8617.56/IntelliJ IDEA.app/Contents/bin -Dfile.encoding=UTF-8 -Dsun.stdout.encoding=UTF-8 -Dsun.stderr.encoding=UTF-8 -classpath /Users/doubleq666/Documents/IdeaProjects/Semaphore/Semaphore/out/production/Semaphore SynchronizedDemo.demo
窗口一在卖票,还剩下99张
窗口三在卖票,还剩下98张
窗口二在卖票,还剩下97张
窗口二在卖票,还剩下96张
窗口三在卖票,还剩下95张
窗口一在卖票,还剩下94张
窗口二在卖票,还剩下93张
窗口三在卖票,还剩下92张
窗口一在卖票,还剩下91张
窗口三在卖票,还剩下90张
窗口一在卖票,还剩下89张
窗口二在卖票,还剩下88张
窗口二在卖票,还剩下87张
窗口三在卖票,还剩下86张
窗口一在卖票,还剩下85张
窗口三在卖票,还剩下84张
窗口一在卖票,还剩下83张
窗口二在卖票,还剩下82张
窗口二在卖票,还剩下81张
窗口三在卖票,还剩下80张
窗口一在卖票,还剩下79张
窗口二在卖票,还剩下78张
窗口三在卖票,还剩下77张
窗口一在卖票,还剩下76张
窗口二在卖票,还剩下75张
窗口三在卖票,还剩下74张
窗口一在卖票,还剩下73张
窗口二在卖票,还剩下72张
窗口三在卖票,还剩下71张
窗口一在卖票,还剩下70张
窗口二在卖票,还剩下69张
窗口一在卖票,还剩下68张
窗口三在卖票,还剩下67张
窗口二在卖票,还剩下66张
窗口一在卖票,还剩下65张
窗口三在卖票,还剩下64张
窗口二在卖票,还剩下63张
窗口一在卖票,还剩下62张
窗口三在卖票,还剩下61张
窗口二在卖票,还剩下60张
窗口一在卖票,还剩下59张
窗口三在卖票,还剩下58张
窗口二在卖票,还剩下57张
窗口一在卖票,还剩下56张
窗口三在卖票,还剩下55张
窗口二在卖票,还剩下54张
窗口一在卖票,还剩下53张
窗口三在卖票,还剩下52张
窗口二在卖票,还剩下51张
窗口一在卖票,还剩下50张
窗口三在卖票,还剩下49张
窗口二在卖票,还剩下48张
窗口一在卖票,还剩下47张
窗口三在卖票,还剩下46张
窗口二在卖票,还剩下45张
窗口三在卖票,还剩下44张
窗口一在卖票,还剩下43张
窗口二在卖票,还剩下42张
窗口三在卖票,还剩下41张
窗口一在卖票,还剩下40张
窗口二在卖票,还剩下39张
窗口三在卖票,还剩下38张
窗口一在卖票,还剩下37张
窗口二在卖票,还剩下36张
窗口三在卖票,还剩下35张
窗口一在卖票,还剩下34张
窗口二在卖票,还剩下33张
窗口三在卖票,还剩下32张
窗口一在卖票,还剩下31张
窗口二在卖票,还剩下30张
窗口三在卖票,还剩下29张
窗口一在卖票,还剩下28张
窗口二在卖票,还剩下27张
窗口三在卖票,还剩下26张
窗口一在卖票,还剩下25张
窗口二在卖票,还剩下24张
窗口三在卖票,还剩下23张
窗口一在卖票,还剩下22张
窗口二在卖票,还剩下21张
窗口三在卖票,还剩下20张
窗口一在卖票,还剩下19张
窗口二在卖票,还剩下18张
窗口三在卖票,还剩下17张
窗口一在卖票,还剩下16张
窗口二在卖票,还剩下15张
窗口三在卖票,还剩下14张
窗口一在卖票,还剩下13张
窗口二在卖票,还剩下12张
窗口三在卖票,还剩下11张
窗口一在卖票,还剩下10张
窗口二在卖票,还剩下9张
窗口三在卖票,还剩下8张
窗口一在卖票,还剩下7张
窗口二在卖票,还剩下6张
窗口三在卖票,还剩下5张
窗口一在卖票,还剩下4张
窗口二在卖票,还剩下3张
窗口三在卖票,还剩下2张
窗口一在卖票,还剩下1张
窗口二在卖票,还剩下0Process finished with exit code 0
  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

DoubleQ666

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值