多线程基础知识,常被面试官挂在嘴边的问题

简介: 本篇文章主要围绕为什么使用多线程,进程、线程、管程 (monitor 监视器),,多线程并行和并发的区别,synchronized 和 lock 的区别,线程实现方式,线程的生命周期,线程同步的这部分内容进行讲解, 感兴趣的大佬戳进来

@[toc]

在这里插入图片描述

多线程概述

🥝为什么使用多线程
  1. 摩尔定律失效 (硬件方面):
  • 集成电路上可以容纳的晶体管数目在大约每经过 18 个月便会增加一倍,可是从 2003 年开始 CPU

    • 主频已经不再翻倍,而是采用多核而不是更快的主频 在主频不再提高且核数不断增加的情况下,要想让程序更快就要用到并行或并发编程
  1. 高并发系统,异步 + 回调的生产需求 (软件方面)
🥝进程、线程、管程 (monitor 监视器)

比较官方解释:进程是系统资源分配的单位,线程是 cpu 调度的单位。

线程就是程序执行的一条路径,一个进程中可以包含多条线程,多线程并发执行可以提高程序的效率,可以同时完成多项工作。

举例 : 进程是一个工厂,占用着一定的空间资源,在里边有很多条生产线,生产线上有很多工人。生产线可以看作是 cpu 核数,工人可以看作是线程。

管程:Monitor (监视器),也就是我们平时所说的锁

  1. Monitor 其实是一种同步机制,它的义务是保证 (在同一时间) 只有一个线程可以访问被保护的数据和代码
  2. JVM 中同步时基于进入和退出的监视器对象 (Monitor,管程),每个对象实例都有一个 Monitor 对象。
  3. Monitor 对象和 JVM 对象一起销毁,底层由 C 来实现
🥝多线程并行和并发的区别

并行就是两个任务同时运行,就是甲任务进行的同时,乙任务也在进行 (需要多核 CPU);
并发是指两个任务都请求运行,而处理器只能接收一个任务,就是把这两个任务安排轮流进行,由于时间间隔较短,使人感觉两个任务都在运行;

简单说:并行就是在某一个时间点上,cpu 在同时执行两个任务。并发就是在某一个时间段内,cpu 在有序执行两个任务。

wait | sleep 的区别

功能都是当前线程暂停,有什么区别?

wait 放开手去睡,放开手里的锁;
wait 是 Object 类中的方法
sleep 握紧手去睡,醒了手里还有锁;
sleep 是 Thread 中的方法
🥝synchronized 和 lock 的区别

1. 原始构成

  1. synchronized 是关键字属于 JVM 层面, monitor 对象,每个 java 对象都自带了一个 monitor ,需要拿到 monitor 对象才能做事情。monitorenter (底层是通过 monitor 对象来完成,其实 wait/notify 等方法也依赖 monitor 对象,只能在同步块或方法中才能调用 wait/notify 等方法) 进入 ,monitorexit:退出;
  2. lock 是 api 层面的锁,主要使用 ReentrantLock 实现

2. 使用方法

  1. synchronized 不需要用户手动释放锁,当 synchronized 代码完成后系统会自动让线程释放对锁的占用

  2. ReentrantLock 则需要用户手动释放锁若没有主动释放锁,就有可能会导致死锁的现象

  3. 等待是否可中断?
  1. synchronized 不可中断,除非抛出异常或者正常运行完成
    1. ReentrantLock 可中断 (设置超时时间 tryLock (long timeout,TimeUnit unit),调用 interrupt 方法中断)
  • 加锁是否公平
    1. synchronized 非公平锁
    2. ReentrantLock 两者都可以,默认是非公平锁,构造方法可以传入 boolean 值, true 为公平锁, false 为非公平锁
  • 锁绑定多个 Condition
    1. synchronized 没有
    2. ReentrantLock 用来实现分组唤醒需要唤醒线程们,可以精确唤醒, 而不是像 synchronized 要么随机唤醒一个,要么多个
      🥝线程实现方式
  • thread 实现方式一: 继承 Thread thread
  • 实现方式二: 实现 Runnable thread
  • 实现方式三: 实现Callable thread
  • 实现方式四: 线程池

