Java多线程开发(一)| 基本的线程机制

0. 前言

Java 为了实现跨平台,在语言层面上实现了多线程。我们只需要熟悉 Java 这一套多线程机制就行了,比 C/C++ 要容易多了。

1. 定义任务

我们编写程序,最终是为了完成特定的任务。为了更有效的利用系统资源,我们把任务合理地划分成多个子任务,放到多个线程中来执行。所以,首先我们需要一种描述任务的方式。在 Java 中,一般我们都用 Runable 接口来定义任务。

public interface Runnable {
    // 在run方法中定义任务
    public void run();
}

想要定义任务,只需要实现 Runable 接口,然后在 run() 方法中写上执行步骤。请注意,Runable 只是定义了一个任务,本身不会去启动一个新线程来执行。看下面的例子。可以看到,在外面直接打印的线程名和在 Runable 的 run() 方法中打印的线程名是相同的。

public static void main(String[] args) {
    System.out.println(Thread.currentThread().getName());
    Runnable runnable = new Runnable() {
        @Override
        public void run() {
            System.out.println(Thread.currentThread().getName());
        }
    };
    runnable.run();
}

输出结果:

main
main

2. Thread类

​ 要让任务在新的线程执行,最直接的方法是用它来创建一个 Thread 类。这里用 Thinking in Java 书上的例子来展示 Thread 类的使用。

/**
 * 显示发射之前的倒计时
 */
public class LiftOff implements Runnable {
    private static int sTaskCount = 0;
    private final int mId = sTaskCount++;
    protected int mCountDown = 10;

    public LiftOff() {
    }

    public LiftOff(int countDown) {
        mCountDown = countDown;
    }

    private String status() {
        return "#" + mId + "(" +
                ((mCountDown > 0) ? mCountDown : "Liftoff!") + "), ";
    }

    @Override
    public void run() {
        while (mCountDown-- > 0) {
            System.out.print(status());
            Thread.yield(); // Thread.yield() 是对线程调度器的一种建议,表示当前线程准备让出处理器
        }
    }
}

LiftOff 任务会显示发射前的倒计时。注意在 run() 方法中调用的 Thread.yield() 方法。这个方法的作用是对线程调度器的一种建议,表示当前线程可以让出处理器。当然,线程调度器不一定会真的切换执行线程。LifiOff 任务整个执行时间实际上很短,如果不使用 Thread.yield() 很可能直到任务执行完成线程调度器才会切换新的线程,不利于观察多线程的效果。

public class MoreBasicThreads {
    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            new Thread(new LiftOff()).start();
        }
        System.out.println("Waiting for Liftoff");
    }
}

输出结果:
#0(9), #1(9), #2(9), #3(9), #0(8), Waiting for Liftoff
#4(9), #1(8), #2(8), #3(8), #0(7), #4(8), #1(7), #2(7), #3(7), #0(6), #4(7), #1(6), #2(6), #3(6), #0(5), #4(6), #1(5), #2(5), #3(5), #0(4), #4(5), #1(4), #2(4), #3(4), #0(3), #4(4), #1(3), #2(3), #3(3), #0(2), #4(3), #1(2), #2(2), #3(2), #0(1), #4(2), #1(1), #2(1), #3(1), #0(Liftoff!), #4(1), #1(Liftoff!), #2(Liftoff!), #3(Liftoff!), #4(Liftoff!), 

​ Thread的构造器接收 Runable 对象,并在调用 start() 方法之后启动新的线程去执行 Runable中的 run() 方法。输出结果很有意思。我们启动了5个发射前的倒计时任务,“ Waiting for Liftoff ” 在倒计时没完成之前就输出了,这证明现在的任务确实是在新的线程执行的。各个任务的倒计时混杂在一起,说明不同任务的执行线程在被不断的换进换出。

3. 使用Executor

java.util.concurrent 包中的执行器( Executor ),可以帮我们管理Thread对象。 Executor 是一个接口,只有一个方法,就是 execute 。当我们把一个 Runable 交给 Executor 去执行,它可能会启动一个新的线程、或者从线程池中选择一个线程、甚至直接使用当前线程。但是,这些我们都不需要关心,我们只需要选择合适的 Executor 的实现,然后把任务扔给它去执行就好了。

