深入探究Java线程的创建与构造方法

 更新时间:2022年04月26日 10:08:17   作者:淡沫初夏Zz  
这篇文章主要给大家分享的是java线程的创建以及构造方法,想了解具体方式的小伙伴可以参考下面文章内容,希望对你有所帮助

一、创建线程

启动线程—start 方法

通过覆写 run 方法创建⼀个线程对象,但线程对象被创建出来并不意味着线程就开始运行了

  • 覆写run方法是给线程指令清单
  • 但是start方法,则是让线程去真正的执行

方法一

继承Thread类

/**
 * 继承Thread创建线程
 */
class MyThread1 extends Thread{
    @Override
    public void run() {
        //业务代码
        Thread thread = Thread.currentThread();
        System.out.println("名称:" + thread.getName());
    }
}
public class ThreadDemo1 {
    public static void main(String[] args) {
        //获得当前的线程
        Thread mainThread = Thread.currentThread();
        System.out.println("名称:" + mainThread.getName());
        Thread thread = new MyThread1();
        //开启线程
        thread.start();
    }
}

因为 Java 是单继承,继承了 Thread 就不能继承其他类了,然而 Java 可以实现多个接口,于是有了下⼀种方式

方法二

实现Runnable接口

/**
 * 使用Runnable接口创建线程
 */
class MyThread2 implements Runnable{
    @Override
    public void run() {
        Thread thread = Thread.currentThread();//得到当前线程
        System.out.println("名称:" + thread.getName());
    }
}
public class ThreadDemo2 {
    public static void main(String[] args) {
        MyThread2 myThread2 = new MyThread2();
        //创建线程
        Thread thread = new Thread(myThread2);
        //启动线程
        thread.start();
    }
}

方法三

继承Thread类使用匿名内部类

/**
 * 继承Thread使用匿名内部类创建
 */
public class ThreadDemo4 {
    public static void main(String[] args) {
        Thread thread = new Thread(){
            @Override
            public void run() {
//                业务代码
                Thread thread2 = Thread.currentThread();
                System.out.println("名称" + thread2.getName());
            }
        };
        //开始线程
        thread.start();

    }
}

方法四

实现Runnable接口,使用匿名内部类

/**
 * runnable使用匿名内部类创建
 */
public class ThreadDemo3 {
    public static void main(String[] args) {
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                //具体业务
                Thread thread1 = Thread.currentThread();
                System.out.println("名称:" + thread1.getName());
            }
        });
        //开启线程
        thread.start();
    }
}

方法五

使用lambda表达式

/**
 * 使用lambda表达式
 */
public class ThreadDemo5 {
    public static void main(String[] args) {
        //创建线程
        Thread thread = new Thread(()->{
            //业务代码
            Thread thread3 = Thread.currentThread();
            System.out.println("名称" + thread3.getName());
        });
        //启动线程
        thread.start();
    }
}

方法六

带返回值的 Callable

import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

/**
 * 使用callable创建线程
 */
public class ThreadDemo6 {
    public static void main(String[] args)throws ExecutionException, InterruptedException {
        // 创建 Callable 实例
        MyCallable callable = new MyCallable();
        // 用于接收 Callable 结果的对象
        FutureTask<Integer> futureTask = new FutureTask<Integer>(callable);
        // 创建新线程
        Thread thread = new Thread(futureTask);
        // 启动线程
        thread.start();
        // 接收新线程执行的结果
        int result = futureTask.get();
        System.out.println(Thread.currentThread().getName() +
                "——新线程返回的结果为:" + result);
    }
}
/**
 * Callable<V> 泛型里面可以是任意数据类型
 */
class MyCallable implements Callable<Integer> {
    @Override
    public Integer call() throws Exception {
        // 生成随机数:0-9
        int randomNum = new Random().nextInt(10);
        System.out.println(Thread.currentThread().getName() +
                "——随机数:" + randomNum);
        return randomNum;
    }
}

在创建线程时, 如果是 JDK 1.8 以上版本,在不需要获得线程执行结果的情况下,推荐使用Lambda 方式来创建线程,因为它的写法足够简洁;如果想要获取线程执行结果,可使用FutureTask + Callable 的方式来实现

二、run方法和start方法的区别

run 方法和 start 方法的主要区别如下:

①方法性质不同

run 是一个普通方法,而 start 是开启新线程的方法。

②执行速度不同

调用 run 方法会立即执行任务,调用 start 方法是将线程的状态改为就绪状态,不会立即执行。

③调用次数不同

run 方法可以被重复调用,而 start 方法只能被调用一次。start 方法之所以不能被重复调用的原因是,线程的状态是不可逆的,Thread 在 start 的实现源码中做了判断,如果线程不是新建状态 NEW,则会抛出非法线程状态异常IllegalThreadStateException

  public static void main(String[] args) {
        // 创建线程一
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                // 获取到当前执行的线程
                Thread currThread = Thread.currentThread();
                System.out.println("执行线程一,线程名:" + currThread.getName());
            }
        });
        // 调用 run 方法
        thread.run();
        // 多次调用 run 方法
        thread.run();

        // 创建线程二
        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                // 获取到当前执行的线程
                Thread currThread = Thread.currentThread();
                System.out.println("执行线程二,线程名:" + currThread.getName());
            }
        });
       // 调用 start 方法
        thread2.start();
       // 多次调用 start 方法
        thread2.start();
    }

