多线程中join、yield、wait、sleep的区别
Thread类的方法:sleep(),yield()等
Object的方法:wait()和notify()等
由于sleep()方法是Thread 类的方法,因此它不能改变对象的机锁。所以当在一个Synchronized方法中调用sleep()时,线程虽然休眠了,但是对象的机锁没有被释放,其他线程仍然无法访问这个对象。而wait()方法则会在线程休眠的同时释放掉机锁,其他线程可以访问该对象。
Yield()方法是停止当前线程,让同等优先权的线程运行。如果没有同等优先权的线程,那么Yield() 方法将不会起作用。
一个线程结束的标志是:run()方法结束。
一个机锁被释放的标志是:synchronized块或方法结束。
1.sleep()方法(线程虽然休眠了,但是对象的机锁没有被释放,其他线程仍然无法访问这个对象)
在指定时间内让当前正在执行的线程暂停执行,但不会释放“锁标志”。不推荐使用。
sleep()使当前线程进入阻塞状态,在指定时间内不会执行。
2.wait()方法(会释放锁,让其他线程有机会获得它之前所占用的锁)
在其他线程调用对象的notify或notifyAll方法前,导致当前线程等待。线程会释放掉它所占有的“锁标志”,从而使别的线程有机会抢占该锁。
当前线程必须拥有当前对象锁。
唤醒当前对象锁的等待线程使用notify或notifyAll方法,也必须拥有相同的对象锁。waite()和notify()必须在synchronized函数或synchronized block中进行调用。
Wait()方法和notify()方法:当一个线程执行到wait()方法时,它就进入到一个和该对象相关的等待池中,同时失去了对象的机锁。当它被一个notify()方法唤醒时,等待池中的线程就被放到了锁池中。该线程从锁池中获得机锁,然后回到wait()前的中断现场。
3.yield()方法(跟sleep类似,但是它的时间是随机的)
暂停当前正在执行的线程对象。
yield()只是使当前线程重新回到可执行状态,所以执行yield()的线程有可能在进入到可执行状态后马上又被执行。
yield()只能使同优先级或更高优先级的线程有执行的机会。
调用yield方法并不会让线程进入阻塞状态,而是让线程重回就绪状态,它只需要等待重新获取CPU执行时间,这一点是和sleep方法不一样的。
4.join方法(底层用了wait方法)
等待该线程终止。
等待调用join方法的线程结束,再继续执行。
oin的意思是会等到调用改join方法的线程执行完毕之后才会执行其他线程,举例说明:
public static void main(String[] args) throws InterruptedException {
//开启一个线程
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
System.out.println(“我是新开副线程的执行”);
//这个方法会先让副线程沉睡1秒之后在执行,但是期间不会影响主线程的执行
Thread.sleep(1000L);
}
});
thread.start();//启动线程
//这个方法会等副线程执行完成之后才会接下来执行主线程,会影响主线程的执行
thread.join();
System.out.println(“我是主线程的执行”);
}