Java线程通讯方式

网友投稿 265 2022-11-15

Java线程通讯方式

线程间通信常用方式如下:

l  休眠唤醒方式:

Object的wait、notify、notifyAll

Condition的await、signal、signalAll

l  CountDownLatch:用于某个线程A等待若干个其他线程执行完之后,它才执行

l  CyclicBarrier:一组线程等待至某个状态之后再全部同时执行

l  Semaphore:用于控制对某组资源的访问权限

Object的wait、notify、notifyAll

public class WaitNotifyRunnable{ private Object obj = new Object(); private Integer i=0; public void odd() { while(i<10){ synchronized (obj){ if(i%2 == 1){ System.out.println("奇数:"+i); i++; obj.notify(); } else { try { obj.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } } }

public void even(){ while(i<10){ synchronized (obj){ if(i%2 == 0){ System.out.println("偶数:"+i); i++; obj.notify(); } else { try { obj.wait(); } catch (InterruptedException e) { e.printStackTrace(); }} } } } public static void main(String[] args){ final WaitNotifyRunnable runnable = new WaitNotifyRunnable(); Thread t1 = new Thread(new Runnable() { public void run() { runnable.odd(); } }, "偶数线程"); Thread t2 = new Thread(new Runnable() {public void run() { runnable.even(); } }, "奇数线程"); t1.start(); t2.start(); }}

Condition的await、signal、signalAll

import java.util.concurrent.locks.Condition;import java.util.concurrent.locks.Lock;import java.util.concurrent.locks.ReentrantLock;public class WaitNotifyRunnable{ private Lock lock = new ReentrantLock(); private Condition condition = lock.newCondition(); private Integer i=0; public void odd() { while(i<10){ lock.lock(); try{ if(i%2 == 1){ System.out.println("奇数:"+i); i++; condition.signal(); } else { condition.await(); } } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.unlock(); } } } public void even(){ while(i<10){ lock.lock(); try{ if(i%2 == 0){ System.out.println("偶数:"+i); i++; condition.signal(); } else { condition.await(); } } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.unlock(); } } } public static void main(String[] args){ final WaitNotifyRunnable runnable = new WaitNotifyRunnable(); Thread t1 = new Thread(new Runnable() { public void run() { runnable.odd(); } }, "偶数线程"); Thread t2 = new Thread(new Runnable() { public void run() { runnable.even(); } }, "奇数线程"); t1.start(); t2.start(); }}

Object和Condition休眠唤醒区别

l  object wait()必须在synchronized(同步锁)下使用,

l  object wait()必须要通过Nodify()方法进行唤醒

l  condition await() 必须和Lock(互斥锁/共享锁)配合使用

l  condition await() 必须通过 signal() 方法进行唤醒

CountDownLatch方式

CountDownLatch这个类能够使一个线程等待其他线程完成各自的工作后再执行。

CountDownLatch是通过一个计数器来实现的,计数器的初始值为线程的数量

每当一个线程完成了自己的任务后,计数器的值就会减1。当计数器值到达0时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务

import java.util.concurrent.CountDownLatch;public class CoachRecerDemo { private CountDownLatch countDownLatch=new CountDownLatch(3); //设置要等待的运动员 //运动员方法 public void racer(){ //获取线程名 String name=Thread.currentThread().getName(); System.out.println(name+"正在准备...."); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(name+"准备完毕"); countDownLatch.countDown();//-1 } //教练方法 public void coach(){ String name=Thread.currentThread().getName(); System.out.println(name+"等待运动员准备..."); try { countDownLatch.await(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("所有运动员就绪,"+name+"开始训练"); } public static void main(String[] args) { //1.创建CoachRecerDemo 实例 CoachRecerDemo coachRecerDemo=new CoachRecerDemo(); //2.创建三个线程对象 Thread thread1=new Thread(new Runnable(){ public void run() { coachRecerDemo.racer(); } },"运动员1"); Thread thread2=new Thread(new Runnable() { public void run() { coachRecerDemo.racer(); } },"运动员2"); Thread thread3=new Thread(new Runnable() { public void run() { coachRecerDemo.racer(); } },"运动员3"); //3.教练 Thread thread4=new Thread(new Runnable() { public void run() { coachRecerDemo.coach(); } },"教练"); thread4.start(); thread1.start(); thread2.start(); thread3.start(); }}

CyclicBarrier方式

CyclicBarrier是在java1.5被引入的,存在于java.util.concurrent包下。

CyclicBarrier实现让一组线程等待至某个状态之后再全部同时执行。

CyclicBarrier底层是三个线程同时启动

import java.util.Date;import java.util.concurrent.BrokenBarrierException;import java.util.concurrent.CyclicBarrier;public class ThreeThreadStartDemo { private CyclicBarrier cyclicBarrier = new CyclicBarrier(3);// 所有参与启动的线程数 public void startThread() { // 打印线程准备启动 String name = Thread.currentThread().getName(); System.out.println(name + "正在准备..."); // 2.调用CyclicBarriar的await方法等待线程准备完毕 try { cyclicBarrier.await(); } catch (InterruptedException | BrokenBarrierException e) { // TODO Auto-generated catch block e.printStackTrace(); } // 3.打印线程启动信息 System.out.println(name + "已经启动完毕" + new Date().getTime()); } public static void main(String[] args) { final ThreeThreadStartDemo threadStartDemo = new ThreeThreadStartDemo(); Thread thread1 = new Thread(new Runnable() { public void run() { threadStartDemo.startThread(); } }); Thread thread2 = new Thread(new Runnable() { public void run() { threadStartDemo.startThread(); } }); Thread thread3 = new Thread(new Runnable() { public void run() { threadStartDemo.startThread(); } }); thread1.start(); thread2.start(); thread3.start(); } }

Semaphore方式

Semaphore是在java1.5被引入的,存在于java.util.concurrent包下。

Semaphore用于控制对某组资源的访问权限。

import java.util.concurrent.Semaphore;public class WorkerMachineDemo { static class Work implements Runnable { private int workerNum; private Semaphore semaphore; public Work(int workerNum, Semaphore semaphore) { this.workerNum = workerNum; this.semaphore = semaphore; } public void run() { // 1.获取机器 // 2.打印工人获取到机器,开始工作 // 3.线程睡眠1000毫秒 // 4.使用完毕 try { semaphore.acquire(); String name = Thread.currentThread().getName(); System.out.println(name + "获取到机器,开始工作"); Thread.sleep(1000); semaphore.release(); System.out.println(name + "使用完毕,释放机器!"); } catch (Exception e) { // TODO: handle exception } } } public static void main(String[] args) { int works = 8; Semaphore semaphore = new Semaphore(3);// 机器数 for (int i = 0; i < works; i++) { new Thread(new Work(i, semaphore)).start(); } }}

sleep和wait区别:

wait和notify区别

wait和notify都是Object中的方法

wait和notify执行前线程都必须获得对象锁

wait的作用是使当前线程进行等待

notify的作用是通知其他等待当前线程的对象锁的线程

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:java如何从不规则的字符串中截取出日期
下一篇:浅谈模拟公共接口的输入-输出
相关文章

 发表评论

暂时没有评论,来抢沙发吧~