当前位置: 首页 > news >正文

【Java|多线程与高并发】详解start()方法和run()方法的区别

文章目录

  • 前言
  • 1.start()方法和run()方法
  • 2.不能两次调用start()方法
  • 3.线程的执行是随机的
    • start()方法和run()方法的执行顺序不一定相同
  • 4.run()方法由JVM调用
    • public Thread(Runnable target)中的target

前言

本篇文章主要讲解以下几个问题:

  • start()方法和run()方法的区别与联系
  • 为什么不能连续两次调用shart()方法
  • run()方法中可能忽略的问题

1.start()方法和run()方法

我们在创建线程时,会重写run()方法.run()方法可以理解为线程要做的任务,但是直接调用run()方法,只是main线程也就是主线程去执行的,是没有新线程产生的
如果要想让线程去执行run()方法里面的代码,就需要让创建线程的对象去调用start()方法,shart()方法可以创建并启动线程,JVM调用run()方法后(后面会有介绍) 线程才会去执行

2.不能两次调用start()方法

同一个线程对象,只能调用一次start()方法,不能两次调用start()方法,调用两次的话,会抛出 IllegalThreadStateException 这个异常
示例如下:

public class Example6 {public static void main(String[] args) {Thread t1 = new Thread(()->{System.out.println("1");});t1.start();t1.start();}
}

在这里插入图片描述
为什么会这样?我们可以查看start()的源码
在这里插入图片描述
上面有一段话: A zero status value corresponds to state "NEW" 意思是 零状态值对应于状态“NEW”
线程运行时有六种状态,先熟悉一下,之后还会写文章进行详细介绍:

状态描述
新建(NEW)表示线程已经创建好了,但是还没有调用start()方法
就绪(RUNNABLE)表示线程可能在运行,也可能在就绪队列
阻塞 (BLOCKED)表示线程处于等在锁的状态
等待(WAITING)表示线程处于条件等待状态,当触发条件后会唤醒
计时等待(TIME_WAIT)比WAITING多了个超时条件触发的机制
终止(TERMINATED)表示线程执行结束

因此线程再调用start()方法之后,可能处于终止,或者其它非NEW状态,第二次调用的时候,相当于重新让线程运行一遍,从线程安全和线程本身的执行逻辑来看,都是不合理的,因此在调用start()方法之后,会对线程的状态进行一个判断,如果线程不是在NEW状态下,就会抛出异常

3.线程的执行是随机的

线程的执行是随机的,这也是个老生常谈的问腿了,究其原因还是因为线程的"抢占式执行",谁先"抢"到
操作系统分配的CPU资源,谁先去执行

start()方法和run()方法的执行顺序不一定相同

示例1:
看一下代码及代码运行结果

class MyThread extends Thread{private int val;public MyThread(int val) {this.val = val;}@Overridepublic void run() {System.out.println(val);}
}
public class Example1 {public static void main(String[] args) {MyThread myThread1 = new MyThread(1);MyThread myThread2 = new MyThread(2);MyThread myThread3 = new MyThread(3);MyThread myThread4 = new MyThread(4);MyThread myThread5 = new MyThread(5);myThread1.start();myThread2.start();myThread3.start();myThread4.start();myThread5.start();}
}

在这里插入图片描述

由此我们可以看到虽然是myThread1先调用的start()方法,但是输出的结果却是在第二位,而myThread3后执行却在第一位
在这里插入图片描述
再次运行一次代码的执行结果虽然与上次不同,但仍然是随机的,当然也不是没有运行结果是1 2 3 4 5的可能

示例2:
看一下代码及代码运行结果

public class Example4 {public static void main(String[] args) {Thread t1 = new Thread(()->{while (true){System.out.println("t1");}});Thread t2 = new Thread(()->{while (true){System.out.println("t2");}});t1.start();t2.start();}
}

在这里插入图片描述

这里我们也能看到,start()执行的顺序与run()方法执行的顺序无关,随机执行t1和t2的run()方法 虽然这里是while(true),但是线程之间是相互独立的,所以并不影响
因此可以得出结论:start()方法和run()方法的执行顺序不一定相同

4.run()方法由JVM调用

start()执行完后,创建的新线程不会立刻去执行run()方法, run()方法会先被JVM去调用,之后对应的线程才会去执行.

public Thread(Runnable target)中的target

之前再讲创建线程的5种方法,介绍了实现Runnable接口,创建线程的方法
在这里插入图片描述

但其实这里面有一个坑,那就是public Thread(Runnable target)中的target参数,来看一下run()方法的底层源码

在这里插入图片描述
如果这里面的target不为空,才会去执行target的run()方法.如果传一个null,就不会去执行,如图所示:
在这里插入图片描述
代码如下:

class MyRunnable implements Runnable{@Overridepublic void run() {System.out.println("1");}
}
public class Example7 {public static void main(String[] args) {MyRunnable myRunnable1 = new MyRunnable();MyRunnable myRunnable2 = null;Thread t1 = new Thread(myRunnable1);t1.start();Thread t2 = new Thread(myRunnable2);t2.start();}
}

因此要注意使用继承Runnable接口 创建线程的时候 要注意target不要为null

文章到这里去结束了,感谢观看!
“希望你继续兴致盎然的与世界交手,一直走在充满鲜花的道路”
在这里插入图片描述

http://www.lryc.cn/news/8645.html

相关文章:

  • mysql 一些有意思的sql语句,备忘
  • hive自定义函数
  • 数仓理论【范式】【维度建模】
  • 卷积神经网络
  • 解决Qt提示xxx.so not found( using -rpath or -rpath-link)问题
  • Blazor 托管模型 BlazorWebAssembly和Blazor Server
  • 从未想过制作数据可视化展示竟可以如此简单
  • 企业电子招投标采购系统源码之功能模块的描述
  • LeetCode-2341. 数组能形成多少数对【哈希表,计数】
  • vue-echarts实现多功能图表
  • C#快键精灵
  • 谷歌、微软、Meta?谁才是 Python 最大的金主?
  • 面向对象笔记
  • tofu:一款功能强大的模块化Windows文件系统安全测试工具
  • VS中scanf为什么会报错
  • 使用kubeadm部署k8s1.24.0版本,遇到的坑总结
  • 【C++】特殊类设计
  • 中创教育PMP如何轻松应对公司90%以上的沟通难题
  • #笨鸟先飞# 数据结构与算法基础 课程笔记 第六章 图
  • 深入浅出带你学习Apache中间件常见漏洞
  • 用多种指针方法访问数据元素,实现逆序输出
  • WebDAV之葫芦儿·派盘+NMM
  • Redis多级缓存
  • 【原创】java+swing+mysql会议室管理系统设计与实现
  • 【Redis】Redis 常用数据类型操作 ① ( 数据库操作 | Redis 数据库连接参数 | Redis 数据库个数 | Redis 访问机制 )
  • GAMES101-计算机图形学入门 LEC4: TRANSFORMATION-3D
  • robot实战:截取字符串
  • 【面经】滴滴测开一面
  • 数据治理-主数据
  • 软考-中级-软件设计师-成绩