多线程

zszdevelop大约 7 分钟

多线程

1. 什么是线程和进程

1.1 什么是进程

进程是程序的一次执行过程,是操作系统分配资源的最小单位。系统运行一个程序即是一个进程从创建,运行到消亡的过程

在java中,当我们启动 main 函数时其实就是启动了一个 JVM 的进程,而main函数所在的线程就是这个进程中的一个线程,也称主线程

1.2 什么是线程

他是操作系统运算调度(程序执行)的最小单位,一个进程包含一个或多个线程(重点是调度)

与进程不同的是同类的多个线程共享堆和方法区资源,但每个线程有自己的程序计数器、虚拟机栈和本地方法栈。所以系统在产生一个线程,或者在各个线程切换工作时,负担要比进程小得多

2. 线程与进程的关系与区别及优缺点

2.1 进程与线程的关系

一个进程包含多个线程

2.2 进程与线程的区别

  1. 进程是资源分配的最小单位,线程是程序执行的最小单位

  2. 线程之间的通信更方便,因为同一进程下的线程共享全局变量,静态变量等数据,而进程之间的通信需要以IPC的方式进行通信

  3. 进程有自己独立的地址空间,创建销毁开销大

    线程是共享进程中的数据的,使用相同的地址空间,创建、切换、销毁开销小

  4. 各进程是独立的,而线程则不一定。因为同一进程中的线程极有可能相互影响

3. 进程与线程的内存区域

下图是java 的内存区域

image-20190919231017632
image-20190919231017632

从上图可以看出:一个进程中可以有多个线程,多个线程共享方法区(JDK1.8之后的元空间)资源,但是每个线程有自己的程序计数器、虚拟机栈和本地方法栈

3.1 从内存区域角度来分析进程与线程区别

  1. 线程是进程划分成更小的运行单位
  2. 线程和进程最大的不同在于各进程是独立的,而线程则不一定。因为同一进程中的线程极有可能相互影响
  3. 线程执行开销小,但不利于资源的管理和保护,而进程相反

3.2 程序计数器为什么是私有的?

程序计数器主要有下面两个作用:

  1. 字节码解释器通过改变程序计数器来依次读取指令,从而实现代码的流程控制,如:顺序执行、选择、循环、异常处理。
  2. 在多线程的情况下,程序计数器用于记录当前线程执行的位置,从而当线程被切换回来的时候能够知道该线程上次运行到哪儿了。

需要注意的是,如果执行的是 native 方法,那么程序计数器记录的是 undefined 地址,只有执行的是 Java 代码时程序计数器记录的才是下一条指令的地址。

所以,程序计数器私有主要是为了线程切换后能恢复到正确的执行位置

3.3 虚拟机栈和本地方法栈为什么是私有的

  • 虚拟机栈: 每个 Java 方法在执行的同时会创建一个栈帧用于存储局部变量表、操作数栈、常量池引用等信息。从方法调用直至执行完成的过程,就对应着一个栈帧在 Java 虚拟机栈中入栈和出栈的过程。
  • 本地方法栈: 和虚拟机栈所发挥的作用非常相似,区别是: 虚拟机栈为虚拟机执行 Java 方法 (也就是字节码)服务,而本地方法栈则为虚拟机使用到的 Native 方法服务。 在 HotSpot 虚拟机中和 Java 虚拟机栈合二为一。

所以,为了保证线程中的局部变量不被别的线程访问到,虚拟机栈和本地方法栈是线程私有的。

3.4. 一句话简单了解堆和方法区

堆和方法区是所有线程共享的资源,其中堆是进程中最大的一块内存,主要用于存放新创建的对象 (所有对象都在这里分配内存),方法区主要用于存放已被加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

4. 并发和并行

  • 并发: 同一时间段,多个任务都在执行 (单位时间内不一定同时执行);
  • 并行: 单位时间内,多个任务同时执行。

5. 为什么要使用多线程

  • 从计算机底层来说:

    1. 线程比作轻量级的进程,是程序执行的最小单位,线程间的切换和调度成本远远小于进程

    2. 多核CPU时代意味着多个线程可以同时运行,这减少了线程上下文切换的开销

  • 从当代互联网发展趋势来说:

    1. 现在的系统动不动就要求百万级甚至千万级的并发量,而多线程并发编程正式开发高并发系统的基础
    2. 利用好多线程机制可以大大提高系统整体的并发能力以及性能

6. 多线程可能带来的问题

并发编程的目的就是为了能提高程序的执行效率提高程序运行速度,但是并发编程并不总是能提高程序运行速度的,而且并发编程可能会遇到很多问题,比如:

  • 内存泄漏
  • 上下文切换
  • 死锁
  • 还有受限于硬件和软件的资源闲置问题。

7. 什么是上下文切换

当前任务在执行完 CPU 时间片切换到另一个任务之前会先保存自己的状态,以便下次再切换到回这个任务时,可以再加载这个任务的状态。任务从保存到再加载的过程就是一次上下文切换

7.1 为什么需要上下文切换

多线程编程中一般线程的个数都大于CPU核心的个数,而一个CPU核心在任意时刻只能被一个线程使用,为了让这些线程都能得到有效的执行,CPU 采取的策略是为每个线程分配时间片并轮转的形式,当一个线程的时间片用完的时候就会重新处于就绪状态让其他线程使用,这个过程就是上下文切换

7.2 为什么说上下文切换消耗时间

上下文切换通常是计算密集型的。也就是说,它需要相当可观的处理器时间,在每秒几十上百次的切换中,每次切换都需要纳秒量级的时间。所以,上下文切换对系统来说意味着消耗大量的 CPU 时间,事实上,可能是操作系统中时间消耗最大的操作。

8. 为什么我们不能直接调用 run() 方法?

调用 start 方法方可启动线程并使线程进入就绪状态,而 run 方法只是 thread 的一个普通方法调用,还是在主线程里执行。

9. 什么是线程安全

多个线程访问同一个对象时,如果不用考虑这些线程在运行时环境下的调度和交替运行,也不需要进行额外的同步,或者在调用方进行任何其他的协调操作,调用这个对象的行为都可以获取正确的结果,那这个对象是线程安全的。 《深入java虚拟机》

Loading...