Java多线程程序休眠、暂停与停止

休眠

  在Java多线程中,可以使用sleep()方法在指定毫秒数内让当前正在执行的线程休眠。

  下面这段代码,使得主函数的main线程休眠了2000ms,最后输出的间隔时间也是2000ms。

public class MyThread extends Thread {
public static void main(String[] args) {
try {
long begin;
long end;
begin = System.currentTimeMillis();
System.out.println("begin = " + begin);
Thread.sleep(2000);
end = System.currentTimeMillis();
System.out.println("end = " + end);
System.out.println("end - begin = " + (end - begin) + "ms");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出结果:

begin = 1486711105366

end = 1486711107366

end - begin = 2000ms

暂停

  虽然suspend和resume方法可以分别使得线程暂停和回复,但是这两个方法因为有缺点而已经被弃用。

缺点:

  • 独占:在使用suspend和resume方法时,如果使用不当,极易造成公共的同步对象独占,使得其他线程无法访问公共同步对象。
  • 不同步:在使用suspend与resume方法时也容易因为线程而导致数据不同步的情况。

yield方法

  可以使用yield方法进行暂停。

  yield()方法的作用是放弃当前的CPU资源,将它让给其他任务去占用CPU执行时间。但放弃的时间不确定,有可能刚刚放弃,马上又获得CPU时间片。

public class MyThread extends Thread {
@Override
public void run() {
long beginTime = System.currentTimeMillis();
int count = 0;
for (int i = 0; i < 50000000; i++) {
//Thread.yield();
count = count + (i + 1);
}
long endTime = System.currentTimeMillis();
System.out.println("用时: " + (endTime - beginTime) + "ms");
} public static void main(String[] args) {
MyThread myThread = new MyThread();
myThread.start();
}
}

输出结果:

用时: 21ms

  去掉注释//Thread.yield()后。

输出结果:

用时: 4471ms

  从第二次输出可以看出,用时明显变长。

停止

  虽然stop()可以停止一个线程,但是这个方法是不安全的,而且是已经被弃用作废的,最好不要使用它。

interrupt()方法

  interrupt()方法的使用效果并不像for+break语句那个,马上就停止循环。调用interrupt()方法仅仅是在当前线程中打了一个停止的标记,并不是真正停止线程。

判断线程是否是停止状态

  • this.interrupted():测试当前线程是否已经是中断状态,执行后具有将状态标志清楚为false的功能,为static方法。
  • this.isInterrupted():测试线程Thread对象是否已经是中断状态,但是不清除状态标志。
public class MyThread extends Thread {
public static void main(String[] args) {
Thread.currentThread().interrupt();
System.out.println("是否停止1? " + Thread.interrupted());
System.out.println("是否停止2? " + Thread.interrupted());
System.out.println("end");
}
}

输出结果:

是否停止1? true

是否停止2? false

end

public class MyThread extends Thread {
@Override
public void run() {
for (int i = 0; i < 50000; i++) {
System.out.println(i);
}
} public static void main(String[] args) {
try {
MyThread myThread = new MyThread();
myThread.start();
Thread.sleep(1000);
myThread.interrupt();
System.out.println("是否停止1? " + myThread.isInterrupted());
System.out.println("是否停止2? " + myThread.isInterrupted());
System.out.println("end");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出结果(末尾):

49997

49998

49999

是否停止1? false

是否停止2? false

end

能停止线程的方法

  在run中加一个判断,如果停止了,则break跳出循环体。

public class MyThread extends Thread {
@Override
public void run() {
for (int i = 0; i < 50000000; i++) {
if (this.interrupted()) {
System.out.println("已经是停止状态了,我要退出了");
break;
}
System.out.println(i);
}
} public static void main(String[] args) {
try {
MyThread myThread = new MyThread();
myThread.start();
Thread.sleep(1000);
myThread.interrupt();
System.out.println("end");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出结果(末尾):

160560

160561

160562

160563

160564

end

已经是停止状态了,我要退出了

  上述代码虽然可以停止线程,但是如果for的后面还有语句,那么还是会继续执行。

  因此可以用下述方法来解决。

public class MyThread extends Thread {
@Override
public void run() {
try {
for (int i = 0; i < 50000000; i++) {
if (this.interrupted()) {
System.out.println("已经是停止状态了,我要退出了");
throw new InterruptedException();
}
System.out.println(i);
}
System.out.println("for结束");
} catch (InterruptedException e) {
System.out.println("进入run中的catch了");
e.printStackTrace();
} } public static void main(String[] args) {
try {
MyThread myThread = new MyThread();
myThread.start();
Thread.sleep(1000);
myThread.interrupt();
System.out.println("end");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出结果(末尾):

152891

152892

152893

152894

end

已经是停止状态了,我要退出了

进入run中的catch了

java.lang.InterruptedException

at mythread.MyThread.run(MyThread.java:13)

上一篇:[BZOJ5462][APIO2018]新家(线段树+堆)


下一篇:Javasocket1