public interface Executor {

    /**
     * Executes the given command at some time in the future.  The command
     * may execute in a new thread, in a pooled thread, or in the calling
     * thread, at the discretion of the {@code Executor} implementation.
     *
     * @param command the runnable task
     * @throws RejectedExecutionException if this task cannot be
     * accepted for execution
     * @throws NullPointerException if command is null
     */
    void execute(Runnable command);
}

先来看一个具体的使用示例。在这个示例中,我们通过 Executors 来创建了一个 线程池 CachedThreadPool。并通过这个线程池来执行5个发射前的倒计时任务。

public class CachedThreadPool {
    public static void main(String[] args) {
        ExecutorService exectorService = Executors.newCachedThreadPool();
        for (int i=0; i<5;++i) {
            exectorService.execute(new LiftOff());
        }
        exectorService.shutdown();
    }
}

在上面的示例中有几个需要解释的概念:

  • Executors:一个工厂和工具类。
  • ExecutorService:有生命周期的 Executor 。也是一个接口,继承于 Executor 。
  • CachedThreadPool:线程池。当新任务过来,会首先找池中有没有可用的线程,没有才新建线程。

在 Executors 中还定义了另外三种线程池:FixedThreadPool 、SingleThreadPool 、 ScheduledThreadPool (也提供了单线程的 ScheduledThreadPool )。FixedThreadPool 线程数量是稳定的,线程创建后不会销毁,达到设定的数量后,不再创建新线程。SingleThreadExecutor 是只能有一个线程的线程池。而 ScheduledThreadPool 可以定时执行任务。现在把上面的示例中的 CachedThreadPool 换成 FixedThreadPool ,最大线程数量为3。

public class FixedThreadPool {
    public static void main(String[] args) {
        ExecutorService exector = Executors.newFixedThreadPool(3);
        for (int i=0; i<5;++i) {
            exector.execute(new LiftOff());
        }
        exector.shutdown();
    }
}

输出结果:
#1(9), #2(9), #0(9), #1(8), #2(8), #0(8), #1(7), #2(7), #0(7), #1(6), #0(6), #2(6), #1(5), #0(5), #2(5), #1(4), #0(4), #2(4), #1(3), #0(3), #2(3), #0(2), #1(2), #0(1), #2(2), #0(Liftoff!), #1(1), #2(1), #1(Liftoff!), #2(Liftoff!), #3(9), #4(9), #3(8), #4(8), #3(7), #4(7), #3(6), #4(6), #3(5), #4(5), #3(4), #4(4), #3(3), #4(3), #3(2), #4(2), #3(1), #4(1), #3(Liftoff!), #4(Liftoff!), 

从输出结果可以看到,只有三个任务在同时执行。后面两个任务等前面的任务执行完成了,才开始执行。

对线程池的进一步研究

来看一下 Executors 中这四种线程池是怎么创建的。

public static ExecutorService newCachedThreadPool() {
    return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
                                  60L, TimeUnit.SECONDS,
                                  new SynchronousQueue<Runnable>());
}

public static ExecutorService newFixedThreadPool(int nThreads) {
    return new ThreadPoolExecutor(nThreads, nThreads,
                                  0L, TimeUnit.MILLISECONDS,
                                  new LinkedBlockingQueue<Runnable>());
}

public static ExecutorService newSingleThreadExecutor() {
    return new FinalizableDelegatedExecutorService
        (new ThreadPoolExecutor(1, 1,
                                0L, TimeUnit.MILLISECONDS,
                                new LinkedBlockingQueue<Runnable>()));
}

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
    return new ScheduledThreadPoolExecutor(corePoolSize);
}
public class ScheduledThreadPoolExecutor
        extends ThreadPoolExecutor
        implements ScheduledExecutorService {
    ...
}       

前面三种线程池,都是直接创建了 ThreadPoolExecutor 类的对象。ScheduledThreadPool 因为要实现定时功能,创建的是 ScheduledThreadPoolExecutor 类的对象。但 ScheduledThreadPoolExecutor 也是继承自ThreadPoolExecutor 。所以我们主要关注一下 ThreadPoolExecutor 。下面这个构造方法是参数最全的一个。

/**
 * Creates a new {@code ThreadPoolExecutor} with the given initial
 * parameters.
 *
 * @param corePoolSize 线程池中维持的线程数量。
 *                     当线程数量不超过这个数时,即使线程处于空闲状态也不会被销毁,会一直等待任务到来。
 *                     但是如果设置 allowCoreThreadTimeOut 为 true,corePoolSize 就不再有效了。
 * @param maximumPoolSize 线程池中线程的最大数量。
 * @param keepAliveTime 当线程数量超过了 corePoolSize 时,多余的线程销毁前等待的时间。
 * @param unit keepAliveTime 的时间单位
 * @param workQueue 用来管理待执行任务的队列。
 * @param threadFactory 创建线程的工厂。
 * @param handler RejectedExecutionHandler 接口的实现对象。用于处理任务被拒绝执行的情况。
 *                被拒绝的原因可能是所有线程正在执行任务而任务队列容量又满了
 */
public ThreadPoolExecutor(int corePoolSize,
                          int maximumPoolSize,
                          long keepAliveTime,
                          TimeUnit unit,
                          BlockingQueue<Runnable> workQueue,
                          ThreadFactory threadFactory,
                          RejectedExecutionHandler handler) {
    ...
}

理解了这些参数,就很容易理解 Executors 中创建的几种线程池。当这几种线程池都不能满足需求的时候,我们直接可以通过 ThreadPoolExecutor 的构造方法来创建一个合适的线程池。那么,ThreadPoolExecutor 是怎么调度线程来执行任务的呢?

从 execute() 方法入手去理解。其中 ctl 只是一个原子操作的 int 型(AtomicInteger类)变量,但可以同时保存线程池状态和线程数量。我在另一篇文章中专门分析了这个 ctl 的实现

public void execute(Runnable command) {
    if (command == null)
        throw new NullPointerException();
 
    int c = ctl.get();
    if (workerCountOf(c) < corePoolSize) {
        if (addWorker(command, true))
            return;
        c = ctl.get();
    }
    if (isRunning(c) && workQueue.offer(command)) {
        int recheck = ctl.get();
        if (! isRunning(recheck) && remove(command))
            reject(command);
        else if (workerCountOf(recheck) == 0)
            addWorker(null, false);
    }
    else if (!addWorker(command, false))
        reject(command);
}

如果线程数量小于核心线程数量,就创建新的线程来执行任务;不然就添加到任务队列中;如果添加到任务队列失败,就创建新的线程来执行;如果创建线程再失败(可能是线程池不再是RUNNING状态,或者线程数量已经达到了最大线程数量),就只能拒绝任务了。

上面说的线程,实际上都通过 Worker 来管理,每个 Worker 对象持有一个线程。而 Woker 实现了 Runable 接口,会在自己的管理的线程中来执行。Worker 的 run() 方法就是直接调用了 runWorker 这个方法。

final void runWorker(Worker w) {
    ...
    Runnable task = w.firstTask;
    w.firstTask = null;
    ...
    while (task != null || (task = getTask()) != null) {
        ...
        task.run();
        ...
    }
    ...
    processWorkerExit(w, completedAbruptly);
}   

如果在创建 Worker 时,就指定了一个任务,会先执行这个任务。后面就是循环,不断地从任务队列获取任务去执行。获取任务时,核心线程会一直等待获取到新的任务。而一般线程会设置一个超时时间,这个时间就是创建线程池时指定的 keepAliveTime。超时之后,就退出循环了,Worker 的使命完成,马上会被释放。有两点要补充一下:

  1. 核心线程和一般线程没有区分,只是去 getTask 时,根据当前线程的数量是否大于核心线程数量来决定要不要一直等待。
  2. 可以设置 allowCoreThreadTimeOut 为 true,让核心线程获取任务时也会超时。

现在我们基本上搞清楚了线程池是如何调度线程来执行任务的。再来回顾一下前面 Executors 中创建的几种线程池。

Executors中创建的CachedThreadPoollExecutor,是用的同步队列,只有当前有线程在等待任务时,才能加入,实际上也不在队列中管理,是直接扔给了执行线程去执行。所以CachedThreadPool中,当新任务到来时,如果线程数小于核心线程数,是直接创建,不然就看当前有没有在等待任务的线程,有就交给该线程执行,没有就创建一个新线程去执行。

Executors中创建的FixedThreadPoolExecutor和SingleThreadlExecutor,都是核心数量等于最大数量,且它们的任务队列是无限容量的。当新任务到来时,如果线程数小于核心线程数,创建新线程去执行,不然就加到任务队列中等待。

最后,研究一下 ScheduledThreadPoolExecutor 是怎么实现定时任务的。ScheduledThreadPoolExecutor 实现了 ScheduledExecutorService 接口中的 schedule 等方法。调用 schedule() 方法时,会把需要定时执行的任务打包在 ScheduledFutureTask 对象中,然后加入到等待执行的队列中去。

public ScheduledFuture<?> schedule(Runnable command, long delay, TimeUnit unit) {
    if (command == null || unit == null)
        throw new NullPointerException();
    RunnableScheduledFuture<?> t = decorateTask(command,
        new ScheduledFutureTask<Void>(command, null,
                                      triggerTime(delay, unit)));
    delayedExecute(t);
    return t;
}

ScheduledThreadPoolExecutor 中用 DelayedWorkQueue 来管理等待执行的任务。添加时,会根据执行时间,把任务排到队列中合适的位置,保证队列中的任务按执行时间先后排列。取出时,取队列头部的任务,如果队列头部没有任务,或者任务的执行时间还没到,就要等待。

private void delayedExecute(RunnableScheduledFuture<?> task) {
    if (isShutdown())
        reject(task);
    else {
        super.getQueue().add(task);
        if (isShutdown() &&
            !canRunInCurrentRunState(task.isPeriodic()) &&
            remove(task))
            task.cancel(false);
        else
            ensurePrestart();
    }
} 

delayedExecute() 方法中,先把任务加入到任务队列中,然后调用 ensurePrestart() 方法去启动一个新线程(线程数量小于限定的核心线程数量才会启动新线程)。这个线程就会去队列中等待任务,任务队列会在任务执行时间到时返回任务给线程去执行。这样就实现了定时任务的执行。

4. 从任务中产生返回值

前面我们用 Runable 来定义任务,但是 Runable 执行完成后不会有返回值。当需要返回值时,可以实现 Callable 接口。Callable 需要通过 ExecutorService 中声明的 submit() 方法去执行。

public interface Callable<V> {
    V call() throws Exception;
}

下面举一个计算年级平均分的例子。为了简化,假定每个班学生人数都是50人。为了计算年级平均分,要让各班去计算各自的总分。每个班计算总分的过程用 Callable 去执行。

private static final int STUDENT_NUM_OF_EACH_CLASS = 50;

static class ClassScoreCaculator implements Callable<Integer> {
    private List<Integer> loadScore() {
        List<Integer> scoreList = new ArrayList<>();
        for (int i = 0; i < STUDENT_NUM_OF_EACH_CLASS; ++i) {
            scoreList.add((int) (Math.random() * 100));
        }
        return scoreList;
    }

    @Override
    public Integer call() throws Exception {
        List<Integer> scoreList = loadScore();
        Integer sum = 0;
        for (Integer score : scoreList) {
            sum += score;
        }
        return sum;
    }
}

public static void main(String[] args) {
    List<Future<Integer>> results = new ArrayList<>();
    ExecutorService executor = Executors.newCachedThreadPool();
    for (int i = 0; i < 12; ++i) {
        results.add(executor.submit(new ClassScoreCaculator()));
    }
    int sumScore = 0;
    for (Future<Integer> result : results) {
        try {
            sumScore += result.get();
        } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
        }
    }
    int average = sumScore / (STUDENT_NUM_OF_EACH_CLASS * 12);
    System.out.print("average score is " + average);
}   

