常见方法
sleep(long millis)
使当前正在执行的线程休眠(暂时停止执行)指定的毫秒数。时间未必精准,具体取决于系统计时器和调度程序的精度和准确性。
run()
run方法是线程的处理逻辑的方法
start()
启动线程的方法,java虚拟机会通过此方法调用run方法
notify()
调用某个对象的notify()方法能够唤醒一个正在等待这个对象的monitor的线程,如果有多个线程都在等待这个对象的monitor,则只能唤醒其中一个线程;
notifyAll()
调用notifyAll()方法能够唤醒所有正在等待这个对象的monitor的线程
wait()
使当前线程进入等待状态,需要另一个线程调用此对象的notify()或者调用notifyAll()
wait()和notify()分析
下面写个简单的demo来展示一下线程的wait()和notify()。
注意:wait()和notofy()必须在同步块中,否则将会抛出java.lang.IllegalMonitorStateException
public class MyThread extends Thread {
static int i = 0;
@Override
public void run() {
synchronized (this){
System.out.println("累加计算");
i += 100;
System.out.println("唤醒此对象等待中的线程");
this.notify();
}
}
}
class MyThreadTest {
public static void main(String[] args) throws InterruptedException {
MyThread myThread = new MyThread();
myThread.start();
// MyThreadTest的main线程取得了myThread线程的锁
// 此时myThread线程就无法运行了
synchronized (myThread) {
System.out.println("myThread线程进入等待状态");
myThread.wait();
System.out.println("DONE,运算结果: i="+ MyThread.i );
}
}
}
运行结果:
myThread线程进入等待状态
累加计算
唤醒此对象等待中的线程
DONE,运算结果: i=100
分析:
打印出来的第一句是:myThread线程进入等待状态。证明myThread线程还未执行run方法,就已经被MyThreadTest的main方法锁住了线程实例。
正常情况下myThread线程是无法运行了,但是第二句却打印出了myThread.run方法信息。
由此可以看出wait()会释放对象的锁,MyThreadTest线程锁住了myThread,wait()释放了锁,myThread线程对象获得了锁并运行计算,通过notify()正在唤醒等待状态的myThread,MyThreadTest线程重新锁住了myThread,直至执行完毕。
notify()/notifyAll()并不是真正释放锁,必须等到synchronized方法或者语法块执行完才真正释放锁
等待状态线程对象被notify()唤醒后,需要重新由cpu调度获取锁
…
sleep()和wait()的区别
将以上例子中的wait()换成sleep()
运行结果:
myThread线程进入等待状态
DONE,运算结果: i=0
累加计算
唤醒此对象等待中的线程
由此可以看出,sleep并不会释放锁,是等待MyThreadTest的同步块执行完毕只会才运行myThread的run方法