Java线程学习之线程常见问题

Java线程学习之线程常见问题

一、线程安全问题

当多个线程同时运行时,极有可能会产生线程安全问题。

例如:三个线程同时卖10张票

public class TestThread {
    public static void main(String[] args) {
        //三个线程同时卖票
        Runnable ticket = new Ticket();
        new Thread(ticket).start();
        new Thread(ticket).start();
        new Thread(ticket).start();
    }

    static class  Ticket implements Runnable{
        //初始票数
        private int count = 10;
        @Override
        public void run() {
            while (count > 0){
                //卖票
                System.out.println("开始准备卖票");
                //睡眠1秒
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                count --;
                System.out.println("出票成功" + "余票为:" + count);
            }
        }
    }
}

执行结果如下所示,票数出现了负数,产生了线程安全问题
在这里插入图片描述

解决线程安全的方式
  1. 使用synchronized关键字的同步代码块方式

    public class TestThread {
        public static void main(String[] args) {
            //三个线程同时卖票
            Runnable ticket = new Ticket();
            new Thread(ticket).start();
            new Thread(ticket).start();
            new Thread(ticket).start();
        }
    
        static class  Ticket implements Runnable{
            //初始票数
            private int count = 10;
            Object o = new Object();
            @Override
            public void run() {
                while (true){
                    synchronized (o){
                        if(count > 0){
                            //卖票
                            System.out.println("开始准备卖票");
                            //睡眠1秒
                            try {
                                Thread.sleep(100);
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                            count --;
                            System.out.println(Thread.currentThread().getName()+"出票成功" + "余票为:" + count);
                        }else {
                            break;
                        }
                    }
    
                }
            }
    
        }
    
    

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

  2. 使用synchronized关键字的同步方法方式

    public class TestThread {
        public static void main(String[] args) {
            //三个线程同时卖票
            Runnable ticket = new Ticket();
            new Thread(ticket).start();
            new Thread(ticket).start();
            new Thread(ticket).start();
        }
    
        static class  Ticket implements Runnable{
            //初始票数
            private int count = 10;
            @Override
            public void run() {
                while (true){
                    boolean flag = sale();
                    if(!flag){
                        break;
                    }
                }
            }
    
            public synchronized boolean sale(){
                if(count > 0){
                    //卖票
                    System.out.println("开始准备卖票");
                    //睡眠1秒
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    count --;
                    System.out.println(Thread.currentThread().getName()+"出票成功," + "余票为:" + count);
                    return true;
                }
                return false;
            }
    
        }
    
    

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

  3. 使用显示锁Lock的方式

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class TestThread {
    public static void main(String[] args) {
        //三个线程同时卖票
        Runnable ticket = new Ticket();
        new Thread(ticket).start();
        new Thread(ticket).start();
        new Thread(ticket).start();
    }

    static class  Ticket implements Runnable{
        //初始票数
        private int count = 10;
        //显示锁
        Lock l = new ReentrantLock();
        @Override
        public void run() {
            while (true){
                //上锁
                l.lock();
                if(count > 0){
                    //卖票
                    System.out.println("开始准备卖票");
                    //睡眠1秒
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    count --;
                    System.out.println(Thread.currentThread().getName()+"出票成功," + "余票为:" + count);
                }else {
                    break;
                }
                //释放锁
                l.unlock();
            }
        }

        public synchronized boolean sale(){

            return false;
        }

    }
}

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

二、线程死锁问题

线程死锁描述的是这样一种情况:多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。

产生线程死锁的条件
  1. 互斥条件:该资源任意一个时刻只由一个线程占用。
  2. 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
  3. 不剥夺条件:线程已获得的资源在未使用完之前不能被其他线程强行剥夺,只有自己使用完毕后才释放资源。
  4. 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。
如何避免线程死锁
  1. 破坏互斥条件 :这个条件我们没有办法破坏,因为我们用锁本来就是想让他们互斥的(临界资源需要互斥访问)。
  2. 破坏请求与保持条件 :一次性申请所有的资源。
  3. 破坏不剥夺条件 :占用部分资源的线程进一步申请其他资源时,如果申请不到,可以主动释放它占有的资源。
  4. 破坏循环等待条件 :靠按序申请资源来预防。按某一顺序申请资源,释放资源则反序释放。破坏循环等待条件。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值