一、线程的常用属性
属性 | 获取方法 |
ID | getId() |
名称 | getName() |
状态 | getState() |
优先级 | getPriority() |
是否为守护(后台、系统)线程 | isDaemon() |
是否存活 | isAlive() |
是否被中断 | isInterrupted() |
-
线程ID and 线程Name
代码如下:
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/03/16:14
* @Description:拿到线程的名称以及线程的ID
*/
public class ThreadDemo13 {
public static void main(String[] args) throws InterruptedException {
Runnable runnable=new Runnable() {
@Override
public void run() {
//先拿到当前线程
Thread t=Thread.currentThread();
//在拿到当前线程的一个ID
System.out.println("线程的ID"+t.getId());
//在拿到当前线程的一个名称
System.out.println("线程的Name"+t.getName());
}
};
//创建线程给分配任务以及名称
Thread thread=new Thread(runnable,"线程1");
thread.start();
//要是想要分行
Thread.sleep(500);
System.out.println();
Thread thread1=new Thread(runnable,"线程2");
thread1.start();
}
}
执行结果如下:
结论:
每一个线程的ID一定是不同的,是动态进行分配的,两个线程之间名称可以是一样的,但是线程的ID一定是不一样的。
-
线程的状态
在这里我只是简单的演示一下线程的三种状态,后续我会专门写一篇博客专门总结线程的状态的流转。
代码如下:
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/03/16:46
* @Description:线程状态的演示
*/
public class ThreadDemoByState {
public static void main(String[] args) throws InterruptedException {
Thread thread=new Thread(new Runnable() {
@Override
public void run() {
//打印线程的状态
System.out.println("线程的状态2:"+Thread.currentThread().getState());
}
});
//在启动之前先打印一下线程的状态
System.out.println("线程的状态"+ thread.getState());
thread.start();
//让线程在休眠500ms之后在观察一下
Thread.sleep(500);
System.out.println("线程的状态3"+ thread.getState());
}
}
执行结果:
-
线程的优先级
代码如下:
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/03/16:59
* @Description:获取线程的优先级
*/
public class ThreadDemoByPriority {
public static void main(String[] args) throws InterruptedException {
Thread thread=new Thread(new Runnable() {
@Override
public void run() {
System.out.println("线程的优先级1:"+Thread.currentThread().getPriority());
}
});
System.out.println("线程的优先级2:"+Thread.currentThread().getPriority());
thread.start();
System.out.println("线程的优先级3:"+thread.getPriority());
Thread.sleep(500);
System.out.println("线程的优先级4:"+thread.getPriority());
}
}
执行结果:
怎么样设置他的一个优先级?
在启动之前进行设置优先级。setPriority();
代码:
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/03/17:47
* @Description:
*/
public class ThreadDemoByPriority1 {
public static void main(String[] args) {
Thread t1=new Thread(new Runnable() {
@Override
public void run() {
//让他进行循环
Thread t=Thread.currentThread();
int priority=t.getPriority();
for (int i = 0; i < 1000; i++) {
System.out.println(t.getName()+"--优先级"+priority);
}
}
},"线程1");
t1.setPriority(Thread.MAX_PRIORITY);
Thread t2=new Thread(new Runnable() {
@Override
public void run() {
//让他进行循环
Thread t=Thread.currentThread();
int priority=t.getPriority();
for (int i = 0; i < 1000; i++) {
System.out.println(t.getName()+"--优先级"+priority);
}
}
},"线程2");
t2.setPriority(Thread.MIN_PRIORITY);
Thread t3=new Thread(new Runnable() {
@Override
public void run() {
//让他进行循环
Thread t=Thread.currentThread();
int priority=t.getPriority();
for (int i = 0; i < 1000; i++) {
System.out.println(t.getName()+"--优先级"+priority);
}
}
},"线程3");
t3.setPriority(Thread.NORM_PRIORITY);
//同时启动线程
t2.start();
t1.start();
t3.start();
}
}
执行结果:
注意事项:
同时启动多个线程,多个线程设置了不同的优先级,并不是优先级最高的就一定先执行完之后在执行低优先级的线程,而是高优先级的线程获取到CPU的时间片的概率会更高,整个的执行大致符合高优先级的线程最先执行结束。
总结:
首先我们在创建线程之后优先级就已经是存在的了,当线程执行结束的时候,优先级也是存在的,优先级是int类型的值,默认的优先级是5。通过查看源码可以知道,我们的5是处于中间位置的优先级,1是最小的优先级,10是最大的优先级。
-
是否为守护(用户)线程
线程的分类:
1.用户线程(main 默认用户线程)
2.守护线程(后台线程):守护线程是为用户线程服务的,当用户线程全部执行结束以后,守护线程也会跟随结束。JVM中的垃圾回收器就是典型的守护线程应用场景。
代码如下(判断main线程是否为守护线程):
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/05/21:20
* @Description:守护线程示例
*/
public class ThreadDemoByDaemon {
public static void main(String[] args) {
Thread t=Thread.currentThread();
System.out.println("是否是守护线程:"+t.isDaemon());
}
}
结论:
1.main线程(主线程)默认是非守护线程(用户线程);
2.在用户线程中创建的子线程也是用户线程。
3.在守护线程中创建的子线程默认情况下也是守护线程。
如何来设置守护线程?+注意事项
setDaemon()方法;传入参数true;即可以将该线程变为守护线程
注意事项:
线程的类型(用户线程 or 守护线程)不能再线程运行期间,也就是在调用了start之后进行设置,如果这样设置JVM会报错。
守护线程和用户线程的区别?
代码如下:
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/05/21:44
* @Description:对比用户线程和守护线程的区别
*/
public class ThreadDemoByDaemon2 {
public static void main(String[] args) {
//userThread();
daemonThread();
//Thread.sleep(500);
}
/**
* 守护线程
*/
private static void daemonThread() {
Thread thread=new Thread(()->{
for (int i = 0; i < 10; i++) {
System.out.println("执行:"+i);
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
thread.setDaemon(true);
thread.start();
}
/**
* 用户线程
*/
private static void userThread() {
Thread thread=new Thread(()->{
for (int i = 0; i < 10; i++) {
System.out.println("执行:"+i);
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
thread.start();
}
}
执行结果如下:
光执行守护线程:
光执行用户线程:
结论:
JVM会等待所有的用户线程全部执行结束以后才退出, 但JVM不会等待所有的守护线程执行结束以后在退出。JVM是不会等你的。但守护线程是一定会执行的。守护线程是为了辅助用户线程存在的,如果没有了用户线程,守护线程也没有执行的必要了,JVM会自行退出。
-
线程是否存活
代码如下:
package thread.threaddemo;
/**
* @Author: wenjingyuan
* @Date: 2022/11/07/20:38
* @Description:线程是否存活
*/
public class ThreadDemoByAlive {
public static void main(String[] args) throws InterruptedException {
Thread thread=new Thread(()->{
System.out.println("线程是否存活2"+Thread.currentThread().isAlive());
});
System.out.println("线程是否存活"+thread.isAlive());
thread.start();
System.out.println("线程是否存活3"+thread.isAlive());
Thread.sleep(1000);
System.out.println("线程是否存活4"+thread.isAlive());
}
}
结果如下: