并发、并行傻傻分不清楚?线程的一生都经历些什么?

人生一切难题,知识给你答案

温馨提示:阅读本文需要5-6分钟(少量代码)


今天,我们来解决一个问题:

并发、并行傻傻分不清楚?线程的一生都经历些什么?

人生一切难题,知识给你答案。


==并发与并行==

并发是指在某个时间段内,多任务交替处理的能力。并行是指同时处理多任务的能力。

案例:商场销售员需要面对多位顾客时,有的顾客问价格,有的顾客问质量,这时销售员需要不断的回答顾客,中间不停的切换话题并记住之前的话题,方便回过头回答,这种方式可以理解为并发,如果有多位销售员对应多位顾客,多位销售员同时回答顾客问题,这种方式就是并行。

==线程的生命周期==

线程是CPU调度和分配的基本单位。线程可以拥有自己的操作栈、程序计数器、局部变量表等资源,它与同一进程内的其他线程共享该进程的所有资源。

线程的创建有三种方式,第一种是继承自Thread类,如下所示:

public class HandlerThread extends Thread {
    
    @Override
    public void run() {
        
    }
    
}
复制代码

第二种方式是实现Runnable接口,如下所示:

public class HandlerThread implements Runnable {

    @Override
    public void run() {

    }

}
复制代码

推荐使用第二种方式,因为继承Thread类不符合里氏替换原则,实现Runnable接口可以使编程更加灵活,对外暴露的细节比较少。

第三种方式使用Callable接口,如下所示:

public class Demo implements Callable<String> {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Callable<String> callable=new Demo();
        FutureTask<String> futureTask=new FutureTask<>(callable);
        new Thread(futureTask).start();
        System.out.println(futureTask.get());
    }

    @Override
    public String call() throws Exception {
        return "demo";
    }
}
复制代码

Callable和Future接口的区别在于:

  • Callable规定的方法是call(),而Runnable规定的方法是run().
  • Callable的任务执行后可返回值,而Runnable的任务是不能返回值的。
  • call()方法可抛出异常,而run()方法是不能抛出异常的。
  • 运行Callable任务可拿到一个Future对象, Future表示异步计算的结果。它提供了检查计算是否完成的方法,以等待计算的完成,并检索计算的结果。通过Future对象可了解任务执行情况,可取消任务的执行,还可获取任务执行的结果。
  • Callable是类似于Runnable的接口,实现Callable接口的类和实现Runnable的类都是可被其它线程执行的任务。

线程的生命周期有五种状态,如下图:

线程五种状态.jpg

线程的生命周期的状态有:新建状态、就绪状态、运行状态、阻塞状态以及终止状态。

  • new,新建状态,是线程被创建且未启动的状态。
  • Runnable,就绪状态,是调用start()方法之后运行之前的状态,start()不能被多次调用,否则会抛出IllegalStateException异常。
  • Running,运行状态,是run()正在执行时线程的状态。线程可能会由于某些因素而推出Running,如时间、异常、锁、调度等。
  • Blocked,阻塞状态,进入此状态的情况包括:同步阻塞(锁被其他线程占用)、主动阻塞(调用Thread的某些方法,主动让出CPU执行权,比如sleep()、join()等)以及等待阻塞(执行了wait()方法)。
  • Dead,终止状态,是run()执行结束,或因异常退出后的状态。

838794-506ddad529df4cd4.webp.jpg

猜你喜欢

转载自juejin.im/post/5c4fa64df265da6179752fe7