Java Thread.currentThread().getName() 和 this.getName()区别详解
作者:经典鸡翅 发布时间:2021-10-31 01:46:18
currentThread的详解
currentThread方法是Thread类的一个静态方法,用来获取当前运行的代码段,正在被哪个线程调用。我们先来看一眼源码。
是一个native方法。直接与系统层面做交互。
下面看一段代码
public static void main(String[] args) {
String name = Thread.currentThread().getName();
System.out.println(name);
}
输出的结果为main。
为什么为main呢?
java的项目在启动的时候,会创立一个进程,这个进程同样也是一个线程,在java里面他就叫做main线程。他的名字在设定的时候就是main。我们可以看到上面的代码就是在main方法下执行的,也就是由main线程来执行,所以我们打印出来的名字是main。
创建一个新的线程
来看下面的一行代码,我们创建一个新的线程,并且在线程创建的构造方法和其实际执行的run方法内,增加打印出当前执行线程的名称。
public class TestThread extends Thread {
public TestThread() {
System.out.println("构造方法:" + Thread.currentThread().getName());
}
@Override
public void run() {
System.out.println("run方法:" + Thread.currentThread().getName());
}
public static void main(String[] args) {
TestThread testThread = new TestThread();
testThread.start();
}
}
我们运行一下,看到结果
构造方法:main
run方法:Thread-0
如何理解上述的情况呢?
我们先来看构造方法这一行。
TestThread testThread = new TestThread();
这里我们只是声明了一个线程对象类,这个新的线程没有创建也没有启动,我们仅仅把它理解为一个普通对象即可。那么由于是在main方法里面,那么执行他的一定是main线程,所以可以看到构造方法输出的结果是main。
再来看start方法里面,为什么变成了thread-0了呢。
我们知道java的多线程创建的一种方式就是继承thread类。然后实现里面的run方法。这样当线程start的时候,就会调用内部的start0的本地方法,实际就是会执行run的实现方法。当run方法执行的时候,一定是由我们创建的线程去执行的,而不是main线程,所以我们就可以得知打印出来的是新线程的名字thread0。
为什么新线程的名字为thread0呢?
我们是采用继承thread类的形式来创建线程的,那么在我们构造的时候,一定会执行父类的构造方法。我们是一个空参构造,那么我们就要看下父类的空参构造是什么样的。看一下源码
可以看到,里面有一个name参数,已经给我们拼接了一个thread-后面是加了一个变量,我们继续看看这个变量是什么
一个同步的静态方法,再来看看变量的值。
一个静态的int型变量,由此我们知道int的初始值就是0。我们先获取到了0然后又执行了++。其他线程再次初始化的时候就是从1开始了。因为加了同步锁的关键字,我们不用害怕数据会乱掉。
所以我们就知道了新线程的名字为thread-0的来源。
再看一种带this的情况
public class TestThread extends Thread {
public TestThread() {
System.out.println("构造方法开始!");
System.out.println("构造方法:" + Thread.currentThread().getName());
System.out.println("构造方法的this名称:" + this.getName());
System.out.println("构造方法结束!");
}
@Override
public void run() {
System.out.println("run方法开始!");
System.out.println("run方法:" + Thread.currentThread().getName());
System.out.println("run方法的this名称:" + this.getName());
System.out.println("run方法结束!");
}
public static void main(String[] args) {
TestThread testThread = new TestThread();
testThread.start();
}
}
执行的结果如下:
构造方法开始!
构造方法:main
构造方法的this名称:Thread-0
构造方法结束!
run方法开始!
run方法:Thread-0
run方法的this名称:Thread-0
run方法结束!
我们发现构造的this名称 为thread-0。这个不难理解,this指的是当前对象的名称。因为我们的线程在初始化的时候有了name,所以是thread-0。
创建的线程setName的情况
public class TestThread extends Thread {
public TestThread() {
System.out.println("构造方法开始!");
System.out.println("构造方法:" + Thread.currentThread().getName());
System.out.println("构造方法的this名称:" + this.getName());
System.out.println("构造方法结束!");
}
@Override
public void run() {
System.out.println("run方法开始!");
System.out.println("run方法:" + Thread.currentThread().getName());
System.out.println("run方法的this名称:" + this.getName());
System.out.println("run方法结束!");
}
public static void main(String[] args) {
TestThread testThread = new TestThread();
testThread.setName("test");
testThread.start();
}
}
看一下执行的结果
构造方法开始!
构造方法:main
构造方法的this名称:Thread-0
构造方法结束!
run方法开始!
run方法:test
run方法的this名称:test
run方法结束!
这里有人可能就要问了,为啥构造方法的this名称为thread-0。实际run的时候就变为test了呢。
这是因为我们在构造对象的时候,对象还不完整,没有执行到setName这一步。所以二者不一致。
最后看一段复杂的代码
public class TestThread extends Thread {
public TestThread() {
System.out.println("构造方法开始!");
System.out.println("构造方法:" + Thread.currentThread().getName());
System.out.println("构造方法的this名称:" + this.getName());
System.out.println("构造方法结束!");
}
@Override
public void run() {
System.out.println("run方法开始!");
System.out.println("run方法:" + Thread.currentThread().getName());
System.out.println("run方法的this名称:" + this.getName());
System.out.println("run方法结束!");
}
public static void main(String[] args) {
TestThread testThread = new TestThread();
Thread thread = new Thread(testThread);
System.out.println("新线程thread的名称:" + thread.getName());
thread.setName("test");
thread.start();
}
}
可以看到如下的结果
构造方法开始!
构造方法:main
构造方法的this名称:Thread-0
构造方法结束!
新线程thread的名称:Thread-1
run方法开始!
run方法:test
run方法的this名称:Thread-0
run方法结束!
我们根据结果来看,构造方法的打印,和上面解释的如出一辙。我们重点关注一下新线程的打印,这里为什么是1呢,因为我们上面初始化0之后已经++所以为1。
run方法为test是因为我们实际执行的线程Thread thread = new Thread(testThread);所以我们获取到的名称是thread的名称为test。
那么run方法的this名称为什么是thread-0呢。因为this.getName获取到的是当前对象的名称。我们的当前对象是TestThread。他的线程名称从未被改变过,所以我们拿到了thread-0。
来源:https://www.cnblogs.com/jichi/p/15916171.html
猜你喜欢
- [LeetCode] 3. Longest Substring Without Repeating Characters 最长无重复字符的子
- 实现多线程的方式:实现多线程的方式有多种,这里只列举两种常用的,而第一种继承Thread的方式无法实现多窗口卖票。一,继承Thread方式:
- 全面解析java注解Java中的常见注解 a.JDK中的注解 @Override 覆盖父类或者父接口的方
- 概述非对称加密算法与对称加密算法的主要差别在于非对称加密算法用于加密和解密的密钥不相同,非对称加密算法密钥分为公钥和私钥,公钥加密只能用私钥
- 该项目主要实现mybatisplus、多数据源、lombok、druid的集成主要参考 https://mp.baomidou.com/gu
- 背景在接口请求过程中,传递json对象,springboot转换为实体VO对象后,所有属性都为null。post请求:后台接收请求:当时就懵
- 本文实例讲述了Android中断线程的处理方法。分享给大家供大家参考。具体方法如下:我现在对一个用户注册的功能1.用ProgressDial
- 表单提交这个方法是挺方便的,但在java来说就显得有些麻烦了,怎么个麻烦呢,就是当你字段多的时候,你就得一个一个的获取其对应的值,这样代码量
- 一、Rxjava使用场景为了模拟实际场景,从wanandroid网站找了二个接口,如下:(对Wanandroid表示感谢!)public i
- 如果没有安装过maven,是用的idea自带的maven,那就是idea的安装目录下 /plugins/maven/lib/maven3这个
- 静态代理: 由我们开发者自己手动创建或者在程序运行前就已经存在的代理类,静态代理通常只代理一个类, * 是代理一个接口下的多个实现类。动态
- java 中遍历取值异常(Hashtable Enumerator)解决办法用迭代器取值时抛出的异常:java.util.NoSuchEle
- 推荐激活教程IntelliJ IDEA 2020最新激活码(亲测有效,可激活至 2089 年)最新idea2021注册码永久激活(激活到21
- 字符串每隔4位加空格今天弄了个银行卡识别功能,回显的时候想要将银行卡号每四位加一个空格,这样核对卡号会方便很多,这里记录一下1.正则表达式实
- Java 方法执行时的动态分派和静态分派是 Java 实现多态的本质背景Java 的动态分派和静态分派也是 Java 方法的执行原理。 Ja
- 项目概况:Spring Cloud搭的微服务,使用了eureka,FeignClient,现在遇到FeignClient调用接口时不支持上传
- 本文实例为大家分享了Java实现简单邮件发送的具体代码,供大家参考,具体内容如下需要的jar包:activation-1.1.1.jarma
- 前言:在纯 Java 代码里 我们一般都会用class.getResource(String name) 或者 class.getClass
- 栈和队列的本质是相同的,都只能在线性表的一端进行插入和删除。因此,栈和队列可以相互转换。用栈实现队列—力扣232题题目要求:仅使用两个栈实现
- 需求描述:企业开发过程中,经常需要将一些静态文本数据放到Resources目录下,项目启动时或者程序运行