submit() 方法会返回 Future 对象。可以用 get() 方法去获取执行结果,get() 方法会一直阻塞,直到 Callable 执行完成返回结果。如果不希望阻塞,可以先用 isDone() 方法查询是否执行完成。也使用带超时时间参数的 get() 方法。注意到 get() 方法会抛出两种异常:InterruptedException 和 ExecutionException 。其中,InterruptedException 是调用 Future 对象的 cancel() 方法去取消任务时,可能会中断线程而抛出的异常。而 ExecutionException ,是执行任务过程中的异常。因为,Callable 的 call() 方法是会抛出异常的,这个异常会被封装到 ExecutionException 中抛出。

5. 休眠

当我们需要任务暂停一段时间,可以使用线程的 sleep() 方法。在线程休眠过程中,可能会有其他线程尝试中断当前线程,这时 sleep() 方法会抛出 InterruptedException ,结束休眠。我们可以在 catch 到中断异常之后,选择尽快结束当前线程的执行任务,当然也可以忽略,选择继续执行。

public static native void sleep(long millis) throws InterruptedException;
public static void sleep(long millis, int nanos) throws InterruptedException {...}

6. 优先级

线程的优先级表示线程的重要性,线程调度器倾向于让优先级高的线程先执行。可以用 Thread 的 getPriority() 方法读取线程的优先级,通过 setPriority() 方法可以修改线程的优先级。目前 Java 中的线程是映射到底层操作系统的线程,通过底层操作系统来实现的。所以优先级也被映射到底层操作系统中的线程优先级。但是,不同操作系统的优先级级别数量、策略都有所不同,Java 中的 10 个优先级并不能映射得很好。Thinking in Java 书上建议,调整优先级时,只使用 MAX_PRIORITY、NORM_PRIORITY 和 MIN_PRIORITY 三种级别。由于不同操作系统的线程调度策略不一样,因此我们在开发时不应该依赖于线程的执行顺序。

7. 让步

通过 Thread 的 yield() 方法,可以给线程调度器一个建议:当前线程的工作告一段落,可以让出 CPU 给其他线程使用了。当然,这只是一个建议,没有任何机制能保证它一定被采纳。所以,我们在开发时也不应该依赖于 yield() 方法。

8. 后台线程

后台(daemon)线程,也有叫守护线程的。关于后台线程需要了解的主要有三点:

  • 当所有非后台线程结束,程序也就会结束,所有的后台进程都被杀死。因此,不要把必须执行的任务放到后台线程中。
  • 通过 setDaemon(true) 可以把线程标记为后台线程。这个方法要在线程开始运行之前调用,不然会抛出异常。
  • 后台线程中创建的线程会被自动设成后台线程。原理是线程初始化的时候会获取当前线程的 daemon ,来设置自己的 daemon 。

下面看一个使用后台线程的例子。

public class DaemonThreadStudy {
    private static class DaemonThreadFactory implements ThreadFactory {
        @Override
        public Thread newThread(Runnable r) {
            Thread thread = new Thread(r);
            thread.setDaemon(true);
            return thread;
        }
    }

    private static class DaemonFromFactory implements Runnable {
        @Override
        public void run() {
            try {
                while (true) {
                    Thread.sleep(100);
                    System.out.println(Thread.currentThread() + " " + this);
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        Executor executor = Executors.newCachedThreadPool(new DaemonThreadFactory());
        for (int i = 0; i < 10; i++) {
            executor.execute(new DaemonFromFactory());
        }
        System.out.println("All daemons started");
        Thread.sleep(100);
    }
}

9. join

一个线程如果要等待另一个线程执行完成,可以调用另一个线程的 join() 方法。调用 join() 方法之后,当前线程将被挂起,等待另一个线程执行结束。join() 方法也有一个带等待时间参数的重载版本,等待时间到了后,不管等待的线程是否执行完成都会返回。来看一个使用 join() 方法的例子。

public static void main(String[] args) {
    Thread sleeper = new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                System.out.println(Thread.currentThread().getName() + " interrupted");
            }
            System.out.println(Thread.currentThread().getName() + " has awakend");
        }
    }, "Sleeper");
    Thread joiner = new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                sleeper.join();
            } catch (InterruptedException e) {
                System.out.println(Thread.currentThread().getName() + " interrupted");
            }
            System.out.println(Thread.currentThread().getName() + " join completed");
        }
    }, "Joiner");
    sleeper.start();
    joiner.start();
}

