上代码
public class ThreadTest implements Runnable {
int number = 10;
public void addHundred() throws Exception {
System.out.println("addHundred begin");
synchronized (this) {
number += 100;
System.out.println("addHundred:" + number);
}
System.out.println("addHundred end");
}
public void wait2Seconds() throws Exception {
System.out.println("wait2Seconds begin ");
synchronized (this) {
/** * (休息2S,阻塞线程) * 以验证当前线程对象的机锁被占用时, * 是否被可以访问其他同步代码块 */
System.out.println(".............wait begin..................");
this.wait(2000);
number *= 200;
System.out.println(".............wait end..................");
}
System.out.println("wait2Seconds end ");
}
public void sleep2Seconds() throws Exception {
System.out.println("sleep2Seconds begin ");
synchronized (this) {
/** * (休息2S,阻塞线程) * 以验证当前线程对象的机锁被占用时, * 是否被可以访问其他同步代码块 */
System.out.println("............sleep begin...................");
Thread.sleep(2000);
number *= 200;
System.out.println(".............sleep end..................");
}
System.out.println("sleep2Seconds end ");
}
@Override
public void run() {
try {
addHundred();
} catch (Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) throws Exception {
ThreadTest threadTest = new ThreadTest();
Thread thread = new Thread(threadTest);
thread.start();
// threadTest.sleep2Seconds();
// threadTest.wait2Seconds();
}
}
日志如下:
执行sleep2Seconds
![](https://img.haomeiwen.com/i1767630/12d528d7e68ba8f3.png)
执行wait2Seconds
![](https://img.haomeiwen.com/i1767630/8d08371cd19601f4.png)
来分析下上面代码 (sleep2Seconds()/wait2Seconds()用secondMethod()表示)
-
main()
方法中实例化ThreadTest
并启动该线程,然后调用该线程的一个方法(secondMethod()
), - 因为在主线程中调用方法,所以调用的普通方法secondMethod()会先被执行(
但并不是普通方法执行完毕该对象的线程方法才执行,普通方法执行过程中,该线程的方法也会被执行,他们是交替执行的,只是在主线程的普通方法会先被执行而已
),而secondMethod()方法代码片段中有synchronized block,因此secondMethod方法被执行后,该方法会占有该对象机锁导致该对象的线程方法一直处于阻塞状态,不能执行,直到secondeMethod释放锁;
使用Thread.sleep(2000)方法时,因为sleep在阻塞线程的同时,并持有该对象锁,所以该对象的其他同步线程(addHundred())无法执行,直到synchronized block执行完毕(sleep休眠完毕),addHundred()方法才可以执行,因此输出结果为number*200+100;
使用this.wait(2000)方法时,secondMethod()方法被执行后也锁定了该对象的机锁,执行到this.wait(2000)时,该方法会休眠2S并释当前持有的锁,此时该线程的同步方法会被执行(因为secondMethod持有的锁,已经被wait()所释放),因此输出的结果为:number+100;
总结
sleep()
- sleep()使当前线程进入停滞状态(阻塞当前线程),让出CUP的使用,目的是不让当前线程独自霸占该进程所获的CPU资源,以留一定时间给其他线程执行的机会;
- sleep()是Thread类的Static(静态)的方法;因此他不能改变对象的机锁,所以当在一个Synchronized块中调用Sleep()方法是,线程虽然休眠了,但是对象的机锁并木有被释放,其他线程无法访问这个对象(即使睡着也持有对象锁)。
- 在sleep()休眠时间期满后,该线程不一定会立即执行,这是因为其它线程可能正在运行而且没有被调度为放弃执行,除非此线程具有更高的优先级。
wait()
- wait()方法是Object类里的方法;当一个线程执行到wait()方法时,它就进入到一个和该对象相关的等待池中,同时失去(释放)了对象的机锁(暂时失去机锁,wait(long timeout)超时时间到后还需要返还对象锁);其他线程可以访问;
- wait()使用notify或者notifyAlll或者指定睡眠时间来唤醒当前等待池中的线程。
- wiat()必须放在synchronized block中,否则会在program runtime时扔出”java.lang.IllegalMonitorStateException“异常。
方法 | sleep() | wait() |
---|---|---|
类 | Thread的方法 | Object的方法 |
锁 | 不会释放锁 | 会释放锁 |
同步 | 任何地方使用 | 只能在同步控制方法或同步控制块中使用synchronized block |
用法 | 在指定的时间内使当前执行的线程睡眠(暂停执行) | 使当前线程挂起,当对象调用notify()或者notifyAll()或者时间到期,则从wait()中恢复执行 |
用途 | 用于线程状态的控制 | 用于线程间的通信 |
网友评论