线程介绍及其Java如何用Thread 类创建线程和操作线程方法

目录

  • 一、进程和线程
    • 1.1 进程
        • 特征
    • 2.2 线程
        • 特征
    • 2.3 区别
  • 二、利用Thread类创建线程
    • 2.1 通过创建Thread子类,重写run()方法
    • 2.2 通过实现Runnable接口,重写run()方法
    • 2.3. Callable接口 + FutureTask 创建线程
    • 2.3 三种方法区别
        • 1. 通过创建Thread子类,重写run()方法
        • 2. 通过实现Runnable接口,重写run()方法
        • 3. Callable接口 + FutureTask 创建线程
  • 三、关于线程常用操作方法
    • 3.1 操作线程名字方法
        • 代码演示
    • 3.2 获得当前线程
        • 代码演示
    • 3.3 线程优先级
        • 代码演示
        • 注意
    • 3.4 守护线程
        • 注意:
        • 代码演示
        • 结果:
    • 3.5 线程休眠
        • 代码演示
        • 结果:
    • 3.6 礼让线程
    • 3.7 插入(加入)线程
        • 代码演示
    • 3.8 停止/中断线程
  • 四、线程状态
  • 思维导图
    • 最后

一、进程和线程

进程和线程有什么关系?

1.1 进程

  • 进程是操作系统中的一种资源分配单位,它是操作系统中一个独立的执行单元。
  • 进程是电脑资源分配的最小单位。
  • 进程中至少有一个线程。
特征
  • 独立性:每个进程都是独立的执行单元,拥有自己的内存空间和系统资源。
  • 资源分配:操作系统将资源分配给进程,例如内存、CPU 时间等。
  • 执行:进程可以执行不同的任务,例如运行应用程序、提供服务等。

2.2 线程

  • 线程是进程中的一个执行单元。
  • 线程是资源调度的最小单位
特征
  • 轻量级:线程比进程更加轻量级,创建和销毁线程的开销远小于进程。
  • 共享资源:多个线程可以共享同一个进程的资源,如内存、文件等。
  • 高效:线程可以提高程序的响应速度和系统的吞吐量。

2.3 区别

  • 资源分配:进程拥有自己的内存空间和系统资源,而线程共享进程的资源。
  • 执行:进程可以执行不同的任务,而线程执行的是进程中的某个任务。
  • 开销:创建和销毁进程的开销远大于线程。

下面着重讲一下Java如何创建线程


二、利用Thread类创建线程

有三种方法创建

2.1 通过创建Thread子类,重写run()方法

  • 通过创建子类实现
  • 通过匿名内部类实现
    public static void main(String[] args) {for (int i = 0; i < 100; i++) {System.out.println("main线程---->"+ i);}//匿名内部类new Thread(){@Overridepublic void run() {for (int i = 100; i < 200; i++) {System.out.println("创建线程----》"+ i);}}}.start();// 创建子类对象实现new MyThread().start();}
}

