使用synchronized实现同步

本文通过实例代码解析了Java中synchronized关键字的使用,包括同步代码块和同步方法,以及如何避免线程并发问题,确保同一时刻只有一个线程执行特定代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们先来看一段代码:

class TicketThread implements Runnable{
    private int ticket = 10;
    @Override
    public void run() {
        while(this.ticket>0){
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"正在为您服务,当前还剩 "+this.ticket--+" 张票");
        }
    }
}

public class TestDemo {
    public static void main(String[] args) {
        TicketThread ticketThread = new TicketThread();
        Thread thread1 = new Thread(ticketThread,"1号售票员");
        Thread thread2 = new Thread(ticketThread,"2号售票员");
        Thread thread3 = new Thread(ticketThread,"3号售票员");
        thread1.start();
        thread2.start();
        thread3.start();
    }
}

结果如下:
在这里插入图片描述
可以看到,当前代码出现了负数票的情况,这是因为不同步而导致的。

我们可以使用synchronized关键字来实现同步,实际上也就是加锁操作,使得同一时刻只能有一个线程获取到锁而进入同步代码块中。

使用synchronized关键字处理有两种模式:同步代码块与同步方法。接下来我们分别看一下这两种方式,就以上面的代码为例:

1.同步代码块:
此时我们只需要锁住卖票的对象就可以了,这样同一时刻只能有一个对象拿到锁并卖票

public void run() {
        for(int i = 0;i<10;i++){
            synchronized (this){
                if(this.ticket>0) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName()+"正在为您服务,当前还剩 "+this.ticket--+" 张票");
                }
            }
        }
    }

结果如下:
在这里插入图片描述
2.同步方法:

class TicketThread implements Runnable{
    private int ticket = 10;
    @Override
    public void run() {
        for(int i = 0;i<10;i++){
            this.sale();
        }
    }
    public synchronized void sale(){
        if(this.ticket>0) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"正在为您服务,当前还剩 "+this.ticket--+" 张票");
        }
    }
}

结果如下:
在这里插入图片描述
可见,当我们使用同步方法的时候,同一时刻只能有一个线程进入该方法卖票。

我们再来看一段代码:

class Sync{
    public synchronized void test(){
        System.out.println(Thread.currentThread().getName()+"Running");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName()+"End");
    }
}

class SyncThread implements Runnable{

    @Override
    public void run() {
        Sync sync = new Sync();
        sync.test();
    }

}

public class SyncTest {
    public static void main(String[] args) {
        SyncThread syncThread = new SyncThread();
        for(int i = 1;i<=3;i++){
            new Thread(syncThread,"Thread_"+i).start();
        }
    }
}

结果如下:
在这里插入图片描述
我们预期的结果应该是,同一时刻只有一个线程拿到锁进入同步方法,当这个线程释放锁从同步方法出来后,另一个线程才能进入,但此时三个线程都进入了同步方法,这是怎么回事呢?

来看这里:
在这里插入图片描述
实际上我们这里每次都创建了一个新对象,而非static同步方法以及synchronized(this)只能防止多个线程同时执行同一对象的同步代码块,即此时锁住的是当前对象;若要锁住一段代码块,则必须使用全局锁,此时锁的是类。处理方法如下:

1.锁同一个对象:

class Sync{
    public void test(){
        synchronized (this){
            System.out.println(Thread.currentThread().getName()+"Running");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"End");
        }
    }
}

class SyncThread implements Runnable{
    private Sync sync;
    public SyncThread(Sync sync){
        this.sync = sync;
    }

    @Override
    public void run() {
        this.sync.test();
    }
}

public class SyncTest {
    public static void main(String[] args) {
        Sync sync = new Sync();
        SyncThread syncThread = new SyncThread(sync);
        for(int i = 1;i<=3;i++){
            new Thread(syncThread,"Thread_"+i).start();
        }
    }
}

结果如下:
在这里插入图片描述
此时我们锁住的就是同一个对象。

2.锁类对象:

class Sync{
    public void test(){
        synchronized (Sync.class){
            System.out.println(Thread.currentThread().getName()+"Running");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"End");
        }
    }
}

class SyncThread implements Runnable{

    @Override
    public void run() {
        Sync sync = new Sync();
        sync.test();
    }

}

public class SyncTest {
    public static void main(String[] args) {
        SyncThread syncThread = new SyncThread();
        for(int i = 1;i<=3;i++){
            new Thread(syncThread,"Thread_"+i).start();
        }
    }
}

结果如下:
在这里插入图片描述
3.使用全局锁:

class Sync{
    public static synchronized void test(){
        System.out.println(Thread.currentThread().getName()+"Running");
        try {
            Thread.sleep(1000); 
        } catch (InterruptedException e) {
            e.printStackTrace(); 
        }
        System.out.println(Thread.currentThread().getName()+"End");
    }
}

结果如下:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值