GuardedSuspension模式

java多线程设计模式全部源码:

java多线程设计模式的全部源码

模式图

这里写图片描述

类图

类图

时序图

时序图

代码

  • Main.java
public class Main {

    public static void main(String[] args) {

        RequestQueue requestQueue = new RequestQueue();
        new ClientThread(requestQueue, "r1", 1111l).start();
        new ClientThread(requestQueue, "r2", 2222l).start();
        new ServerThread(requestQueue, "s1", 3333l).start();
        new ServerThread(requestQueue, "s2", 4444l).start();
    }

}

  • RequestQueue.java
import java.util.LinkedList;

public class RequestQueue {

    //LinkedList是非线程安全的,所以get和put一个Request对象时要加锁
    private static LinkedList<Request> mQueue = new LinkedList<Request>();

    public Request getRequest() {
        synchronized(this) {
            while(mQueue.peek() == null) {
                try {
                    wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            return mQueue.remove();
        }
    }

    public void putRequest(Request request) {
        synchronized(this) {
            mQueue.offer(request);
            notifyAll();
        }
    }
}
  • Request.java
public class Request {

    private String name;

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

    public String getString() {
        return "[ request " + name + " ]";
    }
}
  • ClientThread
import java.util.Random;

public class ClientThread extends Thread {

    private RequestQueue mRequestQueue;
    Random mRandom;
    public ClientThread(RequestQueue mRequestQueue, String name, long seed) {
        super(name);
        this.mRequestQueue = mRequestQueue;
        mRandom = new Random(seed);
    }
    @Override
    public void run() {
        for(int i = 0; i < 1000; ++i) {
            Request request = new Request(Thread.currentThread().getName() + " NO." + i);
            System.out.println(Thread.currentThread().getName() 
                    + " request " + request.getString());
            mRequestQueue.putRequest(request);

            try {
                Thread.sleep(mRandom.nextInt(1000));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

}
  • ServerThread.java
import java.util.Random;

public class ServerThread extends Thread {

    private Random mRandom;
    private RequestQueue mRequestQueue;
    public ServerThread(RequestQueue mRequestQueue, String name, long seed) {
        super(name);
        this.mRandom = new Random(seed);
        this.mRequestQueue = mRequestQueue;
    }
    @Override
    public void run() {
        for(int i = 0; i < 1000; ++i) {
            Request request = mRequestQueue.getRequest();
            System.out.println(Thread.currentThread().getName() + " handle " + request.getString());

            try {
                Thread.sleep(mRandom.nextInt(1000));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
  • 运行结果

这里写图片描述

总结

Guarded(保护) Suspension(暂停、等待),上述例子中被保护的对象是RequestQueue类里面的mQueue对象。
当mQueue里面没有元素时,想要获取mQueue里面元素的ServerThread线程会一直等待。

如果用LinkedBlockingQueue就不需要加synchronized,因为LinkedBlockingQueue是线程安全的容器。
所以RequestQue.java的代码可以改为如下形式:
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;

public class RequestQueue {

    private static BlockingQueue<Request> mQueue = new LinkedBlockingQueue<Request>();

    public Request getRequest() {
        Request request = null;
        try {
            request = mQueue.take();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return request;

    }

    public void putRequest(Request request) {
        try {
            mQueue.put(request);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值