public class TestDemo2 {
public static void main(String[] args) {
Thread thread1=new myThread1();
thread1.start();
}
}
class myThread1 extends Thread {
@Override
public void run() {
System.out.println("myThread1");
}
}
public class TestDemo2 {
public static void main(String[] args) {
Runnable runnable=new myThread2();
Thread thread2=new Thread(runnable);
thread2.start();
}
}
class myThread2 implements Runnable {
@Override
public void run() {
System.out.println("myThread2");
}
}
public class TestDemo2 {
public static void main(String[] args) {
Thread thread3=new Thread(){
@Override
public void run() {
System.out.println("myThread3");
}
};
thread3.start();
}
}
public class TestDemo2 {
public static void main(String[] args) {
Thread thread4=new Thread(new Runnable() {
@Override
public void run() {
System.out.println("myThread4");
}
});
thread4.start();
}
}
public class TestDemo2 {
public static void main(String[] args) {
Thread thread5=new Thread(()->{
System.out.println("myThread5");
});
thread5.start();
}
}
使用
Thread.sleep()
方法使线程休眠,sleep() 方法可能会抛出InterruptedException
异常
public class TestDemo2 {
public static void main(String[] args) {
Thread thread5=new Thread(()->{
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
System.out.println("myThread5");
});
thread5.start();
}
}
我们会发现重写run
方法里面的sleep
方法必须要用try/catch
语句来处理异常
根据方法重写的规则,我们不能修改方法签名,因此我们就不能用throws
来声明run
方法里面的异常
众所周知,线程的是分时复用的,我们也不知道一个可执行文件会在执行哪个线程,或者已经执行到了哪里?
那我们能不能在主线程执行到一定位置或者得到想要的结果的时候中断其他线程呢?
我们以下面代码为例
public class TestDemo2 {
public static void main(String[] args) throws InterruptedException {
Thread thread5=new Thread(()->{
while (true) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
System.out.println("myThread5");
}
});
thread5.start();
}
}
我们知道上面的代码会一直执行while
循环的内容,那我们能不能让主线程在休眠10s后关闭我们新建的这个线程呢?
答案是有的,下面提供两种方法来实现
public class TestDemo2 {
static boolean key=true;
public static void main(String[] args) throws InterruptedException {
Thread thread5=new Thread(()->{
while (key) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
System.out.println("myThread5");
}
});
thread5.start();
Thread.sleep(10000);
key=false;
}
}
注意: 不能使用局部变量, lambda表达式要求局部变量的值不能修改
interrupt
方法设置标志位使用
currentThread
方法返回对当前正在执行的线程对象的引用,从而使用isInterrupted
方法判断线程是否被中断
public static void main(String[] args) throws InterruptedException {
Thread thread5=new Thread(()->{
while (!Thread.currentThread().isInterrupted()) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
System.out.println("myThread5");
}
});
thread5.start();
Thread.sleep(1000);
thread5.interrupt();
}
如果我们把throw new RuntimeException(e);
给注释了,就会发现线程又死循环了,这是为什么呢?
其实是sleep
被提前唤醒,会发生下面两件事:
InterruptedException
异常Thread
对象的isInterrupted
标志位 (因此标志位又重新设回false
,导致继续循环)因此,想让上面线程结束,我们只要加break
就行了
break
)break
)break
)线程等待顾名思义就是让一个线程等待另一个线程完成
我们来看下面的代码
public class TestDemo2 {
static int count=0;
public static void main(String[] args) throws InterruptedException {
Thread thread5=new Thread(()->{
for (int i = 0; i < 100; i++) {
count++;
}
});
thread5.start();
System.out.println(count);
}
}
相信结果大家应该都知道是0
,而不是我们想要的100
那我们要怎么得到100
呢?
可能有的人会说使用sleep
方法让主线程休眠,等到count
全部加完了再执行打印
这虽然是一个解决办法但我们不知道要休眠多久,不能适用于多种情况,因此需要我们的join
方法了
join
方法在哪个线程使用,就是哪个线程等待使用join
方法的线程
我们来看下面的示例
public class TestDemo2 {
static int count=0;
public static void main(String[] args) throws InterruptedException {
Thread thread5=new Thread(()->{
for (int i = 0; i < 100; i++) {
count++;
}
});
thread5.start();
thread5.join();
System.out.println(count);
}
}
public class TestDemo2 {
static int count=0;
public static void main(String[] args) throws InterruptedException {
Thread thread5=new Thread(()->{
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
for (int i = 0; i < 100; i++) {
count++;
}
});
thread5.start();
thread5.join(500);
System.out.println(count);
}
}
我们就会发现上面结果又变成0了