并发基础篇(六):线程Thread类的start()方法和run()方法
点个赞,看一看,好习惯!本文 GitHub https://github.com/OUYANGSIHAI/JavaInterview 已收录,这是我花了 3 个月总结的一线大厂 Java 面试总结,本人已拿大厂 offer。
另外,原创文章首发在我的个人博客:blog.ouyangsihai.cn,欢迎访问。
一、初识
java的线程是通过java.lang.Thread
类来实现的。VM启动时会有一个由主方法所定义的线程。可以通过创建Thread的实例来创建新的线程。每个线程都是通过某个特定Thread对象所对应的方法run()
来完成其操作的,方法run()称为线程体。通过调用Thread类的start()
方法来启动一个线程。
在Java当中,线程通常都有五种状态,创建、就绪、运行、阻塞和死亡。
第一是创建状态。在生成线程对象,并没有调用该对象的start方法,这是线程处于创建状态。
第二是就绪状态。当调用了线程对象的start方法之后,该线程就进入了就绪状态,但是此时线程调度程序还没有把该线程设置为当前线程,此时处于就绪状态。在线程运行之后,从等待或者睡眠中回来之后,也会处于就绪状态。
第三是运行状态。线程调度程序将处于就绪状态的线程设置为当前线程,此时线程就进入了运行状态,开始运行run函数当中的代码。
第四是阻塞状态。线程正在运行的时候,被暂停,通常是为了等待某个时间的发生(比如说某项资源就绪)之后再继续运行。sleep,suspend,wait等方法都可以导致线程阻塞。
第五是死亡状态。如果一个线程的run方法执行结束或者调用stop方法后,该线程就会死亡。对于已经死亡的线程,无法再使用start方法令其进入就绪。
二、start()方法
1、为什么需要start方法;它的作用是什么?
start()方法来启动线程,真正实现了多线程运行。
start方法的作用就是将线程由NEW状态,变为RUNABLE状态。当线程创建成功时,线程处于NEW(新建)状态,如果你不调用start( )方法,那么线程永远处于NEW状态。调用start( )后,才会变为RUNABLE状态,线程才可以运行。
2、调用start()方法后,线程是不是马上执行?
线程不是马上执行的;准确来说,调用start( )方法后,线程的状态是“READY(就绪)”状态,而不是“RUNNING(运行中)”状态(关于线程的状态详细。线程要等待CPU调度,不同的JVM有不同的调度算法,线程何时被调度是未知的。因此,start()方法的被调用顺序不能决定线程的执行顺序
注意:
由于在线程的生命周期中,线程的状态由NEW ----> RUNABLE只会发生一次,因此,一个线程只能调用start()方法一次,多次启动一个线程是非法的。特别是当线程已经结束执行后,不能再重新启动。
三、run( )方法
1、run方法又是一个什么样的方法?run方法与start方法有什么关联?
run()方法当作普通方法的方式调用
run( )其实是一个普通方法,只不过当线程调用了start( )方法后,一旦线程被CPU调度,处于运行状态,那么线程才会去调用这个run()方法;
2、run()方法的执行是不是需要线程调用start()方法
上面说了,run()方法是一个普通的对象方法,因此,不需要线程调用start()后才可以调用的。可以线程对象可以随时随地调用run方法。
#Example1:
Thread t1 = new Thread(new MyTask(1));Thread t2 = new Thread(new MyTask(2));t1.run();t2.run();
上面的输出结果是固定的:
count的值:1
count的值:2
再看另一个实例:
Thread t1 = new Thread(new MyTask());Thread t2 = new Thread(new MyTask());t1.start();t2.start();
这个输出结果不是固定的,因为线程的运行没法预测。运行结果可能不一样。
MyTask 类:
//实现Runnable接口
class MyTask implements Runnable{int count;public MyTask(int count) {this.count=count;}@Overridepublic void run() {System.out.println("count的值:"+count);}
}
#Example2:
1、用start方法启动线程
public class Main { public static void main(String[] args) { Thread t1 = new Thread(new T1()); Thread t2 = new Thread(new T2()); t1.start(); t2.start(); } } class T1 implements Runnable { public void run() { try { for(int i=0;i<10;i++){ System.out.println(i); Thread.sleep(100); //模拟耗时任务 } } catch (InterruptedException e) { e.printStackTrace(); } } } class T2 implements Runnable { public void run() { try { for(int i=0;i>-10;i--){ System.out.println(i); Thread.sleep(100); //模拟耗时任务 } } catch (InterruptedException e) { e.printStackTrace(); } } }
结果:
说明两线程是并发执行的。
2、先用run方法启动线程
将上面的start()改为run()
public class Main { public static void main(String[] args) { Thread t1 = new Thread(new T1()); Thread t2 = new Thread(new T2()); t1.run(); t2.run(); } }
说明两线程实际是顺序执行的。
总结:
通过实例1和实例和我们可以知道start方法是用于启动线程的,可以实现并发,而run方法只是一个普通方法,是不能实现并发的,只是在并发执行的时候会调用。
说到这,不知道小伙伴们有没有明白这两个方法的区别,如果还有疑问,可以留言交流。
四、start()方法和run()方法源码解析(基于JDK1.7.0_40)
public synchronized void start() { // 如果线程不是"就绪状态",则抛出异常! if (threadStatus != 0) throw new IllegalThreadStateException(); // 将线程添加到ThreadGroup中 group.add(this); boolean started = false; try { // 通过start0()启动线程,新线程会调用run()方法 start0(); // 设置started标记=true started = true; } finally { try { if (!started) { group.threadStartFailed(this); } } catch (Throwable ignore) { } } }
public void run() { if (target != null) { target.run(); }
}
五、真正理解Thread类
Thread类的对象其实也是一个java对象,只不过每一个Thread类的对象对应着一个线程。Thread类的对象就是提供给用户用于操作线程、获取线程的信息。真正的底层线程用户是看不到的了。
因此,**当一个线程结束了,死掉了,对应的Thread的对象仍能调用,除了start( )方法外的所有方法(死亡的线程不能再次启动),如run( )、getName( )、getPriority()**等等
//简单起见,使用匿名内部类的方法来创建线程Thread thread = new Thread(){@Overridepublic void run() {System.out.println("Thread对象的run方法被执行了");}};//线程启动thread.start();//用循环去监听线程thread是否还活着,只有当线程thread已经结束了,才跳出循环while(thread.isAlive()){}//线程thread结束了,但仍能调用thread对象的大部分方法System.out.println("线程"+thread.getName()+"的状态:"+thread.getState()+"---优先级:"+thread.getPriority());//调用run方法thread.run();//当线程结束时,start方法不能调用,下面的方法将会抛出异常thread.start();
参考资料
- http://www.cnblogs.com/jinggod/p/8485143.html
- https://blog.csdn.net/u010568463/article/details/47911181
- https://blog.csdn.net/xuxurui007/article/details/7685076
点个赞,看一看,好习惯!本文 GitHub https://github.com/OUYANGSIHAI/JavaInterview 已收录,这是我花了 3 个月总结的一线大厂 Java 面试总结,本人已拿大厂 offer。
另外,原创文章首发在我的个人博客:blog.ouyangsihai.cn,欢迎访问。
最后,再分享我历时三个月总结的 Java 面试 + Java 后端技术学习指南,这是本人这几年及春招的总结,已经拿到了大厂 offer,整理成了一本电子书,拿去不谢,目录如下:
现在免费分享大家,在下面我的公众号 程序员的技术圈子 回复 面试 即可获取。
有收获?希望老铁们来个三连击,给更多的人看到这篇文章
1、老铁们,关注我的原创微信公众号「程序员的技术圈子」,专注于 Java、数据结构和算法、微服务、中间件等技术分享,保证你看完有所收获。
2、给俺点个赞呗,可以让更多的人看到这篇文章,顺便激励下我继续写作,嘻嘻。
3、另外,原创文章首发在我的个人博客:blog.ouyangsihai.cn,欢迎访问。
点赞是对我最大的鼓励
↓↓↓↓↓↓
并发基础篇(六):线程Thread类的start()方法和run()方法相关推荐
- 线程Thread类的start()方法和run()方法
一.初识 java的线程是通过java.lang.Thread类来实现的.VM启动时会有一个由主方法所定义的线程.可以通过创建Thread的实例来创建新的线程.每个线程都是通过某个特定Thread对象 ...
- 【高并发】又一个朋友面试栽在了Thread类的stop()方法和interrupt()方法上!
来自:冰河技术 写在前面 新一轮的面试已经过去,可能是疫情的原因吧,很多童鞋纷纷留言说今年的面试题难度又提高了,尤其是对并发编程的知识.我细想了下,也许有那么点疫情的原因吧,但无论面试的套路怎么变,只 ...
- 据说有人面试栽在了Thread类的stop()方法和interrupt()方法上
摘要:今天就简单的说说Thread类的stop()方法和interrupt()方法到底有啥区别. 本文分享自华为云社区<[高并发]又一个朋友面试栽在了Thread类的stop()方法和inter ...
- 并发基础(七):Thread 类的sleep()、yeild()、join()
1.Thread.sleep(long millis ) sleep( )是一个静态方法,让当前正在执行的线程休眠(暂停执行),而且在睡眠的过程是不释放资源的,保持着锁. 在睡眠的过程,可以被中断,注 ...
- Thread start()方法和run()方法的区别
Thread start() 方法 ,是一个线程开始的方法 Thread run() 方法 ,就是一个普通的方法,可以理解为main方法里面的一个普通的方法 下面写一个demo 理解下 Thread ...
- java中Object和Objects类、toString方法和equals方法介绍
Object类: Object类位于java.lang包中,lang包下的类无需导包就可以使用,object是Java语言中的根类,即所有类的父类.Object中描述的所有方法其子类都可以使用.在对象 ...
- emberjs重写补充类之reopen方法和reopenClass方法
无需一次性将类定义完全,你可以使用reopen方法来重新打开(reopen)一个类并为其定义新的属性. Person.reopen({isPerson: true }); Person.create( ...
- DateFormat类的format方法和parse方法
package com.learn.demo03.DateFormat;import java.text.ParseException; import java.text.SimpleDateForm ...
- (JAVA学习笔记) Scanner类中next方法和nextline方法的区别
next(): 1.一定要读取到有效字符后才可以结束输入. 2. 对输入有效字符之前遇到的空白,next()方法会自动将其去掉. 3.只有输入有效字符后才将其后面输入的空白作为分隔符或结束符. *4. ...
最新文章
- Qt pro文件下跨平台宏的使用(windows/linux 以及x86 和 arm的区分)
- private MyAddin(System.IntPtr mdlDesc) : base(mdlDesc)这一步后就出bug
- 【bzoj3631】[JLOI2014]松鼠的新家
- JAVA学习笔记-“Hello World”
- 入门C语言10问10答
- centos7.4进入单用户模式
- 【Elasticsearch】index [index] blocked by: [TOO_MANY_REQUESTS/12/index read-only / allow delete (api)]
- 2018 Multi-University Training Contest 2: 1007. Naive Operations
- 别忽视分布式系统这六大“暗流”
- 广州市出租车GPS数据(20210127)
- Eprime Feedback state激活错误 其他报错
- 【面试题系列|前端面试题】前端高频面试题总结(2021年最新版)
- zblog伪原创插件(2021最新版)
- 消息队列MQ的基础知识
- undi是什么意思_undefined是什么意思?
- scum服务器里找不到车,人渣SCUM车辆机制介绍 人渣SCUM车辆为什么消失
- 服务器文件防泄密系统,数据防泄密软件解决图纸泄密问题
- 连自律都做不到 还谈什么梦想
- 搭建confluence服务器(详细操作+踩坑说明)
- Android实现网络下载一(单任务下载--支持断点续传)