java多线程05:线程通信

Posted 我想和这个世界谈谈,

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java多线程05:线程通信相关的知识,希望对你有一定的参考价值。

1、为什么要线程通信

多个线程并发执行时,在默认情况下CPU是随机切换线程的,有时我们希望CPU按照我们的规律执行线程,此时就需要线程之间协调通信。

2、线程通讯方式

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

l  休眠唤醒方式:Object的wait、notify、notifyAll  和  Condition的await、signal、signalAll

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

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

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

2.1 休眠唤醒方式

object的wait、notify、notifyAll

package com.mall.thread;

/**
 * @Auther: mengyang
 * @Date: 2020/1/21 0021 15:05
 * @Description:使用Object类的通信
 * @statement:
 */
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(Thread.currentThread().getName()+"【奇数】:"+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(Thread.currentThread().getName()+"【偶数】:"+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 thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                runnable.odd();
            }
        });
        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                runnable.even();
            }
        });

        thread1.start();
        thread2.start();
    }
}

运行效果:

 

Condition的await、signal、signalAll 

package com.mall.thread;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @Auther: mengyang
 * @Date: 2020/1/21 0021 15:46
 * @Description:使用Condition的await、signal
 * @statement:
 */
public class AwaitSignalRunnable {

    private Lock lock = new ReentrantLock();
    private Condition condition = lock.newCondition();
    private Integer i = 0;


    //奇数
    private void odd(){
        while (i<10){
            lock.lock();
            try{
                if(i%2 == 1){
                    System.out.println(Thread.currentThread().getName()+"【奇数】:"+i);
                    i++;
                    condition.signal();
                }else{
                    condition.await();
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }finally {
                lock.unlock();
            }
        }
    }

    //偶数
    private void even(){
        while (i<10){
            lock.lock();
            try{
                if(i%2 == 0){
                    System.out.println(Thread.currentThread().getName()+"【偶数】:"+i);
                    i++;
                    condition.signal();
                }else{
                    condition.await();
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }finally {
                lock.unlock();
            }
        }
    }

    public static void main(String[] args) {
        final AwaitSignalRunnable runnable = new AwaitSignalRunnable();
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                runnable.odd();
            }
        }, "奇数线程");
        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                runnable.even();
            }
        }, "偶数线程");

        thread1.start();
        thread2.start();
    }
}

  运行结果:

 

 Object和condition休眠唤醒的区别

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

l  Object wait()必须要通过notify() 方法唤醒

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

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

2.2 CountDownLatch方式

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

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

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

 

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

示例代码:

package com.mall.thread;

import java.util.concurrent.CountDownLatch;

/**
 * @Auther: mengyang
 * @Date: 2019/1/21 0021 16:46
 * @Description:使用CountDownLatch
 * 每当一个线程完成了自己的任务后,计数器的值就会减1。当计数器值到达0时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。
 * @statement:
 */
public class CountDown {

    private Integer i = 0;
    private CountDownLatch countDownLatch = new CountDownLatch(1);
    //奇数
    public void odd(){
        while(i <10){
            if(i%2 == 1){
                System.out.println(Thread.currentThread().getName()+"【奇数】:"+i);
                i++;
                countDownLatch.countDown();
            } else {
                try {
                    countDownLatch.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
    //偶数
    public void even(){
        while(i <10){
            if(i%2 == 0){
                System.out.println(Thread.currentThread().getName()+"【偶数】:"+i);
                i++;
                countDownLatch.countDown();
            } else {
                try {
                    countDownLatch.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    //测试
    public static void main(String[] args){
        final CountDown countDown = new CountDown();
        Thread t1 = new Thread(new Runnable() {
            public void run() {
                countDown.odd();
            }
        },"奇数");
        Thread t2 = new Thread(new Runnable() {
            public void run() {
                countDown.even();
            }
        },"偶数");
        t1.start();
        t2.start();
    }

}

执行效果:

 

 2.3 CyclicBarrier方式

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

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

CyclicBarrier底层是

三个线程同时启动,示例代码如下:

 

package com.mall.thread;

import java.util.Date;
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;

/**
 * @Auther: mengyang
 * @Date: 2020/1/21 0021 09:52
 * @Description:使用CyclicBarrier
 * CyclicBarrier实现让一组线程等待至某个状态之后再全部同时执行。
 * @statement:
 */
public class CyclicBarrierDemo {
    public static void main(String[] args){

        final CyclicBarrier cyclicBarrier = new CyclicBarrier(3);

        new Thread(new Runnable() {
            public void run() {
                System.out.println(Thread.currentThread().getName()+":准备...");
                try {
                    cyclicBarrier.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } catch (BrokenBarrierException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"启动完毕:"+new Date().getTime());
            }
        },"线程1").start();
        new Thread(new Runnable() {
            public void run() {
                System.out.println(Thread.currentThread().getName()+":准备...");
                try {
                    cyclicBarrier.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } catch (BrokenBarrierException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"启动完毕:"+new Date().getTime());
            }
        },"线程2").start();
        new Thread(new Runnable() {
            public void run() {
                System.out.println(Thread.currentThread().getName()+":准备...");
                try {
                    cyclicBarrier.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } catch (BrokenBarrierException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"启动完毕:"+new Date().getTime());
            }
        },"线程3").start();
    }

}

执行效果如下:三个线程同时启动

2.4.   Semaphore方式

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

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

工人使用机器工作,示例代码如下:

package com.mall.thread;

import java.util.concurrent.Semaphore;

/**
 * @Auther: mengyang
 * @Date: 2019/1/21 0021 16:56
 * @Description: 使用Semaphore
 * emaphore用于控制对某组资源的访问权限
 * @statement:
 */
public class SemaphoreDemo {
    static class Machine implements Runnable{

        private int num;//工号
        private Semaphore semaphore;

        public Machine(int num, Semaphore semaphore) {
            this.num = num;
            this.semaphore = semaphore;
        }

        public void run() {
            try {
                semaphore.acquire();//请求机器
                System.out.println("工人"+this.num+"请求机器,正在使用机器");
                Thread.sleep(1000);
                System.out.println("工人"+this.num+"使用完毕,已经释放机器");
                semaphore.release();//释放机器
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args){
        int worker = 8;//工人数
        Semaphore semaphore = new Semaphore(3);//机器数
        for (int i=0; i< worker; i++){
            new Thread(new Machine(i, semaphore)).start();
        }
    }

}

执行效果如下:

 

 3、小结

sleep和wait区别

wait和notify区别

wait和notify都是Object中的方法

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

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

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

以上是关于java多线程05:线程通信的主要内容,如果未能解决你的问题,请参考以下文章

Java多线程:线程间通信方式

Java多线程:线程间通信方式

Java:多线程

阶段1 语言基础+高级_1-3-Java语言高级_05-异常与多线程_第4节 等待唤醒机制_5_线程间通信

多个用户访问同一段代码

Java 多线程线程间的通信