Java面试核心基础知识之并发编程

Java面试核心基础知识之并发编程


文章目录

  • Java面试核心基础知识之并发编程
  • 一、基础知识
    • 1、并发编程的优点?
    • 2、并发编程的缺点?
    • 3、并发编程三要素?
    • 4、出现线程安全原因?
    • 5、并行和并发区别?
    • 6、什么是多线程?
    • 7、什么是进程和线程?
    • 8、进程与线程的区别?
    • 9、什么是上下文切换?
    • 10、用户线程和守护线程有什么区别?
    • 11、什么是线程死锁?
    • 12、形成死锁的四个必要条件?
    • 13、如何避免线程死锁?
    • 14、创建线程有哪几种方式?
      • 1)继承Thread类
      • 2)实现 Runnable 接口
      • 3)实现 Runnable 接口
      • 4)使用Executors工具类创建线程池
    • 15、为什么不能直接调用run()方法?
    • 16、线程的 start()和run()有什么区别?
    • 17、runnable和callable有什么区别?
    • 18、线程的生命周期和5种基本状态?
  • 总结


一、基础知识

1、并发编程的优点?

  • 充分利用多核CPU的计算能力:通过并发编程的形式可以将多核CPU的计算能力发挥到极致,性能得到提升
  • 方便进行业务拆分,提升系统并发能力和性能:在特殊的业务场景下,先天的就适合于并发编程。现在的系统动不动就要求百万级甚至千万级的并发量,而多线程并发编程正是开发高并发系统的基础,利用好多线程机制可以大大提高系统整体的并发能力以及性能。面对复杂业务模型并行程序会比串行程序更适应业务需求,而并发编程更能吻合这种业务拆分

2、并发编程的缺点?

  • 并发编程的目的就是为了能提高程序的执行效率,提高程序运行速度,但是并发编程并不总是能提高程序运行速度的,而且并发编程可能会遇到很多问题,比如:内存泄漏、上下文切换、线程安全、死锁等问题。

3、并发编程三要素?

  • 原子性:原子,即一个不可再被分割的颗粒。原子性指的是一个或多个操作要么全部执行成功要么全部执行失败。
  • 可见性:一个线程对共享变量的修改,另一个线程能够立刻看到。
  • 有序性:程序执行的顺序按照代码的先后顺序执行。(处理器可能会对指令进行重排序)

4、出现线程安全原因?

  • 线程切换带来的原子性问题
  • 缓存导致的可见性问题
  • 编译优化带来的有序性问题

解决方案

  • JDK Atomic开头的原子类、synchronized、LOCK,可以解决原子性问题
  • synchronized、volatile、LOCK,可以解决可见性问题
  • Happens-Before 规则可以解决有序性问题

5、并行和并发区别?

  • 并行:单位时间内,多个处理器或多核处理器同时处理多个任务,是真正意义上的“同时进行”。
  • 并发:多个任务在同一个CPU核上,按细分的时间片轮流(交替执行,从逻辑
    上来看那些任务是同时执行。
  • 串行:有n个任务,由一个线程按顺序执行。由于任务、方法都在一个线程执行所以不存在线程不安全情况,也就不存在临界区的问题。

举个例子
并行:两个检票窗口排了两队人,每个窗口各检一队
并发:一个检票窗口排两队人,检票员对两队交替检票
串行:一个检票窗口排了一队人,检票员挨个检票

6、什么是多线程?

  • 多线程是指程序中包含多个执行流,即在一个程序中可以同时运行多个不同的线程来执行不同的任务。
  • 优点:可以提高CPU的利用率。在多线程程序中,一个线程必须等待的时候,CPU可以运行其它的线程而不是等待,这样就大大提高了程序的效率。也就是说允许单个程序创建多个并行执行的线程来完成各自的任务。
  • 缺点
    1)线程也是程序,所以线程需要占用内存,线程越多占用内存也越多;
    2)多线程需要协调和管理,所以需要CPU时间跟踪线程;
    3)线程之间对共享资源的访问会相互影响,必须解决竞用共享资源的问题;

7、什么是进程和线程?

  • 进程:一个在内存中运行的应用程序。每个进程都有自己独立的一块内存空间,一个进程可以有多个线程,比如在Windows系统中,一个运行的xx.exe就是一个进程。
  • 线程:进程中的一个执行任务(控制单元),负责当前进程中程序的执行。一个进程至少有一个线程,一个进程可以运行多个线程,多个线程可共享数据。

8、进程与线程的区别?

