Thread类的构造方法
Thread()
class MyThread extends Thread {
@Override
public void run() {
System.out.println("hello Thread");
}
}
public class ThreadDemo {
public static void main(String[] args) {
Thread t = new MyThread();
t.start();
System.out.println("hello main");
}
}
Thread(Runnable target)
class MyRunnable implements Runnable {
@Override
public void run() {
System.out.println("hello Thread");
}
}
public class ThreadDemo {
public static void main(String[] args) {
Runnable runnable = new MyRunnable();
Thread t = new Thread(runnable);
t.start();
System.out.println("hello main");
}
}
Thread(String name),Thread(Runnable target, String name):这两种都是给线程取了个名字为了调试的时候知道哪个线程在被调度
public class ThreadDemo {
public static void main(String[] args) {
Thread t = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("hello Thread");
}
},"thread-1");
t.start();
System.out.println("hello main");
}
}
这里的thread-1就是找个线程的名字
Thread 的几个常见属性
属性 | 获取方法 |
---|---|
ID | getId() |
名称 | getName() |
状态 | getState() |
优先级 | getPriority() |
是否后台线程 | isDeamon() |
是否存活 | isAlive() |
是否被中断 | isInterrupted() |
ID:是线程的唯一标识,不同线程不会重复
名称:是各种调试工具用到
状态:表示线程当前所处的一个情况,下面我们会进一步说明
优先级:高的线程理论上来说更容易被调度到
后台线程:需要记住一点:JVM会在一个进程的所有非后台线程结束后,才会结束运行。
是否存活:即简单的理解,为 run 方法是否运行结束了
这里主要讲解几个比较重要的属性;
是否是后台线程
举个例子,你的手机同时打开了微信和QQ两个应用,当你在微信聊天时,微信就是前台线程,QQ就是后台线程;这里的后台线程我们往往也会成为守护线程
前台线程:会阻止进程结束,某一前台线程没有结束整个进程也不会结束
后台线程:不会阻止进程结束,后台线程工作没有完成,进程也可以结束
代码中手动创建的线程往往都是前台线程(比如刚才创建的t线程以及主线程main),jvm自带的线程一般默认为后台线程,我们也可以将前台线程通过setDeamon方法设置成后台线程,代码如下:
public class ThreadDemo {
public static void main(String[] args) {
Thread t = new Thread(() -> {
while (true) {
System.out.println("hello Thread");
}
});
t.start();
}
}
这个线程内的方法会一直打印hello Thread,因为创建的线程属于前台线程,不执行完整个进程不会停止,当我们用setDaemon方法把该线程设置为后台线程时进程就会停止,代码如下:
public class ThreadDemo {
public static void main(String[] args) {
Thread t = new Thread(() -> {
while (true) {
System.out.println("hello Thread");
}
});
t.setDaemon(true);
t.start();
}
}
通过结果可以看出此时线程没有执行其内部的run方法进程也可以结束。
是否存活
isAlive方法是判断此时操作系统内核是否有线程
线程的生命周期:在未调用start方法时,没有创建线程,此时isAlive方法的返回值为true,调用start方法之后真正创建了线程,此时返回值为true(但是如果线程执行完run方法之后,操作系统内核创建的PCB随之销毁,此时isAlive方法的返回值也是false);
注意:这里线程会随之销毁,但是t对象不会立刻被回收,t对象直到没有人引用时被GC回收。
举个例子:
此时执行了start方法之后创建了线程,所以调用isAlive方法之后返回值为true;
当我们在调用isAlive方法之前休眠一下会有什么结果:
此时在调用isAlive方法之前进行休眠,会先执行先创建的线程,执行完run方法之后线程也会随之销毁,所以此时的返回值为false。
线程终止
线程终止的意思是系统告诉你该线程要停止了,不是立刻停止,具体怎么执行看你的代码是如何书写的,举个例子:
假如你正在打游戏,你妈妈喊你吃饭,此时你就会有三种处理方式:
- 立刻停止打游戏去吃饭;
- 打完这一把游戏再去吃饭;
- 完全当作没听见不做任何处理;
线程也是这样的,具体怎么执行还是看你的代码如何书写。
线程终止的方式有两种:
1. 使用标志位来控制线程是否要终止;
2. 使用Thread自带的标志位来进行判定。
使用标志位来控制线程是否要终止
定义了一个flag为true,如果不使用标志位来进行终止,该进程将会一直打印不会停止,此时我们将flag改成false时,进程一定时间后就会终止;
这里自己定义的flag就是标志位
使用Thread自带的标志位来进行判定
这里我们使用的是Thread类内部的方法,调用interrupt方法之后说明系统提示线程该终止了,所以打印一段时间后会结束进程
等待一个线程
我们知道每个线程之间的执行顺序是随机(例如上面创建的新线程和主线程main,在不做任何处理的情况下,他们打印的顺序是随机的),等待一个线程控制了线程的执行顺序。
这里加入了join方法之后看到打印结果是先执行的t这个线程再执行的main这个主线程(注意调用join方法时需要抛异常)