2.2 通过实现Runnable接口,重写run()方法

  • 通过实现类实现
  • 通过匿名内部类实现
 public static void main(String[] args) {/** 1.创建Runnable接口实现类对象,并重写run方法* 2.创建Thread对象* 3.开启线程*///创建实现类对象MyRunnable myRunnable = new MyRunnable();//创建Thread对象Thread thread = new Thread(myRunnable);//开启线程thread.start();//使用匿名内部类实现Runnable runnable = new Runnable() {@Overridepublic void run() {System.out.println("匿名线程");}};//创建Thread对象Thread thread2 = new Thread(runnable);//开启线程thread.start();}

2.3. Callable接口 + FutureTask 创建线程

 public static void main(String[] args) throws ExecutionException, InterruptedException {/** 五个步骤* 1.创建Callable子实现类对象* 2.创建FutureTask对象* 3.创建Thread对象* 4.开启多线程* 5.获得返回值  注意:一定要先开启线程,才能获得返回值*/// 普通方法实现MyCallable myCallable = new MyCallable();// 创建Callable子实现类对象FutureTask<Integer> task = new FutureTask<>(myCallable);// 创建FutureTask对象,并将Callable子实现类作为参数传入Thread thread = new Thread(task); // 创建Thread对象,传入FutureTask对象参数thread.start(); // 开启多线程System.out.println(task.get()); // 获得返回值,并打印// 通过匿名内部类实现FutureTask<Integer> task1 = new FutureTask<>(new Callable<Integer>() { // 通过匿名内部类实现Callable接口@Overridepublic Integer call(){ // 重写Callable接口call() 方法return 22;}});new Thread(task1).start(); //创建Thread对象,并开启多线程System.out.println(task1.get());// 获得返回值,并打印
}

2.3 三种方法区别

1. 通过创建Thread子类,重写run()方法
简单,但是要继承,子类就不能在继承其他类了,使用后扩展性较低
2. 通过实现Runnable接口,重写run()方法
步骤较复杂,且重写的run()方法没有返回值,没有传入参数,功能较简单,但是相对于Thread扩展性较高
3. Callable接口 + FutureTask 创建线程
步骤相对复杂,但是线程具有返回值和参数,方法功能性较强

三、关于线程常用操作方法

3.1 操作线程名字方法

方法签名方法描述
void setName(String name)改变线程名称,使之与参数 name 相同
Thread(String name)分配新的 Thread 对象,名字与参数相同
String getName()获得线程名字,并返回
代码演示
// 获得当前线程对象Thread currentThread = Thread.currentThread();// 获得线程名字String name = currentThread.getName();System.out.println(name);// main// 创建新线程 三种方法// 创建Thread子类方法Thread newThread = new Thread(){@Overridepublic void run() {System.out.println("newThread");}};// 获得名字String name1 = newThread.getName();// 默认为Thread-0,那么下一个使用无参构造方法创建的对象名字为:Thread-1,以此类推...System.out.println(name1);//Thread-0// 设置名字newThread.setName("newThread");System.out.println(newThread.getName());//newThread

3.2 获得当前线程

方法签名方法描述
static Thread currentThread()返回对当前正在执行的线程对象的引用
代码演示
 		 // 获得当前线程对象Thread currentThread = Thread.currentThread();// 获得线程名字String name = currentThread.getName();System.out.println(name);// main

3.3 线程优先级

方法签名方法描述
void setPriority(int newPriority)更改线程的优先级
int getPriority()返回线程的优先级
代码演示
Thread t1 = new Thread("T1") {@Overridepublic void run() {for (int i = 1; i < 101; i++) {System.out.println(getName()+"执行-->" +i );}}};// 返回线程的优先级 int i = t1.getPriority();System.out.println(i);// 5 默认为5t1.setPriority(1);
注意
优先级只是相对来说的,优先级大的先执行的概率大,并不是优先级大就一定先执行

3.4 守护线程

方法签名方法描述
boolean setDaemon(boolean n)将该线程标记为守护线程或用户线程
注意:
  • 需要在开启线程前,调用方法
  • 设置成功后,该线程为守护线程,除外其他所以线程均为被守护线程
  • 被守护线程停止,守护线程也将停止
代码演示
 Runnable runnable = new Runnable() {@Overridepublic void run() {for (int i = 0; i < 100; i++) {System.out.println("小兵" + i + "在战斗");}}};//创建对象Thread thread = new Thread(runnable);//设置守护thread.setDaemon(true);//开启线程thread.start();Runnable runnable1= new Runnable() {@Overridepublic void run() {System.out.println("大王战死");}};//创建对象,并开启线程new Thread(runnable1).start();
结果:

在这里插入图片描述

3.5 线程休眠

方法签名方法描述
static void sleep(long millis)在指定的毫秒数内让当前正在执行的线程休眠(暂停执行)
代码演示
 Thread t2 = new Thread("行人") {@Overridepublic void run() {for (int i = 1; i < 1001; i++) {System.out.println("行人" + i + "执行...");}}};Thread t1 = new Thread("汽车") {@Overridepublic void run() {for (int i = 1; i < 1001; i++) {if (i == 100){try {/*** 让当前线程休眠* 让出系统资源,别的线程执行* 等当前线程休眠结束,继续执行*/Thread.sleep(1);} catch (InterruptedException e) {throw new RuntimeException(e);}}System.out.println("汽车" + i + "执行...");}}};
结果:

在这里插入图片描述

3.6 礼让线程

方法签名方法描述
static void yield()暂停当前正在执行的线程对象,并执行其他线程

3.7 插入(加入)线程

方法签名方法描述
void join()无限制时间加入,直到线程结束
void join(long millis)等待该线程终止的时间最长为 millis 毫秒
代码演示
        Thread t2 = new Thread("行人") {@Overridepublic void run() {for (int i = 1; i < 1001; i++) {System.out.println("行人" + i + "执行...");}}};Thread t1 = new Thread("汽车") {@Overridepublic void run() {for (int i = 1; i < 1001; i++) {if (i == 100){try {//t2.join();// 让线程2加入,直到线程2执行完毕,线程1才会继续t2.join(1);// 让线程2加入,但是线程2只运行1毫秒,1毫秒后,线程1 2继续争抢} catch (InterruptedException e) {throw new RuntimeException(e);}}System.out.println("汽车" + i + "执行...");}}};t1.start();t2.start();

3.8 停止/中断线程

方法签名方法描述
void stop()停止线程(该方法已过时)
void interrupt()中断线程

四、线程状态

  • 五种状态
    在这里插入图片描述

思维导图

  • 下面时该文章的思维导图,可以看一下

在这里插入图片描述


最后

如果感觉有收获的话,点个赞 👍🏻 吧。
❤️❤️❤️本人菜鸟修行期,如有错误,欢迎各位大佬评论批评指正!😄😄😄

💘💘💘如果觉得这篇文对你有帮助的话,也请给个点赞、收藏下吧,非常感谢!👍 👍 👍在这里插入图片描述

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

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

相关文章

SQL深度解析:掌握这些技巧,让你的数据库查询如虎添翼!

前言 随着大数据时代的来临&#xff0c;数据库的角色愈发重要。SQL作为使用最为广泛的数据查询语言&#xff0c;其深度解析与优化对于数据密集型应用来说至关重要。掌握高级SQL技巧不仅可以提升开发效率&#xff0c;还能显著提高数据查询的性能和灵活性。本文将探讨一些关键的S…

修改SubVI的LabVIEW默认搜索路径

在启动顶级VI后&#xff0c;LabVIEW可能会遇到找不到subVI的情况。这通常是由于subVI的路径发生了变化或没有被正确配置。 LabVIEW默认搜索路径 默认情况下&#xff0c;LabVIEW会按以下顺序搜索文件位置&#xff08;*表示LabVIEW将搜索子目录&#xff09;&#xff1a; <t…

如何从印刷体的图片中把手写体部分统统去掉?--免费途径

AI图像处理技术 我是从国外某个网站上找到在线AI免费credit的处理方式的。国内的基本没有全功能试用、或者即使收费也不好用。 国内的差距主要是&#xff1a;1、对图片分辨率和大小有更多限制&#xff0c;即使收费用户也是&#xff1b;2、需要安装app之类&#xff0c;然后连线…

MongoDB使用$addToSet向数组中添加元素

学习mongodb&#xff0c;体会mongodb的每一个使用细节&#xff0c;欢迎阅读威赞的文章。这是威赞发布的第66篇mongodb技术文章&#xff0c;欢迎浏览本专栏威赞发布的其他文章。如果您认为我的文章对您有帮助或者解决您的问题&#xff0c;欢迎在文章下面点个赞&#xff0c;或者关…

驱动开发(四):Linux内核中断

驱动开发系列文章&#xff1a; 驱动开发&#xff08;一&#xff09;&#xff1a;驱动代码的基本框架 驱动开发&#xff08;二&#xff09;&#xff1a;创建字符设备驱动 驱动开发&#xff08;三&#xff09;&#xff1a;内核层控制硬件层 驱动开发&#xff08;四&#xf…

java:一个springfox swagger2的简单例子

# 示例程序 【pom.xml】 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><version>2.3.12.RELEASE</version> </dependency> <dependency><groupId>…

视图-什么是(VIEW)?怎么创建(CREATE VIEW)?怎么删除(DROP)?怎么用(SELECT/INSERT/UPDATE/DELETE)?

一、引言 之前对数据库的操作都是针对基本关系表&#xff0c;操作都是在数据库的全局逻辑模式上进行的&#xff0c;而在实际的数据库系统中&#xff0c;可能用户只关心或只被允许使用数据库中的某些基本关系表或基本关系表中的某些属性列&#xff0c;这些数据构成了数据库的外…

vue(v-if,v-else-if-else-show)

基本应用 例子 <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTE-8"> <meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-w…

LabView_波形控件

波形图表 将一定数量的数据点存储在缓冲区&#xff0c;并通过这种方式存储并显示这些数据点。当缓冲区被填满后&#xff0c;波形图表将会用新的数据点覆盖缓冲区中存在时间最久的数据点。 当数据点可用时&#xff0c;波形图表将显示已有的数据点外加最新接收到的数据点 。 您可…

NVIDIA Triton系列02-功能与架构简介

NVIDIA Triton系列02-功能与架构简介 B站&#xff1a;肆十二-的个人空间-肆十二-个人主页-哔哩哔哩视频 (bilibili.com) 博客&#xff1a;肆十二-CSDN博客 问答&#xff1a;(10 封私信 / 72 条消息) 肆十二 - 知乎 (zhihu.com) 前面文章介绍微软 Teams 会议系统、微信软件与腾讯…

Thinkphp一文鸡富贵鸡玫瑰庄园富农场仿皮皮果理财农场源码

Thinkphp一文鸡富贵鸡玫瑰庄园富农场仿皮皮果理财农场源码&#xff0c;喜欢的朋友可以下载研究 一文鸡富贵鸡玫瑰庄园富农场仿皮皮果理财农场源码

Django初学者指南

文章目录 Django初学者指南1 Django简介1.1 Django的历史1.2 使用Django的知名网站1.4 Django的主要特点1.5 Django的工作原理 2 Django 使用2.1 Django 支持的 Python 版本2.2 Django 版本 3 Django 开发 Web 程序3.1 安装Django3.2 创建Django项目3.3 运行开发服务器3.4 创建…

数据结构02 队列及其应用【C++实现】

目录 队列及其特点 利用数组模拟队列的基本操作 创建队列 空队条件 元素入队 元素出队 模拟超市收银问题 队列操作 初始化 入队操作 出队操作 取出队首元素 STL模板中队列的基本使用 训练&#xff1a;约瑟夫问题 参考程序 队列及其特点 队列是一种特殊的线性表&am…

求导,积分

求导公式&#xff1a; 复合函数求导法则&#xff1a;两个函数导函数的乘积. 例如&#xff1a;f(x)2x1,f(x)2,g(x)x^24x4,g(x)2x4 那么复合函数&#xff1a; g(f(x))(2x1)^24(2x1)4 把&#xff08;2x1&#xff09;看做整体,则g2(2x1)4 然后再求&#xff08;2x1&#xff09;的导函…

Stable Diffusion vs DALL·E3

大模型技术论文不断&#xff0c;每个月总会新增上千篇。本专栏精选论文重点解读&#xff0c;主题还是围绕着行业实践和工程量产。若在某个环节出现卡点&#xff0c;可以回到大模型必备腔调或者LLM背后的基础模型新阅读。而最新科技&#xff08;Mamba,xLSTM,KAN&#xff09;则提…

基于System-Verilog的流水灯设计与仿真

文章目录 一、system Verilog1.语言基本介绍2.过程赋值和连续赋值 二、编写testbench仿真1.流水灯testbench2.2位全加器3.实验结果 一、system Verilog 1.语言基本介绍 像 Verilog 和 VHDL 之类的硬件描述语言 (HDL) 主要用于描述硬件行为&#xff0c;以便将其转换为由组合门…

海底管缆先敷后埋与边敷边埋有什么区别?

海缆铺设有两种方式&#xff1a;“边敷边埋”和“先敷后埋”。 “边敷边埋”冲埋式埋设犁施工法———通过埋设犁(水力开沟机)泵送高压水&#xff0c;在海底冲出一条沟槽的同时&#xff0c;将海缆平铺下去,然后利用在潮汐作用下海床面自行回填(必要时采取压盖保护施工)。主要施…

[linux]如何跟踪linux 内核运行的流程呢

前面已经可以把内核编译出来&#xff0c;但是作为技术狗想看到内核是怎么运行的怎么办&#xff1f; 内核很多代码都是C语言写的&#xff0c;那简单&#xff0c;添加2行代码&#xff1a; include/linux/printk.h 529和530原来的&#xff1a; #define pr_info(fmt, ...) \ …

vue input 限制输入,小数点后保留两位 以及 图片垂直居中显示 和 分享 git 小技巧

&#xff08;1&#xff09;input 限制输入&#xff0c;小数点后保留两位 <template><div><el-input v-model"number" input"checkNumber" blur"completeNumber" placeholder"请输入"></el-input></div>…

文章解读与仿真程序复现思路——电工技术学报EI\CSCD\北大核心《计及台风时空特性和灵活性资源协同优化的配电网弹性提升策略》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…