Java语言之线程安全、同步机制

本文深入探讨了多线程程序中的线程安全问题,解释了其产生原因,并详细介绍了三种解决方法:同步代码块、同步方法及Lock锁机制,确保多线程环境下资源访问的正确性和效率。

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

一,线程安全问题

单线程程序不会出现线程安全问题;多线程程序,没有访问共享数据,不会产生问题;多线程访问共享数据,会产生线程安全问题。
线程安全问题产生原因:线程抢夺CPU的执行权,谁抢到谁执行。

注意:
线程安全问题是不能产生的,我们可让一个线程在访问共享数据时无论是否失去CPU执行权,都让其他线程只能等待,等待当前线程买完票,其他线程再进行卖票,以保证当前只有一个线程在卖票。

public class RunnableImpl implements Runnable {
    private int ticket = 100;

    @Override
    public void run() {
       while (ticket <= 100){
           if(ticket > 0){
               System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"号票");
               ticket--;
           }
       }
    }
}
public class DemoRunnableImpl {
    public static void main(String[] args) {
        RunnableImpl rt = new RunnableImpl();
        Thread t = new Thread(rt);
        Thread t2 = new Thread(rt);
        Thread t1 = new Thread(rt);
        t.start();
        t1.start();
        t2.start();
        /*
        Thread-0正在卖100号票
        Thread-2正在卖100号票
        Thread-0正在卖99号票
        Thread-2正在卖98号票
        Thread-0正在卖97号票
        Thread-2正在卖96号票
        Thread-0正在卖95号票
        ...
        Thread-2正在卖0号票
        Thread-0正在卖-1号票
         */
    }
}

解决方法:为保证每个线程都能正常执行原子操作,java引入了线程同步机制。

同步操作方式
1,同步代码块;
2,同步方法;
3,锁机制。

二,同步代码块

格式
synchronized(锁对象){
访问了共享数据的代码;
}
注意
1,代码块中的锁对象,可使用任意的对象;
2,必须保证多个线程使用的锁对象,必须是同一个;
3,锁对象作用:把同步代码块锁住,只让一个线程在同步代码块中执行。
同步技术原理:使用一个锁对象,这个锁对象叫做同步锁,也叫对象锁,也叫对象监视器。同步中的线程,没有执行完毕,不会释放锁对象,同步外的线程没有锁进不去同步代码块。

public class SynchronizeWay implements Runnable {
    private int ticket = 100;
    Object obj = new Object();

    @Override
    public void run() {
        while (true) {
            synchronized (obj){
                if (ticket > 0) {
                    System.out.println(Thread.currentThread().getName() + "正在卖" + ticket + "号票");
                    ticket--;
                }
            }

        }
    }
}

public class DemoSynchronize {
    public static void main(String[] args) {
        SynchronizeWay rt = new SynchronizeWay();
        Thread t = new Thread(rt);
        Thread t2 = new Thread(rt);
        Thread t1 = new Thread(rt);
        t.start();
        t1.start();
        t2.start();
        /*
        Thread-0正在卖100号票
        Thread-0正在卖99号票
        Thread-0正在卖98号票
         */
    }
}
三,使用同步方法

步骤
1,把访问了共享数据的代码抽取出来,放到一个方法中;
2,在方法上添加synchronized修饰符。

格式:定义方法的格式。
修饰符 synchronized 返回值类型 方法名(参数列表){
访问了共享数据的代码;
}

原理:同步方法中,会把方法内部的代码锁住,只让一个线程执行。同步方法的锁对象是实现类对象,即this。若是静态同步方法,则锁对象是本类的class属性(class文件对象,即反射)。

public class SynchronizeWay implements Runnable {
    private int ticket = 100;
    Object obj = new Object();

    @Override
    public void run() {
        while (true) {
            ticketSale();
        }
    }

    public synchronized void ticketSale(){
        if (ticket > 0){
            System.out.println(Thread.currentThread().getName()+"正在卖"+ticket+"号票");
            ticket--;
        }
    }
}
四,使用Lock锁

java.util.concurrent.locks.Lock接口,java.util.concurrent.locks.ReentrantLock implements Lock接口。

Lock接口中的方法
void lock():获取锁;
void unlock():释放锁。

public class ThreadLock implements Runnable {
    private int tickets = 100;
    Lock l = new ReentrantLock();

    @Override
    public void run() {
        while (true){
            l.lock();
            if(tickets > 0){
                try {
                    Thread.sleep(100);
                    System.out.println(Thread.currentThread().getName()+"正在卖"+tickets+"号票");
                    tickets--;
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }finally {
                    l.unlock();
                }

            }

        }
    }
}

public class DemoLock {
    public static void main(String[] args) {
        ThreadLock th = new ThreadLock();
        Thread t0 = new Thread(th);
        Thread t1 = new Thread(th);
        Thread t2 = new Thread(th);
        t0.start();
        t1.start();
        t2.start();
        /*
        Thread-0正在卖100号票
        Thread-0正在卖99号票
        Thread-0正在卖98号票
        Thread-2正在卖97号票
        Thread-1正在卖96号票
         */
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值