当前位置: 首页 > news >正文

【Java面试】十六、并发篇:线程基础

文章目录

  • 1、进程和线程的区别
  • 2、并行和并发的区别
  • 3、创建线程的四种方式
    • 3.1 Runnable和Callable创建线程的区别
    • 3.2 线程的run和start
  • 4、线程的所有状态与生命周期
  • 5、新建T1、T2、T3,如何保证线程的执行顺序
  • 6、notify和notifyAll方法有什么区别
  • 7、wait方法和sleep方法有什么不同
  • 8、如何停止一个正在运行的线程
    • 8.1 使用退出标志
    • 8.2 调用stop方法
    • 8.3 调用interrupt方法

1、进程和线程的区别

在这里插入图片描述

  • 一个程序或者应用被运行,就是开启了一个进程(应用.exe背后的代码被从磁盘加载到内存)
  • 一个进程下,包含了多个线程,分别处理着不同的任务
  • 线程更轻量,上下文成本切换更低(上下文切换即从一个线程切换到另一个线程)

2、并行和并发的区别

  • 并行:多项工作一起执行,下班后,你收拾屋子,你对象做饭。与其相反的就是串行,即一步一步来,或者一件一件事情的做。(4核CPU同时执行4个线程)

在这里插入图片描述

  • 并发:多个工作换着处理,下班后,你屋子收拾一半后,去准备做饭,菜准备好后又回去接着收拾屋子(多个线程轮流使用一个或者多个CPU)

补充:操作系统中,有个任务调度器,将CPU时间片(比如15毫秒)分给不同的程序使用。CPU在线程间的切换速度很快,给人的感觉就是同时运行的。

在这里插入图片描述

并发是一个人同时吃三个馒头(一张嘴,第一个馒头吃一口,转身再吃一口第二个馒头),而并行是三个人同时吃三个馒头(三个人各吃各的,好比多核CPU各自执行指令)。

最后,并发在后端层面上,也指同一时刻,多个线程在访问同一个资源,多个线程对一个点,对应的例子: 春运抢票、电商秒杀

3、创建线程的四种方式

  • 继承Thread类

在这里插入图片描述

  • 实现Runnable接口

在这里插入图片描述

  • 实现Callable接口,Callable的泛型即重写的call方法的返回值类型

在这里插入图片描述

  • 线程池创建线程

在这里插入图片描述

3.1 Runnable和Callable创建线程的区别

  • Runnable 接口 run 方法没有返回值
  • Callable 接口 call 方法有返回值,是个泛型,和 Future、FutureTask配合可以用来获取异步执行的结果
  • Callable 接口的 call 方法允许抛出异常,而 Runnable 接口的run 方法的异常只能在内部try-catch消化,不能继续上抛

3.2 线程的run和start

  • start方法是用来启动线程的,通过该线程调用执行run方法,只能被调用一次
  • run方法里面封装了要被线程执行的代码,是一个普通方法,可以被调用多次
public class ThreadTest {public static void main(String[] args){Thread t1 = new Thread("t1") {@Overridepublic void run() {System.out.println("running...");}};t1.run();t1.run();t1.start();t1.start();}

调用两次run方法:
在这里插入图片描述
调用两次start方法:

在这里插入图片描述

4、线程的所有状态与生命周期

JDK源码中的线程状态枚举值:

在这里插入图片描述

状态流转:
  • NEW:新建状态,创建出一个Thread对象

  • Runnable:可执行状态,执行start方法,此时如果抢到了CPU时间片,那就去执行run方法,然后线程进入终止状态(死亡)

  • Blocked:没抢不到锁(lock、synchronized),进入阻塞状态,抢到锁后切换为Runnable状态

  • Waiting:调用了wait方法,进入等待状态,其他线程调用notify方法后被唤醒,进入Runnable状态
    在这里插入图片描述

  • Timed_Waiting:计时等待状态,线程调用了sleep方法,sleep时间过后进入Runnable状态

流转图:
在这里插入图片描述

5、新建T1、T2、T3,如何保证线程的执行顺序

thread.join();

以下,t2线程中执行t1.join,即等待t1线程执行完成后,再执行t2

在这里插入图片描述

6、notify和notifyAll方法有什么区别

notify是随机唤醒一个wait的线程,notifyAll是唤醒所有wait的线程。

如下,object对象上wait了两个线程t1、t2

在这里插入图片描述
notify时,只随机唤醒了一个object对象上wait的线程

在这里插入图片描述

7、wait方法和sleep方法有什么不同

共同点:

  • wait和sleep都会让当前线程暂时放弃CPU时间片的使用权,进入阻塞状态

不同点:

1)方法归属不同:

  • sleep是Thread类的静态方法
  • wait是Object类的静态方法,每个对象都有

2)醒来时机不同:

  • 执行sleep(long time) 和 wait(long time) 的线程,在相应毫秒过后会自己醒来进入就绪状态
  • wait(long time)wait() 都可以被 notify 唤醒,且wait() 如果不被唤醒就一直等待
  • sleep和wait都可以被打断唤醒

3)锁的特性不同:

  • 调用wait方法,必须先获取wait方法所在对象的锁,否则IllegalMonitorStateException,sleep则不用获取任何锁
  • wait对象执行后,会释放其占有的对象锁,其他线程可抢这个锁(放弃CPU时间片、放弃对象锁)
  • sleep如果是在synchronized代码块里,sleep并不会释放synchronized抢到的对象锁(放弃CPU时间片,如果有锁也不会放弃)

