【JavaEE初阶系列】——多线程 之 创建线程

目录

🎈认识Thread类

🎈Sleep

🎈创建线程

🚩继承Thread,重写run方法

🚩实现Runnable接口,重写run方法

🚩使用匿名内部类创建 Thread 子类对象 

🚩使用匿名内部类,实现Runnable接口

🚩Lambda表达式 


我们之前接触到的程序都是单线程的,不是多线程的,就是程序运行的时候,进行的是多个方向(线程)打印。其实就是单线程只进行一个任务,多线程是同时进行俩个任务。

感受多线程程序和普通程序的区别
  • 每个线程都是一个独立的执行流
  • 多个线程之间是 "并发" 执行的.

🎈认识Thread类

我们首先来认识认识这个类,这个类是java.lang包下的,所以我们就可以直接使用进行。

class Mythread extends Thread{@Overridepublic void run() {//这个方法就是线程的入口System.out.println("创建一个类并且继承Thread类");}
}

我们创建一个类继承父类Thread线程,重写run方法。这个方法就是线程的入口。

Thread里面的成员有俩个 ,一个是run(),一个是start()方法。俩者的区别是:

  • start()则是真正调用了系统API,在系统中创建出线程,让线程再调用run.(兵分俩路,并发执行)

 我们把上述的代码弄成死循环,然后我们可以看到,俩个while循环再“同时执行”,看到的结果是俩边的日志都在交替打印。 

这就是并发编程的效果,充分的使用了多核cpu资源。


  • run()  只是描述了进程的入口(进程要做什么任务)

如果实行run()方法的话,那么此时的代码不会创建新的线程,只有一个主线程,这个主线程里面只能依次循环,执行完一个循环再执行另一个。(因为上面start调用了系统的api,让系统再调用run()方法,所以我们就实现了后面的代码)但是这个run()不能创建新的线程,只能去创建一个主线程,再自己创建的类里面的run()方法。

因为这一题我都是死循环,所以不管只能执行上面调用run()方法的线程,而下面的线程是不能执行的。main线程其实也是一个线程,但是一个java中至少会有一个main线程。


🎈Sleep

因为用while(true)循环之后,我们发现执行的太快了,我们需要他慢点执行,让我们更加清楚的看到执行的情况。Thread里面重写了sleep方法。是静态方法,是依赖于类的,而不是依赖于对象的

这样的设计也合乎情理,我们是依赖于Thread类进行实现睡眠状态。


我们看到每秒打印的内容是不一样的。操作系统,对于多个线程的调度顺序,是不确定的,随机的,取决于操作系统对于线程调度的模块(调度器),不过关于谁先打印,大概率是main方法下的,因为调用start之后,新的线程被创建也是有一定的开销的,创建过程中,主线程就执行了自己main方法下的线程。(当然无法排除极端情况,主线程正好卡了下,使新线程的日志先打印....)


🎈创建线程

🚩继承Thread,重写run方法

   通过自定义一个类(这里起名为:MyThread),继承Thread类,重写run方法,最后在main方法中new出MyThread实例,调用这个实例的继承的Thread类的start方法创建一个线程。