在这里插入图片描述

🥝常用 API
♟️线程名称

设置

1. Thread thread = new Thread(new ThreadTwo(),"implRunable");//构造
2. thread.setName("implRunable");//setter

获取

thread.getName();
♟️线程优先级

线程有两种调度模型:

  • 分时调度模式:所有线程轮流使用 CPU 的使用权,平均分配每个线程占有 CPU 的时间片
  • 抢占式调度模型:优先让优先级高的线程使用 CPU , 如果线程的优先级相同,那么会随机选择一个,优先级高的线程获取的 CPU 时间片相对多一些 [Java 使用的是抢占式调度模型]

设置和获取线程优先级:

thread.setPriority(int newPriority);//设置
thread.getPriority();//获取

线程默认优先级是 5;线程优先级范围是:1-10; 线程优先级高仅仅表示线程获取的 CPU时间的几率高,但是要在次数比较多,或者多次运行的时候才能看到效果。

♟️线程控制
  1. sleep (long millis) : 使当前正在执行的线程停留 (暂停执行) 指定的毫秒数 (休眠线程)
    1. join () : 当前线程暂停,等待指定的线程执行结束后,当前线程再继续 (相当于插队加入),void join (int
      millis): 可以等待指定的毫秒之后继续 (相当于插队,有固定的时间)
    2. yield () : 让出 cpu 的执行权 (礼让线程)
    3. setDaemon (boolean on) : 将此线程标记为守护线程,当运行的线程都是守护线程时,Java 虚拟机将退出(守护线程)
      1. 守护线程是区别于用户线程,用户线程即我们手动创建的线程,而守护线程是程序运行的时候在后台提供一种通用服务的线程。垃圾回收线程就是典型的守护线程
      2. 守护线程拥有自动结束自己生命周期的特性,非守护线程却没有。如果垃圾回收线程是非守护线程,当 JVM 要退出时,由于垃圾回收线程还在运行着,导致程序无法退出,这就很尴尬。这就是为什么垃圾回收线程需要是守护线程
      3. t1.setDaemon (true) 一定要在 start () 方法之前使用
🥝线程的生命周期
  • 新建:就是刚使用 new 方法,new 出来的线程

  • 就绪:就是调用的线程的 start () 方法后,这时候线程处于等待 CPU 分配资源阶段,谁先抢的 CPU 资源,谁开始执行

  • 运行:当就绪的线程被调度并获得 CPU 资源时,便进入运行状态,run 方法定义了线程的操作和功能

  • 阻塞:在运行状态的时候,可能因为某些原因导致运行状态的线程变成了阻塞状态。比如 sleep ()、wait () 之后线程就处于了阻塞状态,这个时候需要其他机制将处于阻塞状态的线程唤醒,比如调用 notify 或者 notifyAll ()
    方法。唤醒的线程不会立刻执行 run 方法,它们要再次等待 CPU 分配资源进入运行状态

  • 销毁:如果线程正常执行完毕后或线程被提前强制性的终止或出现异常导致结束,那么线程就要被销毁,释放资源

🥝线程同步

synchronized

为什么出现问题?(这也是我们判断多线程程序是否会有数据安全问题的标准)

1.是否有多线程坏境
2.是否有共享数据
3.是否有多条语句操作共享数据

如何解决多线程安全问题?

1.基本思想:让程序没有安全问题的坏境
2.把多条语句操作的共享数据的代码给锁起来,让任意时刻只能有一个线程执行即可

怎么锁起来呢?

synchronized (任意对象){} : 相当于给代码加锁了,任意对象就可以看成是一把锁

同步的好处和弊端

好处:解决了多线程的数据安全问题
弊端:当线程很多时,因为每个线程都会判断同步上的锁,这是很浪费资源的,无形中会降低程序的运行效率

同步方法

同步方法:就是把 synchronized 关键字加到方法上

同步方法的锁对象是什么呢?this

