Java中如何创建线程?
- 继承Thread类,重写run方法。
package com.offer.interview.thread;
public class myThread extends Thread {
private String name;
public myThread(String name){
this.name=name;
}
@Override
public void run(){
for(int i=0;i<10;i++){
System.out.println("Thread start " + name + ",i=" + i);
}
}
public static void main(String[] args) {
for(int i=0;i<5;i++){
myThread mt = new myThread("Thread" +i);
mt.start();
}
}
}
- 实现Runnable接口。
如何终止一个java线程?
Java中终止线程的方式主要有三种:
1、使用stop()方法,已被弃用。原因是:stop()是立即终止,会导致一些数据被到处理一部分就会被终止,而用户并不知道哪些数据被处理,哪些没有被处理,产生了不完整的“残疾”数据,不符合完整性,所以被废弃。
2、使用volatile标志位
看一个简单的例子:
首先,实现一个Runnable接口,在其中定义volatile标志位,在run()方法中使用标志位控制程序运行。
package com.offer.interview.thread;
public class StopThread implements Runnable{
//定义退出标志,true会一直执行,false会退出循环
//使用volatile目的是保证可见性,一处修改了标志,处处都要去主存读取新的值,而不是使用缓存
public volatile boolean flag = true;
@Override
public void run() {
System.out.println("第" + Thread.currentThread().getName() + "个线程创建");
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
//退出标志生效位置
while (flag) {
}
System.out.println("第" + Thread.currentThread().getName() + "个线程终止");
}
// 在main()方法中创建线程,在合适的时候,修改标志位,终止运行中的线程。
public static void main(String[] args) throws InterruptedException {
StopThread runnable = new StopThread();
//创建3个线程
for (int i = 1; i <= 3; i++) {
Thread thread = new Thread(runnable, i + "");
thread.start();
}
//线程休眠
Thread.sleep(2000L);
System.out.println("——————————————————————————");
//修改退出标志,使线程终止
runnable.flag = false;
}
}
运行结果:
Connected to the target VM, address: '127.0.0.1:50140', transport: 'socket'
第2个线程创建
第1个线程创建
第3个线程创建
——————————————————————————
第2个线程终止
第1个线程终止
第3个线程终止
Disconnected from the target VM, address: '127.0.0.1:50140', transport: 'socket'
Process finished with exit code 0
3、使用interrupt()中断的方式,注意使用interrupt()方法中断正在运行中的线程只会修改中断状态位,可以通过isInterrupted()判断。如果使用interrupt()方法中断阻塞中的线程,那么就会抛出InterruptedException异常,可以通过catch捕获异常,然后进行处理后终止线程。有些情况,我们不能判断线程的状态,所以使用interrupt()方法时一定要慎重考虑。
第一种:正在运行中终止。
package com.offer.interview.thread;
public class StopThread1 extends Thread{
@Override
public void run() {
super.run();
try {
for(int i=0; i<1000000; i++){
if(this.interrupted()) {
System.out.println("线程已经终止, for循环不再执行");
throw new InterruptedException();
}
System.out.println("i="+(i+1));
}
System.out.println("这是for循环外面的语句,也会被执行");
} catch (InterruptedException e) {
System.out.println("StopThread1类中run方法捕获异常了。。。");
e.printStackTrace();
}
}
public static void main(String[] args) {
Thread thread = new StopThread1();
thread.start();
try {
Thread.sleep(2000);
thread.interrupt();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
运行结果:
i=204612
i=204613
i=204614
i=204615
i=204616
线程已经终止, for循环不再执行
StopThread1类中run方法捕获异常了。。。
java.lang.InterruptedException
at com.offer.interview.thread.StopThread1.run(StopThread1.java:11)
Process finished with exit code 0
第二种:阻塞状态(sleep,wait等)终止。
package com.offer.interview.thread;
public class StopThread2 extends Thread{
@Override
public void run(){
super.run();
try {
System.out.println("线程开始。。。");
for(int i=0; i<300000; i++){
System.out.println("i=" + i);
}
Thread.sleep(200000);
System.out.println("线程结束。");
} catch (InterruptedException e) {
System.out.println("在沉睡中被停止, 进入catch, 调用isInterrupted()方法的结果是:" + this.isInterrupted());
e.printStackTrace();
}
}
public static void main(String[] args) {
Thread thread = new StopThread2();
thread.start();
thread.interrupt();
}
}
运行结果:
i=299996
i=299997
i=299998
i=299999
在沉睡中被停止, 进入catch, 调用isInterrupted()方法的结果是:false
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at com.offer.interview.thread.StopThread2.run(StopThread2.java:13)
从打印的结果来看, 如果在sleep状态下停止某一线程,会进入catch语句,并且清除停止状态值,使之变为false。
前一个实验是先sleep然后再用interrupt()停止,与之相反的操作在学习过程中也要注意。
总结
对于线程的终止,个人觉得用得最多也最安全的还是属于第二种使用volatile标志位来终止线程,因为第三种的使用受到很多方面的制约,一旦没用好会出各种问题。