  • 1.创建出MyThread实例,并不代表在系统真的创建一个线程,只有调用start方法时,才创建出一个新的线程,新线程会执行run里的逻辑,直到run里逻辑执行完,线程就结束了;
  • 2.运行一次Java程序就启动了一个进程,一个进程里至少会有一个线程,这里JVM默认创建的线程就是main线程(主线程),main主线程和MyThread创建出来的新线程是“并发执行”的关系(并发+并行),也可以理解为同时执行,各执行各的;
  • 3.直接调用run并没有创建线程,只是在原来的线程中执行代码;
class MyThread extends Thread{@Overridepublic void run() {while (true){System.out.println("Thread begin");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}}
}public static void main(String[] args) throws InterruptedException {//继承Thread 重写run方法Thread thread=new MyThread();thread.start();while(true){System.out.println("main begin");Thread.sleep(1000);}}

🚩实现Runnable接口,重写run方法

通过自定义一个类(这里起名为:MyRunnable)实现Runnable接口,重写run方法,最后在main方法new出MyRunnable实例和Thread实例,最后通过start方法创建并启动线程。

这里相当于把线程要干的活和线程本身分离开了,使用MyRunnable这个自定义的类来表示“线程要完成的任务”,这样做的目的就是为了“解耦合”,假设未来有新的任务需要线程去执行,那么通过这种方式,代码改动就比较小。

class MyRunable implements Runnable{//实现Runnable接口 重写run方法@Overridepublic void run() {while (true){System.out.println("Runnable begin");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}}
}public static void main2(String[] args) throws InterruptedException {//实现Runnable接口 重写run方法Runnable runnable=new MyRunable();Thread thread=new Thread(runnable);thread.start();while (true){System.out.println("main begin");Thread.sleep(1000);}}

🚩使用匿名内部类创建 Thread 子类对象 

 直接创建Thread子类,同时实例化出一个对象,重写run方法,最后通过start方法创建并启动线程.

 Thread thread = new Thread() {@Overridepublic void run() {System.out.println("使用匿名内部类创建 Thread 子类对象");}};thread.start();}

🚩使用匿名内部类,实现Runnable接口

通过使用使用匿名内部类,实现Runnable接口作为Thread构造方法的参数,最后通过start创建并启动线程;

  public static void main4(String[] args) {Thread thread=new Thread(new Runnable() {@Overridepublic void run() {System.out.println("使用匿名内部类,实例Runnable接口作为构造参数;");}});}

🚩Lambda表达式 (推荐方法)

 lambda本质上就是一个“匿名函数”,()表示函数的形参,{}表示函数体,->特殊语法,表示它是lambda表达式(->是从C++那里抄来的)。

 public static void main3(String[] args) throws InterruptedException {//使用Lambda 创建Thread子类对象Thread thread=new Thread(()->{while ((true)){System.out.println("run");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}});thread.start();while (true){System.out.println("进行");Thread.sleep(1000);}}

坚持下去。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/748689.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

非空约束

oracle从入门到总裁:​​​​​​https://blog.csdn.net/weixin_67859959/article/details/135209645 非空约束 所谓的非空约束,指的是表中的某一个字段的内容不允许为空。如果要使用非空约束,只需要在每个列的后面利用“NOT NULL”声明即可 -- 删除数…

【Preprocessing数据预处理】之Scaler

在机器学习中,特征缩放是训练模型前数据预处理阶段的一个关键步骤。不同的缩放器被用来规范化或标准化特征。这里简要概述了您提到的几种缩放器: StandardScaler StandardScaler 通过去除均值并缩放至单位方差来标准化特征。这种缩放器假设特征分布是正…

PFA烧杯透明聚四氟乙烯刻度量杯

PFA烧杯,刻度清晰,耐酸碱,和有机溶剂。

腾讯春招后端一面(八股篇)

前言 前几天在网上发了腾讯面试官问的一些问题,好多小伙伴关注,今天对这些问题写个具体答案,博主好久没看八股了,正好复习一下。 面试手撕了三道算法,这部分之后更,喜欢的小伙伴可以留意一下我的账号。 1…

ElementUI Message 消息提示,多个显示被覆盖的问题

现象截图&#xff1a; 代码&#xff1a;主要是在this.$message 方法外层加上 setTimeout 方法 <script> export default {name: "HelloWorld",props: {msg: String,},methods: {showMessage() {for (let i 0; i < 10; i) {setTimeout(() > {this.$mess…

《荒野大镖客》等优秀的国产游戏能成为国产3a的标杆吗

中国或许不需要3A&#xff0c;但对于一些玩家来说&#xff0c;国产3A更多的是一个梦想&#xff0c;就像动画爱好者期待的优秀国产2D动画一样。 提问者所说的“玩家众多”&#xff0c;其实非核心玩家占比很高。 其中有一些是《王者荣耀》、《和平精英》等轻手游玩家或者国内二次…

yolov8 分割 模型 网络 模块图

下图是使用yolov8n-seg-p6.yaml imgsz1472 类别数2的情况下训练得到的静态导出的onnx文件使用netron工具可视化的结果 简单标注了yolov8n-seg-p6.yaml配置文件中各层和netron工具可视化的结果的对应关系

图解缓存淘汰算法 LRU、LFU | 最近最少使用、最不经常使用算法 | go语言实现

写在前面 无论是什么系统&#xff0c;在研发的过程中不可避免的会使用到缓存&#xff0c;而缓存一般来说我们不会永久存储&#xff0c;但是缓存的内容是有限的&#xff0c;那么我们如何在有限的内存空间中&#xff0c;尽可能的保留有效的缓存信息呢&#xff1f; 那么我们就可以…

前端基础——HTML傻瓜式入门(2)

该文章Github地址&#xff1a;https://github.com/AntonyCheng/html-notes 在此介绍一下作者开源的SpringBoot项目初始化模板&#xff08;Github仓库地址&#xff1a;https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址&#xff1a;https://blog.c…

C/C++程序设计实验报告3 | 数组实验

本文整理自博主本科大一《C/C程序设计》专业课的课内实验报告&#xff0c;适合C语言初学者们学习、练习。 编译器&#xff1a;gcc 10.3.0 ---- 注&#xff1a; 1.虽然课程名为C程序设计&#xff0c;但实际上当时校内该课的内容大部分其实都是C语言&#xff0c;C的元素最多可能只…

stm32学习——串口通信中的奇偶校验位

常用的校验算法有奇偶校验、校验和、CRC&#xff0c;还有LRC、BCC等不常用的校验算法。 以串口通讯中的奇校验为例&#xff0c;如果数据中1的个数为奇数&#xff0c;则奇校验位0&#xff0c;否则为1。 例如原始数据为&#xff1a;0001 0011&#xff0c;数据中1的个数&#xf…

HarmonyOS NEXT星河版——还是Android上套个壳吗?

这真的是我2024年听过最搞笑的话,就在前几天&#xff0c;居然还有人说鸿蒙OS就是安卓套个壳&#xff0c;简直无语&#xff01; 你敢相信&#xff1f;就在前几天&#xff0c;我还听到有人说&#xff1a;鸿蒙os就是安卓上套一个壳。唉&#xff0c;我真是无语了。 哎&#xff0c…

如何在Windows11上通过PHPStudy小皮面板快速大家MySQL环境

首先&#xff0c;下载小皮面板&#xff1a;https://www.xp.cn/ 点Windows版本&#xff1a; 开始下载&#xff1a; 或者直接从百度网盘下载&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1gcaiK54yW7DcrYld22V06A 提取码&#xff1a;4oj8 –来自百度网盘超级会员V9…

【力扣】141. 环形链表

题目描述 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置&a…

Docker配置Nginx、tomcat、elasticsearch

配置nginx 需要先pull下来 #启动nginx -d 表示后台运行 -p 表示暴露端口&#xff0c;将80暴露为3344 [rootiZf8zhsqf64x47n1tpdy6oZ home]# docker run -d -p:3344:80 nginx 5dd62cea7681975d37d1a9867bc9776de0206519f624b461346ac83025656642 [rootiZf8zhsqf64x47n1tpdy6oZ…

Spark-Transformation以及Action开发实战

文章目录 创建RDDTransformation以及ActionTransformation开发Action开发RDD持久化共享变量创建RDD RDD是Spark的编程核心,在进行Spark编程是,首要任务就是创建一个初始的RDDSpark提供三种创建RDD方式:集合、本地文件、HDFS文件 集合:主要用于本地测试,在实际部署到集群运…

51-31 VastGaussian,3D高斯大型场景重建

2024 年 2 月&#xff0c;清华大学、华为和中科院联合发布的 VastGaussian 模型&#xff0c;实现了基于 3D Gaussian Splatting 进行大型场景高保真重建和实时渲染。 Abstract 现有基于NeRF大型场景重建方法&#xff0c;往往在视觉质量和渲染速度方面存在局限性。虽然最近 3D…

C++第五弹---类与对象(二)

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】【C详解】 类与对象 1、类对象模型 1.1、如何计算类对象的大小 1.2、类对象的存储方式猜测 1.3、结构体内存对齐规则 2、this指针 2.1、this指针的引出 2.2…

Cesium 获取 3dtileset的包围盒各顶点坐标

Cesium 获取 3dtileset的包围盒各顶点坐标 /*** 获取 3dtileset的包围盒各顶点坐标, z 方向取高度最低的位置* param {*} tileset* param {*} options* returns* ref https://blog.csdn.net/STANDBYF/article/details/135012273* ref https://community.cesium.com/t/accurate-…

双指针算法_移动零_

题目&#xff1a; 给定一个数组 num &#xff0c;编写一个函数将数组内部的数字0都移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序&#xff01; 同时不能通过复制数组&#xff0c;开辟新的数组空间的情况下原地对数组进行操作 示例&#xff1a; 本题的原理&#x…