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

目录
  • 一、创建线程
    • 启动线程—start 方法
    • 方法一
    • 方法二
    • 方法三
    • 方法四
    • 方法五
    • 方法六
  • 二、run方法和start方法的区别
    • ①方法性质不同
    • ②执行速度不同
    • ③调用次数不同
    • 总结
  • 三、线程的构造方法

一、创建线程

启动线程—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线程内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java的线程与进程以及线程的四种创建方式

    目录 问题描述 case代码截图 数据库DO controller定义 dao定义 mapper实现 mysql相关properties配置 数据库数据 测试结果 具体错误信息 解决 总结 问题描述 这里我想测试某个与springboot相关的问题,结果在搭建mybatis时,发现没有成功从数据库中获取数据,报错信息为 com.mysql.cj.exceptions.DataConversionException: Unsupported conversion from LONG to java

  • Java并发编程之线程创建介绍

    目录 1.线程与进程 2.线程的创建与运行 1.线程与进程 进程是代码在数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,线程则是一个实体,一个进程中至少有一个线程,是CPU调度和分配的基本单位,进程中的多个线程共享进程的资源. 进程的三个特征: 动态性 : 进程是运行中的程序,要动态的占用内存,CPU和网络等资源. 独立性 : 进程与进程之间是相互独立的,彼此有自己的独立内存区域. 并发性 : 假如CPU是单核,同一个时刻其实内存中只有一个进程在被执行.CPU会分时轮询切换依次为每

  • Java 构造方法的使用详解

    目录 1. 看一个需求 2. 基本语法 3. 基本介绍 4. 快速入门 5. 注意事项和使用细节 6. 练习 1. 看一个需求 看一个需求:前面我们在创建人类的对象时,是先把一个对象创建好后,再给他的年龄和姓名属性赋值,如果现在要求,在创建人类的对象时,就直接指定这个对象的年龄和姓名,该怎么做? 这时就可以使用构造器. 2. 基本语法 [修饰符] 方法名(形参列表){ 方法体; } 说明: 构造器的修饰符可以默认, 也可以是 public protected private 构造器没有返回值 方

  • Java构造方法和方法重载详解

    目录 第一 构造方法的作用 第二 构造方法的特点 方法重载 总结 类的结构包括 : 1. 成员变量 2. 成员方法 3. 构造方法 4. 代码块 5. 内部类 第一 构造方法的作用 主要有以下三方面的作用: (1)在构造方法中为创建的对象初始化赋值 (2)在创建一个对象的时候,至少需要调用一个构造方法 (3)每一个类都有构造方法 一个例子加深对以上三条的理解 public class Car{ String name; String color; float price; } 上一篇文章已经讲解

  • Java并发编程创建并运行线程的方法对比

    目录 一.创建并运行线程的五种方法 第一种:继承Thread类 第二种:实现Runnable接口 第三种:实现Callable接口 第四种:线程池 + execute 第五种:线程池 + submit 总结 一.创建并运行线程的五种方法 第一种:继承Thread类 这种方式是最基础的一种方式,学过java的朋友都知道,不做赘述.需要注意的是:覆盖实现使用的是run方法,运行线程是start方法. public class FirstWay extends Thread { @Override p

  • 详解Java创建线程的五种常见方式

    目录 Java中如何创建线程呢? 1.显示继承Thread,重写run来指定现成的执行代码. 2.匿名内部类继承Thread,重写run来执行线程执行的代码. 3.显示实现Runnable接口,重写run方法. 4.匿名内部类实现Runnable接口,重写run方法 5.通过lambda表达式来描述线程执行的代码 [面试题]:Thread的run和start之间的区别? Thread类的具体用法 Thread类常见的一些属性 中断一个线程 1.方法一:让线程run完 2.方法二:调用interr

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

    目录 一.创建线程 启动线程—start 方法 方法一 方法二 方法三 方法四 方法五 方法六 二.run方法和start方法的区别 ①方法性质不同 ②执行速度不同 ③调用次数不同 总结 三.线程的构造方法 一.创建线程 启动线程—start 方法 通过覆写 run 方法创建⼀个线程对象,但线程对象被创建出来并不意味着线程就开始运行了 覆写run方法是给线程指令清单 但是start方法,则是让线程去真正的执行 方法一 继承Thread类 /** * 继承Thread创建线程 */ class M

  • 深入探究Java线程与进程有哪些区别

    目录 一.进程线和程的概念 二.为什么要有线程 三.进程和线程的关系 四.线程和进程的区别(重点) 五.用户线程和守护线程区别 一.进程线和程的概念 线程: 一个线程是一个独立的执行流,每个线程之间都可以按照顺讯执行自己的代码. 多个线程之间 “同时” 执行着多份代码 public class ThreadDemo { public static void main(String[] args) { Thread thread = new Thread(new Runnable() { @Ove

  • 深入探究Java线程的状态与生命周期

    目录 一.线程的状态 新建(初始) 就绪(可运行) 运行 阻塞 死亡 二.线程的状态转移 三.线程的生命周期 一.线程的状态 NEW: 安排了工作, 还未开始行动RUNNABLE: 可工作的. 又可以分成正在工作中和即将开始工作.BLOCKED: 这几个都表示排队等着其他事情WAITING: 这几个都表示排队等着其他事情TIMED_WAITING: 这几个都表示排队等着其他事情TERMINATED: 工作完成了 新建(初始) 当继承Thread类和实现了Runnable接口,就可以创建线程,新建

  • 深入探究Java线程不安全的原因与解决

    目录 一.什么是线程安全 二.线程不安全的原因 1.修改共享数据 2.原子性 3.内存可见性 4.指令重排序 三.解决线程安全方案 一.什么是线程安全 想给出一个线程安全的确切定义是复杂的,但我们可以这样认为: 如果多线程环境下代码运行的结果是符合我们预期的,即在单线程环境应该的结果,则说这个程序是线程安全的 二.线程不安全的原因 1.修改共享数据 static class Counter { public int count = 0; void increase() { count++; }

  • 详解Java线程的创建及休眠

    一.进程vs线程 1.进程是系统分配资源的最小单位:线程是系统调度的最小单位 2.一个进程中至少要包含一个线程 3.线程必须要依附于继承,线程是进程实质工作的一个最小单位 二.线程的创建方式 继承Thread类 实现线程的创建(2种写法) 1种写法 public class ThreadDemo03 { static class MyThread extends Thread{ @Override public void run(){ System.out.println("线程名称:"

  • 了解Java线程池创建过程

    前言 最近在改进项目的并发功能,但开发起来磕磕碰碰的.看了好多资料,总算加深了认识.于是打算配合查看源代码,总结并发编程的原理. 准备从用得最多的线程池开始,围绕创建.执行.关闭认识线程池整个生命周期的实现原理.后续再研究原子变量.并发容器.阻塞队列.同步工具.锁等等主题.java.util.concurrent里的并发工具用起来不难,但不能仅仅会用,我们要read the fucking source code,哈哈.顺便说声,我用的JDK是1.8. Executor框架 Executor是一

  • Java多线程之线程的创建

    一.三种创建方式 基于什么创建 创建的方式 Thread类 继承Thread类 Runnable接口 实现Runnable接口 callable接口 实现callable接口 二.通过Thread类创建 2.1 步骤 自定义线程类继承Thread类 重写run()方法,编写线程执行体(当成main()方法用) 创建线程对象,调用start()方法启动线程 2.2 案例 创建两个线程,其中一个线程打印100以内的偶数,另一个线程打印100以内的奇数 //主方法 public class Demo0

  • 很多人竟然不知道Java线程池的创建方式有7种

    目录 前言 什么是线程池? 线程池使用 1.FixedThreadPool 2.CachedThreadPool 3.SingleThreadExecutor 4.ScheduledThreadPool 5.SingleThreadScheduledExecutor 6.newWorkStealingPool 7.ThreadPoolExecutor 线程池的执行流程 线程拒绝策略 自定义拒绝策略 究竟选用哪种线程池? 前言 根据摩尔定律所说:集成电路上可容纳的晶体管数量每 18 个月翻一番,因

  • 创建Java线程安全类的七种方法

    目录 前言 无状态 没有共享状态 消息传递 不可变状态 使用来自 java.util.concurrent 的数据结构 同步块 易失性领域 总结 前言 几乎每个 Java 应用程序都使用线程.像 Tomcat 这样的 Web 服务器在单独的工作线程中处理每个请求,胖客户端在专用工作线程中处理长时间运行的请求,甚至批处理使用 java.util.concurrent.ForkJoinPool 来提高性能. 因此,有必要以线程安全的方式编写类,这可以通过以下技术之一来实现. 无状态 当多个线程访问同

  • java线程池的四种创建方式详细分析

    目录 前言 1. 线程池 2. 创建方式 前言 在讲述线程池的前提 先补充一下连接池的定义 连接池是创建和管理一个连接的缓冲池的技术,这些连接准备好被任何需要它们的线程使用 可以看到其连接池的作用如下: 1. 线程池 线程池(英语:thread pool):一种线程使用模式.线程过多会带来调度开销,进而影响缓存局部性和整体性能.而线程池维护着多个线程,等待着监督管理者分配可并发执行的任务.这避免了在处理短时间任务时创建与销毁线程的代价.线程池不仅能够保证内核的充分利用,还能防止过分调度 特点:

随机推荐