您的位置:首页 > 编程语言 > Java开发

JAVA中多线程知识--线程的生命周期:新建、就绪状态

2019-06-05 23:33 696 查看

今天小白去面试,感觉还不错,只是有些不怎么注意的东西有点记不起来了,所以还是要经常泡一泡旧知识啊~~~

说完了三种线程的创建以及启动,今天就来聊一聊关于线程的整个运行的生命周期

前面线程创建以及运行的过程可以看到,并不是因为当前线程是main,也就是主线程而说优先执行,而是主线程以及子线程交互执行,所以线程不可能独占CPU,而是CPU轮流着在各线程之间切换,整个过程线程会有5种基本状态,前面可以看到的4种分别是新建(new)、就绪(Runnable)、运行(Running)、死亡(dead),还有 一种就是阻塞(Blocked)

接下来先讲线程的新建和 就绪状态

1.当一个线程被new出来之后,线程的状态就为新建,此时仅仅只是一个线程对象,由jvm为其分配内存,初始化其成员变量值,并不会执行该线程的执行体
2.该对象调用start()方法后,线程处于就绪状态,但不会立刻就执行线程执行体,也就是执行run方法,Callable的为call方法,至于何时会执行,完全取决于JVM里线程调度器的调度

注意注意注意!!!
永远永远不要直接调用线程的run方法!!! 当直接调用run()方法时,run()方法仅仅只是一个普普通通的方法,并不是线程执行体,在直接调用该方法后,只有等该方法完全执行完,其他调用start()的线程才会并发执行

代码说明

package com.MrLiu.group;

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

//继承Thread类创建线程
public class TestClazz extends  Thread{

private int index;

//重写run方法
@Override
public void run() {
for (; index < 50; index++) {
//直接调用run方法时,this.getName()返回的是当前对象的名字,并不是当前线程的名称
System.out.println("====="+getName());
//(Thread.currentThread().getName()  获取当前运行时线程名称
System.out.println(Thread.currentThread().getName() + "------i="+index);
}
}

public static void main(String[] args) {

for (int i = 0; i < 50; i++) {
//获取当前线程
System.out.println(Thread.currentThread().getName() + "-----i" +i);

if (i == 10){
//直接调用run方法,只是普通的运行一个普通的方法,并不会创建一个线程与主线程并发执行
TestClazz testClazz = new TestClazz();
testClazz.run();
//当直接调用run方法后 再次调用start开启线程,此时已无法开启
//testClazz.start();
new TestClazz().run();
//调用start方法后run方法才会被当做是线程执行体,与主线程并发执行
new TestClazz().start();
}
}
}
}
package com.MrLiu.group;

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

//继承Thread类创建线程
public class TestClazz extends  Thread{

private int i;

//重写run方法
@Override
public void run() {
for (; i < 100; i++) {
//(Thread.currentThread().getName()  获取当前运行时线程名称
System.out.println("子线程被调度");
System.out.println("当前执行线程----------"+this.getName() + "------i="+i);
}
}

public static void main(String[] args) {

for (int i = 0; i < 100; i++) {
//获取当前线程
System.out.println("当前线程==="+Thread.currentThread().getName() + "-----i" +i);
//  sleep(1000);
if (i == 20){
new TestClazz().start();
}
}
}
}

这里要注意的是 , 线程开启的条件为 i == 10 但线程此时仅仅处于就绪状态 , 并不是立马被执行,有可能 11 ,或者 21 又或者

sleep(1000); 让主线程睡眠一秒就能让子线程立马被调度 ,可以尝试一下

今天就到这~

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: