博客
关于我
java并发编程之线程协作
阅读量:380 次
发布时间:2019-03-05

本文共 3184 字,大约阅读时间需要 10 分钟。

既然多线程是为了提升效率,共同干活,那相互之间的协作也难免。

main线程起了多个线程,可以用countdownlatch来倒计数,等倒数到零时即可开始往下运行。

类似于限流中的令牌桶,semaphore可以限制线程池同时运行的线程数量。

public class SemaphoreDemo {       static Semaphore semaphore = new Semaphore(3);    public static void main(String[] args) {           ExecutorService service = Executors.newFixedThreadPool(50);        for (int i = 0; i < 1000; i++) {               service.submit(new Task());        }        service.shutdown();    }    static class Task implements Runnable {           @Override        public void run() {               try {                   semaphore.acquire();            } catch (InterruptedException e) {                   e.printStackTrace();            }            System.out.println(Thread.currentThread().getName() + "拿到了许可证,花费2秒执行慢服务");            try {                   Thread.sleep(2000);            } catch (InterruptedException e) {                   e.printStackTrace();            }            System.out.println("慢服务执行完毕," + Thread.currentThread().getName() + "释放了许可证");            semaphore.release();        }    }}

CountDownLatch 可以安排多个线程执行的顺序,多个线程等待一个线程或者一个线程等待多个线程结束后再往下执行。

但是CountDownLatch 只能使用一次,如果要多次使用可以用CyclicBarrier,而且初始化CyclicBarrier 时还可以指定栅栏打开时寻要执行的任务,如下

public class CyclicBarrierDemo {       public static void main(String[] args) {           // CyclicBarrier cyclicBarrier = new CyclicBarrier(3);        CyclicBarrier cyclicBarrier = new CyclicBarrier(3, new Runnable() {       		// 每次打开栅栏都会执行下面的任务    		@Override		    public void run() {   		         System.out.println("凑齐3人了,出发!");		    }		});        for (int i = 0; i < 6; i++) {               new Thread(new Task(i + 1, cyclicBarrier)).start();        }    }    static class Task implements Runnable {           private int id;        private CyclicBarrier cyclicBarrier;        public Task(int id, CyclicBarrier cyclicBarrier) {               this.id = id;            this.cyclicBarrier = cyclicBarrier;        }        @Override        public void run() {               System.out.println("员工" + id + "现在从大门出发,前往自行车驿站");            try {                   Thread.sleep((long) (Math.random() * 10000));                System.out.println("员工" + id + "到了自行车驿站,开始等待其他人到达");                cyclicBarrier.await();                System.out.println("员工" + id + "开始骑车");            } catch (InterruptedException e) {                   e.printStackTrace();            } catch (BrokenBarrierException e) {                   e.printStackTrace();            }        }    }}

demo 中六个线程每等待三个到来就会执行一次,CyclicBarrier 不用重复初始化就可以执行。

lock 与 condition

lock 与 condition 与C++中的有点类似了。具体的api不同而已,思路类似,都是

lock();...// 常用在 while 判断后,防止惊群condition.await();...unlock();// 通知lock();...condition.signal();...unlock();

与Syncchronized /wait/notify 类似。

lock.lock() 对应进入 synchronized 方法condition.await() 对应 object.wait()condition.signalAll() 对应 object.notifyAll()lock.unlock() 对应退出 synchronized 方法

如果说 Lock 是用来代替 synchronized 的,那么 Condition 就是用来代替相对应的 Object 的 wait/notify/notifyAll,所以在用法和性质上几乎都一样。

Condition 把 Object 的 wait/notify/notifyAll 转化为了一种相应的对象,其实现的效果基本一样,但是把更复杂的用法,变成了更直观可控的对象方法,是一种升级。
await 方法会自动释放持有的 Lock 锁,和 Object 的 wait 一样,不需要自己手动释放锁。
另外,调用 await 的时候必须持有锁,否则会抛出异常,这一点和 Object 的 wait 一样。

转载地址:http://mywwz.baihongyu.com/

你可能感兴趣的文章
Mysql学习总结(52)——最全面的MySQL 索引详解
查看>>
Mysql学习总结(53)——使用MySql开发的Java开发者规范
查看>>
Mysql学习总结(54)——MySQL 集群常用的几种高可用架构方案
查看>>
Mysql学习总结(55)——MySQL 语句大全再温习
查看>>
Mysql学习总结(56)——MySQL用户管理和权限设置
查看>>
Mysql学习总结(57)——MySQL查询当天、本周、本月、上周、本周、上月、距离当前现在6个月数据
查看>>
Mysql学习总结(58)——深入理解Mysql的四种隔离级别
查看>>
Mysql学习总结(59)——数据库分库分表策略总结
查看>>
Mysql学习总结(5)——MySql常用函数大全讲解
查看>>
Mysql学习总结(60)——并发量大、数据量大的互联网业务数据库设计规范总结
查看>>
Mysql学习总结(61)——MySQL优化之DBA级优化整理汇总
查看>>
Mysql学习总结(62)——MySQL连接com.mysql.jdbc.exceptions.jdbc4.CommunicationsException: Communications link问题
查看>>
Mysql学习总结(63)——Mysql数据库架构方案选择与分析
查看>>
Mysql学习总结(64)——Mysql配置文件my.cnf各项参数解读
查看>>
Mysql学习总结(65)——项目实战中常用SQL实践总结
查看>>
Mysql学习总结(66)——设置MYSQL数据库编码为UTF-8
查看>>
Mysql学习总结(67)——MYSQL慢查询日志
查看>>
Mysql学习总结(68)——MYSQL统计每天、每周、每月、每年数据 SQL 总结
查看>>
Mysql学习总结(69)——Mysql EXPLAIN 命令使用总结
查看>>
Mysql学习总结(6)——MySql之ALTER命令用法详细解读
查看>>