java利用Condition做超时机制

利用Condition做超时机制,主要是碰到阻塞的代码。我们希望做一个超时。当一直阻塞的时候,主线程也可以继续往下走。而当没有阻塞时,可以直接往下继续走。


我们知道Lock和Condition功能,可以做线程同步用。这里我们做一个超时机制,主要是碰到会阻塞的情况。这时候我们就新建一个线程,然后在主线程中利用Condition的await函数。而在新建的线程中,有一点要特别注意就是这个Lock的lock函数放的位置,千万不能把阻塞的代码,放在lock和unlock之间,因为这样你阻塞住了,即使主线程await时间到了,但是这个时候主线程需要重新持锁,而这个锁现在还是被新的线程持着,导致超时机制失败。


所以还是要将Lock的lock函数放在阻塞的代码之后,这样新线程如果跑完阻塞代码,可以发送notify唤醒主线程,如果新线程阻塞住,那么主线程超时时间到,重新持锁,由于我们新线程的持锁在阻塞代码之后,所以主线程可以持锁,持锁侯就执行完await函数。


还有一点要注意,新线程不能执行很快,直接在主线程还没有await,新线程就notify了。这样主线程就直接等待超时了,而其实新线程要完成的事情已经完成,浪费了时间。

    public class SendCommandRunnable implements Runnable {
        String mat = null;
        String mReply = null;
        Lock mLock;
        Condition mCondition;
        boolean mIsSafeCard;
        
        public SendCommandRunnable(String at, Lock lock, Condition condition, boolean isSafeCard) {
            mat = at;   
            mLock = lock;
            mCondition = condition;
            mIsSafeCard = isSafeCard;
        }

        public String getReply() {
            return mReply;
        }
        
        public Lock getLock() {
            return mLock;
        }
        public Condition getCondition() {
            return mCondition;
        }
        
        public void run() {
            //阻塞代码 read节点 读取数据到mReply
            mLock.lock();//持锁发那个字阻塞代码之后
            mCondition.signalAll();
            mLock.unlock();

       }
    }

    public String sendCommandSecureCardAndStick(String at) {
        at = at.trim() + "\r\n";
        Lock lock = new ReentrantLock();        
        Condition condition = lock.newCondition();
        SendCommandRunnable runnable = new SendCommandRunnable(at, lock, condition, false);
        new Thread(runnable, "thread—1").start();

        lock.lock();
        try {
            condition.await(5, TimeUnit.SECONDS);//主线程5秒等待
        } catch (InterruptedException e) {
        }
        lock.unlock();

        return runnable.getReply();//获取新线程读取节点的内容
    }


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java中,Condition是一个用于线程间通信的对象,它可以让线程在等待某个条件成立时进入等待状态,直到其他线程发出通知唤醒它。Condition通常与Lock一起使用,它提供了类似于Object.wait()和Object.notify()的功能,但是更加灵活和强大。 在使用Condition时,我们需要通过Lock对象的newCondition()方法来创建一个Condition对象,然后使用await()方法让线程进入等待状态,在条件满足时使用signal()或signalAll()方法来唤醒等待的线程。 例如,下面的代码演示了如何使用Condition来实现生产者消费者模式: ```java import java.util.LinkedList; import java.util.Queue; import java.util.concurrent.locks.Condition; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; public class ProducerConsumerExample { private static final int CAPACITY = 5; private final Queue<Integer> queue = new LinkedList<>(); private final Lock lock = new ReentrantLock(); private final Condition notFull = lock.newCondition(); private final Condition notEmpty = lock.newCondition(); public void put(int value) throws InterruptedException { lock.lock(); try { while (queue.size() == CAPACITY) { notFull.await(); } queue.add(value); notEmpty.signal(); } finally { lock.unlock(); } } public int get() throws InterruptedException { lock.lock(); try { while (queue.isEmpty()) { notEmpty.await(); } int value = queue.poll(); notFull.signal(); return value; } finally { lock.unlock(); } } } ``` 在这个例子中,我们使用了一个队列来作为缓冲区,当队列满时,生产者线程调用notFull.await()进入等待状态,直到消费者线程取走一个元素后再唤醒它;当队列空时,消费者线程调用notEmpty.await()进入等待状态,直到生产者线程放入一个元素后再唤醒它。这样就可以有效地控制生产者和消费者的速度,避免了队列溢出和下溢的情况。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值