Java Thread.currentThread().getName() 和 this.getName()区别详解

 更新时间:2022年02月21日 08:39:42   作者:经典鸡翅  
本文主要介绍了Thread.currentThread().getName() 和 this.getName()区别详解,TestThread testThread = new TestThread();

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。

总结

currentThread的getName代表当前执行线程的名称,this.getName代表对象的名称。this.getName()实际上返回的是target.getName(),而Thread.currentThread().getName()实际上是thead.getName()。

到此这篇关于Java Thread.currentThread().getName() 和 this.getName()区别详解的文章就介绍到这了,更多相关Thread.currentThread().getName() 和 this.getName()区别 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java从零编写汽车租赁系统全程分析

    Java从零编写汽车租赁系统全程分析

    这篇文章介绍了Java实现汽车租赁系统的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-12-12
  • SpringBoot排除不需要的自动配置类DataSourceAutoConfiguration问题

    SpringBoot排除不需要的自动配置类DataSourceAutoConfiguration问题

    这篇文章主要介绍了SpringBoot排除不需要的自动配置类DataSourceAutoConfiguration问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07
  • Java中的封装、继承和多态,你真的都懂了吗

    Java中的封装、继承和多态,你真的都懂了吗

    Java中的封装、继承和多态知识点是学习java必备的基础知识,看似简单,真正理解起来还是有一定难度的,今天小编再次通过实例代码给大家讲解java 封装继承多态知识,感兴趣的朋友一起学习下吧
    2021-05-05
  • SpringBoot 多Profile使用与切换方式

    SpringBoot 多Profile使用与切换方式

    这篇文章主要介绍了SpringBoot 多Profile使用与切换方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-04-04
  • Mybatis-plus如何在xml中传入自定义的SQL语句

    Mybatis-plus如何在xml中传入自定义的SQL语句

    这篇文章主要介绍了Mybatis-plus如何在xml中传入自定义的SQL语句问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05
  • IDEA如何开启并配置services窗口

    IDEA如何开启并配置services窗口

    在使用IntelliJ IDEA时,可能会遇到Services窗口不自动弹出的情况,本文介绍了如何手动开启Services窗口的简单步骤,首先,通过点击菜单栏中的“视图”->“工具窗口”->“服务”,或使用快捷键Alt+F8(注意快捷键可能存在冲突)来打开Services窗口
    2024-10-10
  • Spring4下validation数据校验无效(maven)的解决

    Spring4下validation数据校验无效(maven)的解决

    这篇文章主要介绍了Spring4下validation数据校验无效(maven)的解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-06-06
  • SpringBoot 快速实现分库分表的2种方式

    SpringBoot 快速实现分库分表的2种方式

    本文将为您介绍 ShardingSphere 的一些基础特性和架构组成,以及在 Springboot 环境下通过JAVA编码和Yml配置两种方式快速实现分库分表功能,感兴趣的朋友跟随小编一起看看吧
    2023-06-06
  • java8时间 yyyyMMddHHmmss格式转为日期的代码

    java8时间 yyyyMMddHHmmss格式转为日期的代码

    这篇文章主要介绍了java8时间 yyyyMMddHHmmss格式转为日期的代码,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-09-09
  • Java设计模式之观察者模式

    Java设计模式之观察者模式

    本文详细讲解了Java设计模式之观察者模式,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-09-09

最新评论