线程(8)--线程间的通信、互斥锁的应用

线程间的通信

  • 1.什么时候需要通信
    • 多个线程并发执行时, 在默认情况下CPU是随机切换线程的
    • 如果我们希望他们有规律的执行, 就可以使用通信, 例如每个线程执行一次打印
  • 2.怎么通信

    • 如果希望线程等待, 就调用wait()
    • 如果希望唤醒等待的线程, 就调用notify();
    • 这两个方法必须在同步代码中执行, 并且使用同步锁对象来调用
  • 3.多个线程通信的问题

    • notify()方法是随机唤醒一个线程 用if
    • notifyAll()方法是唤醒所有线程 用while
    • JDK5之前无法唤醒指定的一个线程
    • 如果多个线程之间通信, 需要使用notifyAll()通知所有线程, 用while来反复判断条件
package com.fenqing.Thread;

public class d04_notify {

    public static void main(String[] args) {
        printer p=new printer();
        new Thread(){
            public void run(){
                for(int i=0;i<100;i++){
                    try {
                        p.print1();
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            }
        }.start();
        new Thread(){
            public void run(){
                for(int i=0;i<100;i++){
                    try {
                        p.print2();
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            }
        }.start();
    }
}
class printer{
    private int flag=1;
    public void print1() throws InterruptedException{
        synchronized(this){
            if(flag!=1){
                this.wait();
            }
            System.out.print("计");
            System.out.print("算");
            System.out.print("机");
            System.out.println();
            flag=2;
            this.notify();              //随机唤醒单个等待的线程
        }
    }
    public void print2() throws InterruptedException{
        synchronized(this){
            if(flag!=2){
                this.wait();            //用哪个对象锁,就用哪个对象调用wait();
            }
            System.out.print("软");
            System.out.print("件");
            System.out.print("工");
            System.out.print("程");
            System.out.println();
            flag=1;
            this.notify();
            //notifyAll();是唤醒所有的锁,适用于三个及三个以上的线程。
        }
    }

}

思考:
1.为什么wait方法和notify方法都定义在Object类中?
因为所对象可以使任意对象,Object类是所有类的基类
2.sleep和wait的区别:
* sleep方法需要传入参数,,参数是时间,时间到了自动醒来
* wait方法可以传入参数也可以不传入参数,传入参数就是在参数的时间结束后等待,不传入参数就是直接等待。
* sleep方法在同步函数或同步代码块中,不释放锁
* wait方法则释放锁

互斥锁

  • 1.同步
    • 使用ReentrantLock类的lock()和unlock()方法进行同步
  • 2.通信
    • 使用ReentrantLock类的newCondition()方法可以获取Condition对象
    • 需要等待的时候使用Condition的await()方法, 唤醒的时候用signal()方法
    • 不同的线程使用不同的Condition, 这样就能区分唤醒的时候找哪个线程了
package com.fenqing.Thread;

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

public class d05_reentrantLock {

    public static void main(String[] args) {
        final printer2 p=new printer2();

        new Thread(){
            public void run(){
                while(true){
                    try {
                        p.print1();
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            }
        }.start();
        new Thread(){
            public void run(){
                while(true){
                    try {
                        p.print2();
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            }
        }.start();
    }

}
class printer2{
    private ReentrantLock r=new ReentrantLock();
    private Condition c1=r.newCondition();
    private Condition c2=r.newCondition();
    private int flag=1;
    public void print1() throws InterruptedException{
        r.lock();
        if(flag!=1){
                c1.await();
            }
            System.out.print("计");
            System.out.print("算");
            System.out.print("机");
            System.out.println();
            flag=2;
            c2.signal();                
        r.unlock();
    }
    public void print2() throws InterruptedException{
        r.lock();
        if(flag!=2){
                c2.await();         
            }
            System.out.print("软");
            System.out.print("件");
            System.out.print("工");
            System.out.print("程");
            System.out.println();
            flag=1;
            c1.signal();
        r.unlock();
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值