在上面的例子中,sleeper 休眠 10 秒,而 joiner 会一直等待 sleeper 执行完成。注意,join() 方法和 sleep() 方法一样会抛出中断异常。也就是说,线程在等待时,也可以通过调用 interrupt 方法去中断它。

来看一下 join() 方法的实现。

public final synchronized void join(long millis) throws InterruptedException {
    long base = System.currentTimeMillis();
    long now = 0;

    if (millis < 0) {
        throw new IllegalArgumentException("timeout value is negative");
    }

    if (millis == 0) {
        while (isAlive()) {
            wait(0);
        }
    } else {
        while (isAlive()) {
            long delay = millis - now;
            if (delay <= 0) {
                break;
            }
            wait(delay);
            now = System.currentTimeMillis() - base;
        }
    }
}

实际上就是调用了线程对象的 wait() 方法。循环判断线程是否执行结束,没结束就继续 wait()。如果设置了超时时间的话,会在时间到了之后结束 wait(),退出循环。按照注释中的说法,Thread 对象在 terminate 的时候,会调用 notifyAll() 。这样,wait() 方法就能返回,join() 方法也就执行完了。为什么需要设个循环去判断 isAlive() 呢,因为我们有可能在程序的其他地方去调用被等待的线程对象的 notify() 和 notifyAll() 方法。如果没有循环的,join() 就会直接返回,不会等到线程执行结束了。

测试在其他地方调用被等待的线程的 notify() 方法时,还发现调用一个对象的 wait()、notify()、notifyAll() 等方法都需要先成为这个对象的 monitor 所有者,不然会抛出 IllegalMonitorStateException 异常。成为一个对象的 monitor 所有者有三种方法:

  • 在这个对象的 synchronize 的方法中
  • 在 synchronize 这个对象的代码块中
  • 如果这个对象是 Class 类的对象,可以在类的静态的 synchronize 的方法中

其实三种方法本质上都是一样的,就是在调用 wait()、notify() 方法之前,得先对对象做 synchronize 。前面两种就不用说了。第三种方法,由于 Class 类的特殊性,类的静态的 synchronize 的方法,实际上就是对 Class 对象做的 synchronize。

10. 线程组

ThreadGroup ,这个东西没太大作用。看了书和很多资料,都说没什么意义。看了 ThreadGroup 类的源码,就是持有一个线程数组和一个线程组数组,方便进行统一操作,比如:interrupt() 方法。除此之外,还能通过 activeCount() 方法获取一下线程组内的线程数量。有些作用的是,ThreadGroup 可以对线程运行中没有被捕获的异常做处理。

11. 捕获异常

由于线程的特性,我们无法捕获从线程中逃逸的异常。一旦异常逃出任务的 run() 方法,就会向外传播 。我们需要用特殊的方式捕获线程中逃出的异常。在 Java 1.5 以前只能用线程组来捕获,在 1.5 版本之后,就有更好的方式可以处理了。

class ExceptionRunable implements Runnable {
    @Override
    public void run() {
        throw new RuntimeException();
    }
}

