关于多线程学习遇到的一些问题

一、使用同步锁时出现的问题

代码如下,
问题:出现了不能够同步的情况,不能够实现一个线程输出完30个数字之后再输出第二个线程,
线程交替输出。

原因:同步锁声明位置有误,不能在run方法中声明,要在线程还没有开始运行时定义好同步锁。

public class Test {

    public static void main(String[] args) {
        MyTh3 t=new MyTh3();
//      Lock lock=new ReentrantLock();

        Thread th1=new Thread(t);
        th1.setName("线程1");
        Thread th2=new Thread(t);
        th2.setName("线程2");
        th1.start();
        th2.start();


    }

}
class MyTh3 implements Runnable{

    Lock lock=new ReentrantLock();//在这里声明才是正确的

    @Override
    public void run() {
        //声明一个lock对象
//      Lock lock=new ReentrantLock();     此时的锁相当于各线程共享的数据,会导致后面不能正确锁住要执行的代码
//要讲锁声明提到run方法前     
        try {
             lock.lock();  //加锁    cpu控制权在下面遇到解锁之前的代码运行结束之前都不会交给其他线程    
                for (int i = 0; i <30; i++) {
                    System.out.println(Thread.currentThread().getName()+" 输出"+i);
                }   
        } catch (Exception e) {
        e.printStackTrace();
        }finally{
            lock.unlock();//解锁
        }
    }

}

二、买票的经典问题

代码如下
问题:出现有的窗口卖出0号-1号票
原因:多个线程通过while验证时满足条件,进入while循环,在同步锁之前阻塞,但此时的票数为1,三个线程依次进入同步区域,执行卖票操作,导致票数1—>0—–>-1;
解决方案:进入同步区域后再次进行票数的验证。

public class Ticket implements Runnable{


    public static void main(String[] args) {

        Ticket t=new Ticket();

        Thread th1=new Thread(t);
        th1.setName("窗口1");
        Thread th2=new Thread(t);
        th2.setName("窗口2");
        Thread th3=new Thread(t);
        th3.setName("窗口3");

        th1.start();
        th2.start();
        th3.start();
    }
    int ticket=10;
    Lock lock=new ReentrantLock();
    @Override
    public void run() {

        while(ticket>0){
                 lock.lock();  //加锁    锁共享资源

                //再做一次判断,因为有可能线程绕过了while(ticket>0)这个判断后阻塞
                if(ticket>0){
                System.out.println(Thread.currentThread().getName()+"卖了第"+ticket+"号票");
                ticket--;
                }
                lock.unlock();//解锁

            try {
                //卖完一张票休眠100毫秒
                Thread.sleep(100);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }

    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值