从上述结果可以看出,run 方法多次调用可用正常执行,而第二次调用 start 方法时程序就报错了,提示“IllegalThreadStateException”非法线程状态异常

总结

方法性质不同:run 是一个普通方法,而 start 是开启新线程的方法。

执行速度不同:调用 run 方法会立即执行任务,调用 start 方法是将线程的状态改为就绪状态,不会立即执行。

调用次数不同:run 方法可以被重复调用,而 start 方法只能被调用一次。

三、线程的构造方法

1、Thread()创建线程

Thread t1 = new Thread();

2、Thread(Runnable target) 创建线程

Thread t2 = new Thread(new MyRunnable());

3、Thread(String name)创建线程且命名

/**
 * 创建线程,构造方法设置线程名称
 */
public class ThreadDemo9 {
    public static void main(String[] args) {
        //构造方法设置名称
        Thread thread = new Thread("线程1"){
            @Override
            public void run() {
                //休眠线程
                try {
                    Thread.sleep(1000*60*60);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };
        thread.start();
    }
}

4、Thread(Runnable target, String name),用Runnable 对象创建线程对象,并命名

/**
 * 创建线程,并设置名称
 */
public class ThreadDemo10 {
    public static void main(String[] args) {
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000*60*60);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"Runable-Thread");
        thread.start();
    }
}

到此这篇关于深入探究Java线程的创建与构造方法的文章就介绍到这了,更多相关Java线程内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解如何在Java中实现堆排序算法

    详解如何在Java中实现堆排序算法

    这篇文章主要为大家详细介绍了如何利用Java实现堆排序算法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • Java的RocketMQ之消息存储和查询原理详解

    Java的RocketMQ之消息存储和查询原理详解

    这篇文章主要介绍了Java的RocketMQ之消息存储和查询原理详解,一台Broker服务器只有一个CommitLog文件(组),RocketMQ会将所有主题的消息存储在同一个文件中,这个文件中就存储着一条条Message,每条Message都会按照顺序写入,需要的朋友可以参考下
    2024-01-01
  • Java算法设计与分析分治算法

    Java算法设计与分析分治算法

    这篇文章主要介绍了Java算法设计与分析分治算法,一般分治算法在正文中分解为两个即以上的递归调用,并且子类问题一般是不想交的
    2022-07-07
  • java写卷积神经网络(CupCnn简介)

    java写卷积神经网络(CupCnn简介)

    这篇文章主要介绍了java写卷积神经网络(CupCnn简介),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-04-04
  • Java Stream去重常见的多种方法及示例

    Java Stream去重常见的多种方法及示例

    这篇文章主要介绍了Java Stream中实现去重的多种方法,包括使用distinct()、Collectors.toMap、filter和groupingBy等,详细说明了每种方法的应用场景和示例代码,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2025-04-04
  • 使用Jmeter进行http接口测试的详细流程

    使用Jmeter进行http接口测试的详细流程

    本文主要针对http接口进行测试,使用Jmeter工具实现,  Jmter工具设计之初是用于做性能测试的,它在实现对各种接口的调用方面已经做的比较成熟,因此,本次直接使用Jmeter工具来完成对Http接口的测试,需要的朋友可以参考下
    2024-12-12
  • Java使用Armitage进行渗透测试的方法

    Java使用Armitage进行渗透测试的方法

    在网络安全领域,渗透测试是一种重要的安全评估手段,它通过模拟恶意黑客的行为来检测目标系统是否存在安全漏洞,Armitage是一个基于Java的图形化渗透测试工具,本文将介绍如何使用Armitage进行渗透测试,需要的朋友可以参考下
    2024-12-12
  • java使用动态代理来实现AOP(日志记录)的实例代码

    java使用动态代理来实现AOP(日志记录)的实例代码

    AOP(面向方面)的思想,就是把项目共同的那部分功能分离开来,比如日志记录,避免在业务逻辑里面夹杂着跟业务逻辑无关的代码
    2013-09-09
  • RestTemplate使用不当引发的问题及解决

    RestTemplate使用不当引发的问题及解决

    这篇文章主要介绍了RestTemplate使用不当引发的问题及解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • spring三级缓存以及为什么不用二级缓存解读

    spring三级缓存以及为什么不用二级缓存解读

    Spring三级缓存机制解决了循环依赖问题,通过一级缓存存放完全初始化的bean,二级缓存存放实例化但未完成依赖注入和初始化的bean,三级缓存存放bean的创建工厂,避免了重复创建和确保代理对象的正确生成
    2025-02-02

最新评论