class NativeExceptionHandling {
    public static void main(String[] args) {
        try {
            Executor executor = Executors.newCachedThreadPool();
            executor.execute(new ExceptionRunable());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

在上面的示例中,我们执行了一个会抛出异常的任务,并尝试用 try catch 去捕获异常,很显然,这是没有作用的,因为异常是在新的线程中抛出的。那么,我们改怎么去捕获这种异常呢?Java 1.5 引入了一个新的接口 Thread.UncaughtExceptionHandler,我们可以给线程设置一个异常处理器,去处理没有被捕获的异常。

class MyUncaughtExceptionHandler implements Thread.UncaughtExceptionHandler {

    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.println("caught " + e);
    }
}

class HandlerThreadFactory implements ThreadFactory {

    @Override
    public Thread newThread(Runnable r) {
        Thread thread = new Thread(r);
        thread.setUncaughtExceptionHandler(new MyUncaughtExceptionHandler());
        return thread;
    }
}

class CaptureUncaughtException {
    public static void main(String[] args) {
        Executor executor = Executors.newCachedThreadPool(new HandlerThreadFactory());
        executor.execute(new ExceptionRunable());
    }
}

这里我们使用了 HandlerThreadFactory 来创建线程,通过调用 Thread 的成员方法 setUncaughtExceptionHandler() 给每个线程设置了 UncaughtExceptionHandler 。线程运行中没有被捕获的异常,会被扔给 UncaughtExceptionHandler 来处理,而不会向外传递。

进一步研究,看异常是怎么被传到处理器中的。先看 Thread 类中的 dispatchUncaughtException() 方法,这个方法是由 JVM 去调用的。之前的流程应该就是线程执行任务后,有没捕获的异常,然后 JVM 调用线程的 dispatchUncaughtException() 方法来处理异常。然后,获取异常处理器,把异常交给异常处理器的 uncaughtException() 方法。如果该线程对象设置了异常处理器,就用自身的,否则就交给线程组处理(ThreadGroup 也实现了 UncaughtExceptionHandler 接口)。

public class Thread {
    ...
    private volatile UncaughtExceptionHandler uncaughtExceptionHandler;
    private static volatile UncaughtExceptionHandler defaultUncaughtExceptionHandler;
    private ThreadGroup group;
    ...
    /**
     * Dispatch an uncaught exception to the handler. This method is
     * intended to be called only by the JVM.
     */
    private void dispatchUncaughtException(Throwable e) {
        getUncaughtExceptionHandler().uncaughtException(this, e);
    }

    public UncaughtExceptionHandler getUncaughtExceptionHandler() {
        return uncaughtExceptionHandler != null ?
            uncaughtExceptionHandler : group;
    }
    ...
}

线程组中的处理流程是:首先找父线程组的处理方法;其次找线程中设置的默认异常处理器;都找不到就直接打印异常堆栈。

public class ThreadGroup {
    ...
    public void uncaughtException(Thread t, Throwable e) {
        if (parent != null) {
            parent.uncaughtException(t, e);
        } else {
            Thread.UncaughtExceptionHandler ueh =
                Thread.getDefaultUncaughtExceptionHandler();
            if (ueh != null) {
                ueh.uncaughtException(t, e);
            } else if (!(e instanceof ThreadDeath)) {
                System.err.print("Exception in thread \""
                                 + t.getName() + "\" ");
                e.printStackTrace(System.err);
            }
        }
    }
    ...
}

总结一下,线程执行中没捕获的异常优先扔给线程对象中设置的异常处理器,其次给线程组,如果都没处理,会看是否设置了 Thread 类的默认异常处理器。

看到这里,我产生了一个疑问,按照这种机制,没捕获的异常最多是打个错误信息,而不会导致程序 crash 。那么,为什么在 android 中,异常会导致应用 crash 呢。原来,Android 在所有进程启动时,都给 Thread 设置了 defaultUncaughtExceptionHandler ,遇到异常时会让应用 crash 。想了解更多内容,请看这篇文章 理解Android Crash处理流程

12. 结语

这篇文章是我阅读《 Thinking In Java 》书中并发一章第2节,并结合源码以及测试的学习记录。对 Java 基础线程机制的学习到此就告一段落了。下一篇文章学习多线程开发的两个主要问题的解决:Java多线程开发(二)| 多线程的竞争与协作

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容