格式:修饰符 synchronized 返回值类型 方法名 (方法参数){ }

同步静态方法:就是把 synchronized 关键字加到静态方法上

格式:修饰符 static synchronized 返回值类型 方法名 (方法参数){ }

同步静态方法的锁对象是什么呢?类名.class

目录
相关文章
|
4天前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
45 14
|
8天前
|
数据采集 Java Linux
面试大神教你:如何巧妙回答线程优先级这个经典考题?
大家好,我是小米。本文通过故事讲解Java面试中常见的线程优先级问题。小明和小华的故事帮助理解线程优先级:高优先级线程更可能被调度执行,但并非越高越好。实际开发需权衡业务需求,合理设置优先级。掌握线程优先级不仅能写出高效代码,还能在面试中脱颖而出。最后,小张因深入分析成功拿下Offer。希望这篇文章能助你在面试中游刃有余!
30 4
面试大神教你:如何巧妙回答线程优先级这个经典考题?
|
11天前
|
缓存 安全 Java
面试中的难题:线程异步执行后如何共享数据?
本文通过一个面试故事,详细讲解了Java中线程内部开启异步操作后如何安全地共享数据。介绍了异步操作的基本概念及常见实现方式(如CompletableFuture、ExecutorService),并重点探讨了volatile关键字、CountDownLatch和CompletableFuture等工具在线程间数据共享中的应用,帮助读者理解线程安全和内存可见性问题。通过这些方法,可以有效解决多线程环境下的数据共享挑战,提升编程效率和代码健壮性。
42 6
|
27天前
|
算法 安全 Java
Java线程调度揭秘:从算法到策略,让你面试稳赢!
在社招面试中,关于线程调度和同步的相关问题常常让人感到棘手。今天,我们将深入解析Java中的线程调度算法、调度策略,探讨线程调度器、时间分片的工作原理,并带你了解常见的线程同步方法。让我们一起破解这些面试难题,提升你的Java并发编程技能!
68 16
|
1月前
|
监控 Kubernetes Java
阿里面试:5000qps访问一个500ms的接口,如何设计线程池的核心线程数、最大线程数? 需要多少台机器?
本文由40岁老架构师尼恩撰写,针对一线互联网企业的高频面试题“如何确定系统的最佳线程数”进行系统化梳理。文章详细介绍了线程池设计的三个核心步骤:理论预估、压测验证和监控调整,并结合实际案例(5000qps、500ms响应时间、4核8G机器)给出具体参数设置建议。此外,还提供了《尼恩Java面试宝典PDF》等资源,帮助读者提升技术能力,顺利通过大厂面试。关注【技术自由圈】公众号,回复“领电子书”获取更多学习资料。
|
1月前
|
Java Linux 调度
硬核揭秘:线程与进程的底层原理,面试高分必备!
嘿,大家好!我是小米,29岁的技术爱好者。今天来聊聊线程和进程的区别。进程是操作系统中运行的程序实例,有独立内存空间;线程是进程内的最小执行单元,共享内存。创建进程开销大但更安全,线程轻量高效但易引发数据竞争。面试时可强调:进程是资源分配单位,线程是CPU调度单位。根据不同场景选择合适的并发模型,如高并发用线程池。希望这篇文章能帮你更好地理解并回答面试中的相关问题,祝你早日拿下心仪的offer!
39 6
|
1月前
|
安全 Java 程序员
面试直击:并发编程三要素+线程安全全攻略!
并发编程三要素为原子性、可见性和有序性,确保多线程操作的一致性和安全性。Java 中通过 `synchronized`、`Lock`、`volatile`、原子类和线程安全集合等机制保障线程安全。掌握这些概念和工具,能有效解决并发问题,编写高效稳定的多线程程序。
66 11
|
1月前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
2月前
|
缓存 安全 Java
【JavaEE】——单例模式引起的多线程安全问题:“饿汉/懒汉”模式,及解决思路和方法(面试高频)
单例模式下,“饿汉模式”,“懒汉模式”,单例模式下引起的线程安全问题,解锁思路和解决方法
|
2月前
|
Java 调度