Guarded Suspension模式

Guarded Suspension模式

顾名思义保护性暂停模式,可以大大降低多线程获取锁时锁冲突带来的性能开销,当线程访问某个数据时,发现无法请求到,此时就会要求线程在进行访问。暂时挂起,保证实例的安全性,等满足条件再去请求。
比如我们如果要开发一个服务器,处理来自客户端的请求。为了不丢失请求,要维护一个缓冲区,将客户端的请求先储存至缓冲区,然后从缓冲区取出请求执行,如果缓冲区没有请求,就等待直到有新的请求过来。

public class Request {
  private String name;

    public Request(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Request{" +
                "name='" + name + '\'' +
                '}';
    }
public class RequestQueue {
    private static Integer MAX_LIMIT = 10;
    private Queue<Request> queue = new LinkedList<>();
    ReentrantLock lock = new ReentrantLock();
    Condition condition = lock.newCondition();

    public Request getRequest(){
        Request result = null;
        lock.lock();
        try {
            while (queue.isEmpty()){
                condition.await();
            }
            result = queue.poll();
            condition.signalAll();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
        return result;
    }

    public void putRequest(Request request){
        lock.lock();
        try {
            while (queue.size()>=MAX_LIMIT){
                condition.await();
            }
            queue.offer(request);
            condition.signalAll();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }

}
/**Queue的方法
1、poll() 获取并移除队列的头,如果队列为空那么返回null
2、remove() 获取并移除队列的头,如果队列为空抛出NoSuchElementException
3、peek() 获取队列的头,但是不移除此数据,如果为空返回null
4、Element() 获取队列的头但不移除,如果次队列为空,抛出NoSuchElementException
5、offer() 插入数据,成功返回true,失败返回false(有容量限制时更好)
6、add() 插入数据,(有容量限制时更好)容量满了抛出IllegalStateException
**/
//LinkedBlockingDeque是线程安全的,可以不用加锁
public class RequestQueue {
    private static Integer MAX_LIMIT = 10;
    private Queue<Request> queue = new LinkedBlockingDeque<>(MAX_LIMIT);

    public Request getRequest(){
        Request result = null;
        try {
            while (queue.isEmpty()){
                //等待
                wait();
            }
            result = queue.poll();
           
        }catch (Exception e){
            e.printStackTrace();
        }
        return result;
    }

    public void putRequest(Request request){
        try {
            while (queue.size()>=MAX_LIMIT){
                //等待
               wait();
            }
            queue.offer(request);
        }catch (Exception e){
            e.printStackTrace();
        }
    }

}
public class ServerThread extends Thread {
    private Random random;
    private RequestQueue requestQueue;

    public ServerThread(RequestQueue requestQueue,String name,long seed){
        super(name);
        this.random = new Random(seed);
        this.requestQueue =requestQueue;
    }

    @Override
    public void run() {
        for (int i = 0; i < 1000; i++) {
            Request request = requestQueue.getRequest();
            System.out.println(Thread.currentThread().getName()+" handles "+request);
            try {
                Thread.sleep(random.nextInt(1000));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
public class ClientThread extends Thread {
    private Random random;
    private RequestQueue requestQueue;

    public ClientThread(RequestQueue requestQueue,String name ,long seed){
        super(name);
        this.random = new Random(seed);
        this.requestQueue = requestQueue;
    }

    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            Request request = new Request("N0."+i);
            System.out.println(Thread.currentThread().getName()+" requests "+request.toString());
            requestQueue.putRequest(request);
            try {
                Thread.sleep(random.nextInt(1000));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值