8、如何停止一个正在运行的线程

8.1 使用退出标志

加一个标记字段,改了标记之后,线程return,结束执行

class MyRun implements Runnable {boolean flag = false;public void run() {for (int i = 0; i < 100; i++) {if (! this.flag) {System.out.println(Thread.currentThread().getName() + "--->" + i);System.out.println(this.run);try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}} else {System.out.println("这是一些终止线程前要做的事");System.out.println("保存数据中..终止线程成功!");return;}}}
}
public class ThreadTest2 {public static void main(String[] args) {MyRun r = new MyRun();Thread t = new Thread(r);t.start();//sleep主线程三秒try{Thread.sleep(3000);}catch(InterruptedException e){e.printStackTrace();}//终止,改run属性为truer.run = true;}
}

在这里插入图片描述

8.2 调用stop方法

sleep方法已作废

MyThread myThread = new MyThread();
myThread.start();
//强制终止
myThread.stop();

8.3 调用interrupt方法

  • 打断正在阻塞的线程(sleep、wait、join),线程会抛出InterruptedException异常
public class MyInterrupt {public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{System.out.println("t1 正在运行...");try {Thread.sleep(5000);} catch (InterruptedException e) {e.printStackTrace();}}, "t1");t1.start();Thread.sleep(500);t1.interrupt();System.out.println(t1.isInterrupted());}
}

在这里插入图片描述

  • 打断正常线程,改变线程的中止标志位,是否退出由线程自己决定,思路上和方式一相似
public class MyInterrupt {public static void main(String[] args) throws InterruptedException {Thread t2 = new Thread(()->{while(true) {Thread current = Thread.currentThread();boolean interrupted = current.isInterrupted();if(interrupted) {System.out.println("打断状态:"+ interrupted);break;}}}, "t2");t2.start();Thread.sleep(500);t2.interrupt();}
}

如上,主线程中调用t2.interrupt(),t2线程的中止标志位变为true,但是否中止执行,看 t2 线程自己,上面t2线程自己判断标志位为true时,就break结束循环

在这里插入图片描述

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 企业微信hook接口协议,ipad协议http,取消扫码返回
  • EXCEL从图片链接获取图片
  • 分布式光纤测温DTS与红外热成像系统的主要区别是什么?
  • C++多线程并发
  • kettle_Hbase
  • 通过ssr-echarts,服务端生成echarts图
  • 渗透测试之内核安全系列课程:Rootkit技术初探(五)
  • 用函数指针求a和b中的大者
  • 中国蚁剑 安装教程 2024年5月
  • flutter文件分类模板之modules
  • NLP中的Tokenizer分词器的概念与实现
  • 一些关于科技的想法
  • 代码随想录训练营Day 56|力扣300.最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组
  • 概率论与数理统计,重要知识点——全部公式总结
  • Vue3_对接腾讯云COS_大文件分片上传和下载
  • ES6指北【2】—— 箭头函数
  • 时间复杂度分析经典问题——最大子序列和
  • Apache的基本使用
  • JavaScript HTML DOM
  • jquery ajax学习笔记
  • JS笔记四:作用域、变量(函数)提升
  • LeetCode刷题——29. Divide Two Integers(Part 1靠自己)
  • Promise面试题2实现异步串行执行
  • Tornado学习笔记(1)
  • vue:响应原理
  • 浮现式设计
  • 复杂数据处理
  • 关于字符编码你应该知道的事情
  • 理解 C# 泛型接口中的协变与逆变(抗变)
  • 每个JavaScript开发人员应阅读的书【1】 - JavaScript: The Good Parts
  • 使用Maven插件构建SpringBoot项目,生成Docker镜像push到DockerHub上
  • 使用权重正则化较少模型过拟合
  • 腾讯大梁:DevOps最后一棒,有效构建海量运营的持续反馈能力
  • 微信小程序填坑清单
  • 一个JAVA程序员成长之路分享
  • 一个普通的 5 年iOS开发者的自我总结,以及5年开发经历和感想!
  • ​LeetCode解法汇总2670. 找出不同元素数目差数组
  • ​二进制运算符:(与运算)、|(或运算)、~(取反运算)、^(异或运算)、位移运算符​
  • ​直流电和交流电有什么区别为什么这个时候又要变成直流电呢?交流转换到直流(整流器)直流变交流(逆变器)​
  • (22)C#传智:复习,多态虚方法抽象类接口,静态类,String与StringBuilder,集合泛型List与Dictionary,文件类,结构与类的区别
  • (3)选择元素——(17)练习(Exercises)
  • (55)MOS管专题--->(10)MOS管的封装
  • (C++17) std算法之执行策略 execution
  • (solr系列:一)使用tomcat部署solr服务
  • (TipsTricks)用客户端模板精简JavaScript代码
  • (超详细)语音信号处理之特征提取
  • (机器学习-深度学习快速入门)第一章第一节:Python环境和数据分析
  • (转)JAVA中的堆栈
  • (转载)hibernate缓存
  • .NET 4 并行(多核)“.NET研究”编程系列之二 从Task开始
  • .NET CORE 2.0发布后没有 VIEWS视图页面文件
  • .Net Framework 4.x 程序到底运行在哪个 CLR 版本之上
  • .NetCore部署微服务(二)
  • .stream().map与.stream().flatMap的使用
  • @AliasFor注解