• Java多线程系列---“基础篇”03之 Thread中start()和run()的区别


    转自:https://www.cnblogs.com/skywang12345/p/3479083.html  (含部分修改)

    概要

    Thread类包含start()和run()方法,它们的区别是什么?本章将对此作出解答。本章内容包括:

    • start() 和 run()的区别说明
    • start() 和 run()的区别示例
    • start() 和 run()相关源码(基于JDK1.7.0_40)

    一. start() 和 run()的区别说明

    start() : 它的作用是启动一个新线程,新线程会执行相应的run()方法。start()不能被重复调用(否则抛出异常)。 Mynote:runnable没有start方法。要启动,必须要start方法,因此runnable构造方法中需要和Thread来进行绑定。
    run()   : run()就和普通的成员方法一样,可以被重复调用。单独调用run()的话,会在当前线程中执行run(),而并不会启动新线程

    下面以代码来进行说明。

    class MyThread extends Thread{  
        public void run(){
            ...
        } 
    };
    MyThread mythread = new MyThread();

    mythread.start()会启动一个新线程,并在新线程中运行run()方法。
    而mythread.run()则会直接在当前线程中运行run()方法,并不会启动一个新线程来运行run()。

    二. start() 和 run()的区别示例

    下面,通过一个简单示例演示它们之间的区别。源码如下:

    // Demo.java 的源码
    class MyThread extends Thread{  
        public MyThread(String name) {
            super(name);
        }
    
        public void run(){
            System.out.println(Thread.currentThread().getName()+" is running");
        } 
    }; 
    
    public class Demo {  
        public static void main(String[] args) {  
            Thread mythread=new MyThread("mythread");//如果不指定名字,则新开启得线程名字就是thread-0
    
            System.out.println(Thread.currentThread().getName()+" call mythread.run()");
            mythread.run();
    
            System.out.println(Thread.currentThread().getName()+" call mythread.start()");
            mythread.start();//会自动调用run方法
        }  
    }

    运行结果

    main call mythread.run()
    main is running
    main call mythread.start()
    mythread is running

    结果说明
    (01) Thread.currentThread().getName()是用于获取“当前线程”的名字。当前线程是指正在cpu中调度执行的线程。
    (02) mythread.run()是在“主线程main”中调用的,该run()方法直接运行在“主线程main”上。
    (03) mythread.start()会启动“线程mythread”,“线程mythread”启动之后,会调用run()方法;此时的run()方法是运行在“线程mythread”上。

    三. start() 和 run()相关源码(基于JDK1.7.0_40)

    Thread.java中start()方法的源码如下:

    复制代码
    public synchronized void start() {
        // 如果线程不是"就绪状态",则抛出异常!
        if (threadStatus != 0)
            throw new IllegalThreadStateException();
    
        // 将线程添加到ThreadGroup中
        group.add(this);
    
        boolean started = false;
        try {
            // 通过start0()启动线程
            start0();
            // 设置started标记
            started = true;
        } finally {
            try {
                if (!started) {
                    group.threadStartFailed(this);
                }
            } catch (Throwable ignore) {
            }
        }
    }
    复制代码

    说明start()实际上是通过本地方法start0()启动线程的。而start0()会新运行一个线程,新线程会调用run()方法。

    补充:从start的源码来看,里面有两个特殊的部分:一个是throw 了一个IllegalThreadStateException异常。按照道理来讲,应该使用try...catch处理,或者在start()方法上使用throws声明。这就说明了这个异常是RuntimeException的子类。如果某个线程重复执行了启动,就会抛出这个异常,根据status来判断。因此,不可以重复执行start方法。start0(),此方法的结构与抽象方法类似,使用了native修饰,在Java开发里面有一门技术称为JNI技术(Java Native),这门技术的特点是使用Java调用本机操作系统提供的函数。但是这样的技术有一个缺点,不能离开特定的操作系统。

    private native void start0();

    为何要start0?

    如果要想线程能够执行,需要操作系统来进行资源分配所以此操作严格来讲主要是由JVM 负责根据不同的操作系统而实现的。

    Thread.java中run()的代码如下:

    public void run() {
        if (target != null) {
            target.run();
        }
    }

    说明target是一个Runnable对象。run()就是直接调用Thread线程的Runnable成员的run()方法,并不会新建一个线程。

     

    因此,从源码就可以知道为什么用start进行启动,而不用run进行启动。run仅仅是方法调用,而start才会启动一个新线程,并由新的线程来调用run方法的具体实现。

    (note:调用了start方法,只是由new变成了就绪状态,并不能马上运行,这是由CPU的时间分片决定的。等拥有了CPU使用权才开始运行,即状态变成running。相当于由操作系统来决定新线程是不是要调用这个run方法的具体实现了)

    调用start后,就有了两条执行流,新的一条执行run方法,旧的一条继续执行main方法,两条执行流并发执行,操作系统负责调度,在单CPU的机器上,同一时刻只能有一个线程在执行,在多CPU的机器上,同一时刻可以有多个线程同时执行,但操作系统给我们屏蔽了这种差异,给程序员的感觉就是多个线程并发执行,但哪条语句先执行哪条后执行是不一定的。当所有线程都执行完毕的时候,程序退出。 

     

  • 相关阅读:
    spring boot 2.1学习笔记【五】SpringBootTest单元测试及日志
    Java网络编程-UDP
    Java网络编程-TCP
    String的特性
    内存池的使用
    软件定时器的使用
    邮箱
    事件集
    线程优先级翻转
    临界区,互斥量与信号量
  • 原文地址:https://www.cnblogs.com/Hermioner/p/9839427.html
Copyright © 2020-2023  润新知