线程具有许多传统进程所具有的特征,故又称为轻型进程(Light—WeightProcess)或进程元;而把传统的进程称为重型进程(Heavy—Weight Process),它相当于只有一个线程的任务。在引入了线程的操作系统中,通常一个进程都有若干个线程,至少包含一个线程。

  • 根本区别:进程是操作系统资源分配的基本单位,而线程是处理器任务调度和执
    行的基本单位
  • 资源开销:每个进程都有独立的代码和数据空间(程序上下文),程序之间的切换会有较大的开销;线程可以看做轻量级的进程,同一类线程共享代码和数据空间,每个线程都有自己独立的运行栈和程序计数器(PC),线程之间切换的开销小。
  • 包含关系:如果一个进程内有多个线程,则执行过程不是一条线的,而是多条线(线程)共同完成的;线程是进程的一部分,所以线程也被称为轻权进程或者轻量级进程。
  • 内存分配:同一进程的线程共享本进程的地址空间和资源,而进程之间的地址空
    间和资源是相互独立的
  • 影响关系:一个进程崩溃后,在保护模式下不会对其他进程产生影响,但是一个
    线程崩溃整个进程都死掉。所以多进程要比多线程健壮,
  • 执行过程:每个独立的进程有程序运行的入口、顺序执行序列和程序出口。但是线程不能独立执行,必须依存在应用程序中,由应用程序提供多个线程执行控制,两者均可并发执行

9、什么是上下文切换?

  • 多线程编程中一般线程的个数都大于 CPU 核心的个数,而一个CPU 核心在任意时刻只能被一个线程使用,为了让这些线程都能得到有效执行,CPU 采取的策略是为每个线程分配时间片并轮转的形式。当一个线程的时间片用完的时候就会重新处于就绪状态让给其他线程使用,这个过程就属于一次上下文切换。
  • 概括来说就是:当前任务在执行完 CPU 时间片切换到另一个任务之前会先保存自己的状态,以便下次再切换回这个任务时,可以再加载这个任务的状态。任务从保存到再加载的过程就是一次上下文切换
  • Linux 相比与其他操作系统(包括其他类 Unix 系统)有很多的优点,其中有一
    项就是,其上下文切换和模式切换的时间消耗非常少。

10、用户线程和守护线程有什么区别?

  • 用户线程:运行在前台,执行具体的任务,如程序的主线程、连接网络的子线程等都是用户线程
  • 守护线程:运行在后台,为其他前台线程服务。也可以说守护线程是JVM中非守护线程的“佣人”。一旦所有用户线程都结束运行,守护线程会随JVM一起结束工作。
    main 函数所在的线程就是一个用户线程,main函数启动的同时在JVM内部
    同时还启动了好多守护线程,比如垃圾回收线程。
    比较明显的区别之一是用户线程结束,JVM退出,不管这个时候有没有守护线
    程运行。而守护线程不会影响JVM 的退出。

注意事项

  • setDaemon(true)必须在start()方法前执行,否则会抛出IlegalThreadStateException异常
  • 在守护线程中产生的新线程也是守护线程
  • 不是所有的任务都可以分配给守护线程来执行,比如读写操作或者计算
    逻辑
  • 守护(Daemon)线程中不能依靠finally块的内容来确保执行关闭或清理资源的逻辑。因为我们上面也说过了一旦所有用户线程都结束运行,守护线程会随JVM一起结束工作,所以守护(Daemon)线程中的finally语句块可能无法被执行。

11、什么是线程死锁?

  • 百度百科:死锁是指两个或两个以上的进程(线程)在执行过程中,由于竞争资源或者由于彼此通信而造成的一种阻塞的现象,若无外力作用,它们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程(线程)称为死锁进程(线程)。
  • 多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。

如下图所示:线程A持有资源2,线程B持有资源1,他们同时都想申请对方
的资源,所以这两个线程就会互相等待而进入死锁状态。

Java面试核心基础知识之并发编程_第1张图片

12、形成死锁的四个必要条件?

  • 互斥条件:线程(进程)对于所分配到的资源具有排它性,即一个资源只能被一个线程(进程)占用,直到被该线程(进程)释放
  • 请求与保持条件:一个线程(进程)因请求被占用资源而发生阻塞时,对已获得的资源保持不放。
  • 不剥夺条件:线程(进程)已获得的资源在末使用完之前不能被其他线程强行剥夺,只有自己使用完毕后才释放资源。
  • 循环等待条件:当发生死锁时,所等待的线程(进程)必定会形成一个环路(类似于死循环),造成永久阻塞

13、如何避免线程死锁?

  • 我们只要破坏产生死锁的4个条件中的其中一个

你可能感兴趣的:(Java,java,面试,开发语言)