• Object.wait()与Object.notify()的用法


    object.wait()和object.notify()和object.notifyall()

    正文

    wait、notify和notifyAll方法是Object类的final native方法。所以这些方法不能被子类重写,Object类是所有类的超类,因此在程序中有以下三种形式调用wait等方法。

    wait();//方式1:
    this.wait();//方式2:
    super.wait();//方式3

    void notifyAll()

    解除所有那些在该对象上调用wait方法的线程的阻塞状态。该方法只能在同步方法同步块内部调用。如果当前线程不是锁的持有者,该方法抛出一个IllegalMonitorStateException异常。

    void notify()

    随机选择一个在该对象上调用wait方法的线程,解除其阻塞状态。该方法只能在同步方法同步块内部调用。如果当前线程不是锁的持有者,该方法抛出一个IllegalMonitorStateException异常。

    void wait()

    导致线程进入等待状态,直到它被其他线程通过notify()或者notifyAll唤醒。该方法只能在同步方法中调用。如果当前线程不是锁的持有者,该方法抛出一个IllegalMonitorStateException异常。

    void wait(long millis)和void wait(long millis,int nanos)

    导致线程进入等待状态直到它被通知或者经过指定的时间。这些方法只能在同步方法中调用。如果当前线程不是锁的持有者,该方法抛出一个IllegalMonitorStateException异常。

    Object.wait()和Object.notify()和Object.notifyall()必须写在synchronized方法内部或者synchronized块内部,这是因为:这几个方法要求当前正在运行object.wait()方法的线程拥有object的对象锁。即使你确实知道当前上下文线程确实拥有了对象锁,也不能将object.wait()这样的语句写在当前上下文中。如: 

    View Code

    程序运行会报错,运行结果如下:

    ThreadID:1, ThreadName:main
    Exception in thread "main" java.lang.IllegalMonitorStateException
        at java.lang.Object.wait(Native Method)
        at java.lang.Object.wait(Object.java:485)
        at edu.sjtu.erplab.ObjectTest.ObjectWaitTest.main(ObjectWaitTest.java:24)

    正确的写法应该是

    复制代码
    package edu.sjtu.erplab.ObjectTest;
    
    class A
    {
        public synchronized void printThreadInfo() throws InterruptedException
        {
            Thread t=Thread.currentThread();
            System.out.println("ThreadID:"+t.getId()+", ThreadName:"+t.getName());
    //        this.wait();//一直等待
            this.wait(1000);//等待1000ms
    //        super.wait(1000);
        }
    }
    
    
    
    public class ObjectWaitTest {
        public static void main(String args[])
        {
            A a=new A();
            //因为printThreadInfo()方法抛出InterruptedException异常,所以这里必须使用try-catch块
            try {
                a.printThreadInfo();
                //a.wait();
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            
            Thread t=Thread.currentThread();
            System.out.println("ThreadID:"+t.getId()+", ThreadName:"+t.getName());
        }
    }
    复制代码

    具体的可以参考多线程开发中提到的消费者与生产者案例的最后一个代码示例。

  • 相关阅读:
    C++对象内存布局③测试多继承中派生类的虚函数在哪一张虚函数表中
    C++对象内存布局⑨VS编译器虚拟继承菱形继承
    C++对象内存布局⑥GCC编译器虚拟继承的虚基类表可能有两个
    C++对象内存布局⑦VS编译器虚拟继承多个基类
    C++对象内存布局⑧GCC编译器虚拟继承多个基类
    C++对象内存布局②测试派生类跟基类的虚函数表
    C++对象内存布局④VS编译器单个虚拟继承
    关于gridview绑定数据为空时的界面设计
    防止浏览器不小心被关闭的方法
    javascript 继承之原型链
  • 原文地址:https://www.cnblogs.com/kabi/p/5242108.html
Copyright © 2020-2023  润新知