//: concurrency/SemaphoreDemo.java // Testing the Pool class import java.util.concurrent.*; import java.util.*; import static net.mindview.util.Print.*; // A task to check a resource out of a pool: class CheckoutTask implements Runnable { private static int counter = 0; private final int id = counter++; private Pool pool; public CheckoutTask(Pool pool) { this.pool = pool; } @Override public void run() { try { T item = pool.checkOut(); print(this + "checked out " + item); TimeUnit.SECONDS.sleep(1); print(this +"checking in " + item); pool.checkIn(item); } catch(InterruptedException e) { // Acceptable way to terminate } } @Override public String toString() { return "CheckoutTask " + id + " "; } } public class SemaphoreDemo { final static int SIZE = 25; public static void main(String[] args) throws Exception { final Pool pool = new Pool<>(Fat.class, SIZE); ExecutorService exec = Executors.newCachedThreadPool(); for(int i = 0; i < SIZE; i++) exec.execute(new CheckoutTask<>(pool)); print("All CheckoutTasks created"); List list = new ArrayList<>(); for(int i = 0; i < SIZE; i++) { Fat f = pool.checkOut(); printnb(i + ": main() thread checked out "); f.operation(); list.add(f); } Future blocked = exec.submit(() -> { try { // Semaphore prevents additional checkout, // so call is blocked: pool.checkOut(); } catch(InterruptedException e) { print("checkOut() Interrupted"); } }); TimeUnit.SECONDS.sleep(2); blocked.cancel(true); // Break out of blocked call print("Checking in objects in " + list); for(Fat f : list) pool.checkIn(f); for(Fat f : list) pool.checkIn(f); // Second checkIn ignored exec.shutdown(); } } /* (Execute to see output) *///:~