Thread类的介绍

news2024/10/20 12:27:33

线程是操作系统中的概念,操作系统中的内核实现了线程这种机制,同时,操作系统也提供了一些关于线程的API让程序员来创建和使用线程。

在JAVA中,Thread类就可以被视为是对操作系统中提供一些关于线程的API的的进一步的封装。

多线程程序的特点

1.每一个线程都是一个执行流

2.CPU对线程的执行是并发执行,同时对线程的执行也是随机调度的。

1.创建线程

1. 通过Thread类创建线程

class  MyThread extends Thread{
    @Override
    public void run() {
        while(true){
            System.out.println("hello Thread");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

    }
}
public class Demo1 {
    public static void main(String[] args) throws InterruptedException {
        Thread t= new MyThread();
        t.start();
        while(true){
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }
}

我们通过创建一个Thread类的子类MyThread,重写Thread类中的run方法,并通过MyThread子类来创建一个线程的实例化对象,run方法是线程的入口函数。而在主函数中的start方法,表示创建一个线程,且一个线程只能start一次。 

当我们运行上面代码,发现打印的内容的顺序会变换,这是因为多线程的影响,上面代码中有两个线程,分别位t线程和主线程,由于线程在CPU上是并发执行的,且又因为CPU对线程又是随机调度的,这就导致我们两个线程的执行顺序会改变,自然而然打印的顺序也会改变。

2.通过Runnable接口来创建线程

class MyRunnable implements Runnable{
    @Override
    public void run() {
        while (true){
            System.out.println("hello t");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
public class Demo2 {
    public static void main(String[] args) {
        Runnable runnable=new MyRunnable();
        Thread t=new Thread(runnable);
        t.start();
    }
}

3. 通过匿名内部类来创建线程

匿名内部类的写法本质完成了3件事

1.创建了一个Thread的子类,不知到该子类的名字,所以为匿名。

2.{ }代码块里面可以编写子类的定义代码,需要哪些属性,需要重写哪些父类的方法等等。

3. 创建了匿名内部类的实例,并将该实例传给了t1.

3.1 Thread
    public static void main(String[] args) throws InterruptedException {
        Thread t1=new Thread(){
            @Override
            public void run() {
                while(true){
                    System.out.println("hello t");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        };
        t1.start();
        while (true){
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }
3.2 Runnable接口
public static void main(String[] args) throws InterruptedException {
        Runnable runnable=new Runnable() {
            @Override
            public void run() {
                while(true){
                    System.out.println("hello t");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        };
        Thread t=new Thread(runnable);
        t.start();
        while (true){
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }

4.通过Lambda表达式式创建线程-----  ()->{}

由于Java中,方法必须依附于类的体系中,由于Lambda表达式本质就是一个回调函数,所以,为了快速实现这个回调函数,在Java中,就创建了一个函数式接口----- ()->{ },{ }代码块可以写上回调函数所需要功能的代码。

public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(()->{
            while (true){
                System.out.println("hello t");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t.start();
        while (true){
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }

2.Thread类及常见方法

Thread类是JVM用来管理线程的一个类,也就是说,每个线程都有唯一的Thread类对象与之相关联。在Java中,每一个执行流(线程)都要用一个对象来表示,而Thread类对象就可以用来表示一个线程执行流,JVM会将这些Thread类对象组织起来,进行线程调度和线程管理。

2.1 Thread类常见的构造方法

方法

说明

Thread()创建线程对象
Thread(Runnable target)

使用Runnable对象创建线程对象

Thread(String name)创建线程对象,并给线程命名
Thread(Runnable target,String name)使用Runna对象创建线程对象,并命名
【了解】Thread(ThreadGroup group,Runnable target)

线程可以被用来分组管理,分好的组即为线程组

注释:关于线程的名字,我们可以通过Jconsole来观察。

2.2 Thread的几个常见属性

属性获取方法
IDgetID()
名称getName()
状态getState()
优先级getPriority()
是否为后台线程isDaemon()
是否存活

isAlive()

是否被中断isInterrupted()

1.ID是线程的唯一标识,不同线程的ID不会重复

2.名称是各种调试工具用到

3.状态表示线程当前所处的一个状态

4.优先级高的线程理论上更容易被CPU调度到

5.关于后台线程:线程分为后台线程(守护线程)和前台线程(用户线程)。我们需要记住一点:JVM会在一个线程中的所有的前台线程结束之后,JVM才会选择退出(程序终止执行)。也就是说,后台线程的存在不会阻止JVM的终止,相反,前台线程的存在会阻止JVM的终止,即使主线程已经结束。在Java中,线程在默认情况下是后台线程,例如垃圾回收机制就是一种后台线程。但是我们可以通过调用setDaemon(true)方法将一个线程设置为后台线程。

6.是否存活,可以简单理解为run方法是否运行结束。

2.3 如何启动一个线程----start()

一个线程对象被创建出来并不意味着线程就开始运行了。我们需要去调用start方法,接着线程中的run方法就会自动执行,只有run方法执行以后,一个线程才是真正运行起来。而调用了start方法,才是真正的在操作系统的层面建立了一个线程。

如下图:

2.4 中断一个线程

中断一个线程就是让该线程直接停止掉,不会再回复。中断一个线程就是让该线程的run方法(入口方法)尽快结束掉

目前常见终止线程的方式有以下两种方式:

1.通过共享的标记来沟通

2.调用interrupt()方法来通知

1.通过共享的标记来沟通

该方法是通过设计一个外部成员变量来终止一个线程。

如以下代码

public class Demo9 {
    public static boolean isFinished=false;//共享标记
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(()->{
            while (!isFinished){
                System.out.println("hello t");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t.start();
        Thread.sleep(3000);
        System.out.println("main尝试终止线程t");
        isFinished=true;
    }
}

我们通过isFinished 这个外部成员变量的值来通知操作系统是否终止t线程。

运行结果

注意事项:我们不能将该外部成员变量设置成一个局部变量 。否则会报出以下错误

原因解释:

这就涉及到变量捕获的问题 。简单来说,就是lambda表达式希望可以使用外面的变量,触发“变量捕获”的语法。由于lambda是一个回调函数,执行时机是很久以后,当操作系统真正创建出线程之后,才会执行。很有可能在创建线程的过程中,main线程就已经运行结束了,自然而然,身为局部变量的isFinished就会被销毁了。

为了解决这个问题,Java中的做法是,将被捕获的变量拷贝一份到lambda里面,外面变量的是否销毁,就不影响lambda里面的执行了。拷贝就以为着该变量就不适合被修改,尽管你在外面修改了这一变量的值,也不会影响拷贝到内部的值(本质上,被拷贝的变量和拷贝的变量是两个变量)

这种一边变,一边不变,可能给程序员带来更多的疑惑。所以在Java中就规定了,拷贝的变量就压根不允许被改变。所以后面修改isFinished的值时会报一个isFinished should be final or effectively final 的错,因为在Java中,被final修饰的变量是无法被修改的。

那为什么将isFinished设置为成员变量会没事呢?

因为当isFinished是一个成员变量时,此时触发的语法不在是“变量捕获”,而是切换成“内部类访问外部类的成员”的语法。

那是lambda表达式本质上是一个函数式接口,也相当于是一个内部类。isFinished本身就是一个外部类成员,内部类本来就能够访问外部类的成员。由于成员变量的生命周期是让GC(垃圾回收)来管理的,GC可以自动识别不在被引用的的对象,并将其占用的内存空间释放掉。所以在lambda里面就不必担心变量生命周期失效的问题,也就不必拷贝,也就不必有被final修饰的限制。

2.使用interrupt()方法来通知

Java的Thread类中提供了现成的方法直接进行判定线程方法是否被终止,不需要我们自己在创建了。

方法说明

public void interrupt()

中断对象关联的线程,如果线程正在阻塞,则以异常方式通知,否则设置中断标志位
public static boolean interrupted()判断当前线程的中断标志位是否设置,调用后清除标志位,不建议使用,静态方法位所有线程共有的
public boolean isInterrupted()判断对象关联的线程标志位是否设置,调用后不清除标志位

thread收到通知的方式有两种:

1.如果线程因为调用wait/join/sleep等方法而阻塞挂起,则以InterruedException异常的形式通知,清楚中断标志 ,当出现InterruedException的时候,要不要结束进程取决于catch中代码的写法可以忽略该异常,也可以跳出循环结束进程。

2.否则,只是内部的一个中断标志位被设置了,可以通过isInterrupted()方法判断中断标志是否被设置,不清除中断标志,这种方法通知收到的更及时,即使线程在sleep也可以马上收到。

public class Demo10 {
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(()->{
            while (!Thread.currentThread().isInterrupted()){
                System.out.println("hello t");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                      e.printStackTrace();//报异常,清楚中断标志
                    //break;//跳出循环,结束进程
                    //啥都不写,忽略异常,进程继续执行
                }
            }
        });
        t.start();
        Thread.sleep(3000);
        System.out.println("main尝试终止线程t");
        t.interrupt();
    }
}

注意事项:针对上述代码由于每次循环,线程大部分时间在处sleep状态,当主线程调用interrup()方法时,会大概率唤醒sleep方法, sleep方法就会报InterruedException异常。

正常来说,调用interrupt()方法就会将isInterrupted()方法内部的标志位改为true,但是上述代码,能够把sleep()方法唤醒,sleep方法在唤醒之后就会将isInterrupted()方法内部的标志位的值重新改为false。因此在进程不结束的情况下,如果继续执行到循环的条件判断,就会发现能够继续执行循环中的代码。

2.5 等待一个线程-----join()

有时我们需要等到一个线程完成它的所有操作后,才进行下一步操作。这时候就用到了join()方法。

方法说明
public void join()等待线程结束
public void join(long millis)等待线程结束,最多等待millis毫秒
public void jooin(long millis,int nanos)同理当精度更高
public class Demo11 {
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(()->{
            for(int i=0;i<3;i++){
                System.out.println("hello t");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t.start();
        System.out.println("main线程在等t线程结束");
        t.join();
        System.out.println("t线程结束,main线程结束");
    }
}

上面的代码中,主线程中线程对象t调用了join方法,这就意味着main线程必须要先等待t线程结束后,才会执行main线程里面的内容。 

运行代码

 

如果调用的是join(long millis)版本,则会表示main线程只会等待t线程millis秒,不管t线程在这段时间内是否终止,main线程不会在等待t线程,而是继续执行自己的任务。 

public class Demo11 {
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(()->{
            for(int i=0;i<3;i++){
                System.out.println("hello t");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t.start();
        System.out.println("main线程在等t2秒");
        t.join(2000);
        System.out.println("2秒内t线程没有结束,main线程不在等待,执行main线程");
    }
}
 

 2.6 获取当前线程引用

方法说明
public static ThreadcurrentThread()返回当前线程对象的引用

2.7 休眠当前线程-----sleep()

方法说明
public static void sleep(long millis)休眠当前线程millis毫秒
public static void sleep(long millis,int nanos)可以获得更高精度的睡眠

 关于sleep方法,有一点我们需要记得,因为线程调度是不可控的,所以,sleep方法只能保证实际休眠时间是大于等于参数设置的休眠时间的。

 这是因为调用sleep方法,相当于让当前线程,让出cpu的资源。后续休眠时间结束的时候,该线程需要操作系统内核时,操作系统会将该线程重新调度到cpu上,该线程才能继续执行。

换言之,sleep的时间到了,意味着该线程可以被调度,而不是立即执行,所以实际休眠时间会大于等于参数设置的休眠时间。

特殊用法:sleep(0)

sleep(0)意味着让当前线程立刻放弃CPU资源,等待操作系统重新调度。

 3.线程状态

从操作系统的角度来看,进程的有就绪和阻塞的两种状态。

Java线程也是对操作系统中的线程的重新封装。所以,针对线程的状态,Java中也重新进行了封装。

状态说明
NEWThread对象已经创建,但是start()方法没有被调用
TerminatedThread对象还在,但是内核中线程已经结束
Runnable就绪状态,线程正在CPU上执行或者线程可以随时去CPU上执行
TIME_WAITING线程阻塞,阻塞的时间有上限。一般是由于join(时间),sleep(时间)产生的阻塞。
WAITING死等,没事时间限制的等待。一般是由于join(),wait()产生的阻塞
BLOCKED由于锁竞争产生的阻塞

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

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

相关文章

基于SpringBoot+Vue+uniapp微信小程序的澡堂预订的微信小程序的详细设计和实现

项目运行截图 技术框架 后端采用SpringBoot框架 Spring Boot 是一个用于快速开发基于 Spring 框架的应用程序的开源框架。它采用约定大于配置的理念&#xff0c;提供了一套默认的配置&#xff0c;让开发者可以更专注于业务逻辑而不是配置文件。Spring Boot 通过自动化配置和约…

Goland 搭建Gin脚手架

一、使用编辑器goland 搭建gin 打开编辑器 新建项目后 点击 create 二、获得Gin框架的代码 命令行安装 go get -u github.com/gin-gonic/gin 如果安装不上&#xff0c;配置一下环境 下载完成 官网git上下载 这样就下载完成了。、 不过这种方法需要设置一下GOPATH 然后再执…

Electron-(三)网页报错处理与请求监听

在前端开发中&#xff0c;Electron 是一个强大的框架&#xff0c;它允许我们使用 Web 技术构建跨平台的桌面应用程序。在开发过程中&#xff0c;及时处理网页报错和监听请求是非常重要的环节。本文将详细介绍 Electron 中网页报错的日志记录、webContents 的监听事件以及如何监…

【uniapp】打包成H5并发布

目录 1、设置配置mainifest.sjon 1.1 页面标题 1.2 路由模式 1.3 运行的基础路径 2、打包 2.1 打包入口 2.2 打包成功 2.3 依据目录找到web目录 3、 将web目录整体拷贝出来 4、上传 4.1 登录uniapp官网注册免费空间 4.2 上传拷贝的目录 4.3 检查上传是否正确 5、…

【软件测试】JUnit

Junit 是一个用于 Java 编程语言的单元测试框架&#xff0c;Selenium是自动化测试框架&#xff0c;专门用于Web测试 本篇博客介绍 Junit5 文章目录 Junit 使用语法注解参数执行顺序断言测试套件 Junit 使用 本篇博客使用 Idea集成开发环境 首先&#xff0c;创建新项目&#…

【Python-AI篇】人工智能python基础-计算机组成原理

1. 计算机组成原理 2. python基础&#xff08;查漏补缺&#xff09; 2.1 字符串 2.1.1 字符串查找方法 find()&#xff1a; 检测某个字符串是否包含在这个字符串中&#xff0c;在的话返回下标&#xff0c;不在的话返回-1index()&#xff1a; 检测某个字符串是否包含在这个字…

git命令使用一览【自用】

git常见操作&#xff1a; git initgit remote add master【分支名字】 gitgits.xxxxx【仓库中获取的ssh链接或者http协议的链接】检查远程仓库是否链接成功。 git remote -v出现以下画面就可以git pull,git push了

cefsharp63.0.3(Chromium 63.0.3239.132)支持H264视频播放-PDF预览 老版本回顾系列体验

一、版本 版本:Cef 63/CefSharp63.0.3/Chromium63.0.3239.132/支持H264/支持PDF预览 支持PDF预览和H264推荐版本 63/79/84/88/100/111/125 <

Java EE规范

1、简介 Java EE的全称是Java Platform, Enterprise Edition。早期Java EE也被称为J2EE&#xff0c;即Java 2 Platform Enterprise Edition的缩写。从J2EE1.5以后&#xff0c;就改名成为Java EE。一般来说&#xff0c;企业级应用具备这些特征&#xff1a;1、数据量特别大&…

java 文件File类概述

前言 在Java中&#xff0c;File类是一个与文件和目录&#xff08;文件夹&#xff09;路径名相关的抽象表示形式。它是java.io包中的一个重要类&#xff0c;用于表示和操作文件系统中的文件和目录。 File类的基本概念 表示路径&#xff1a;File类既可以表示文件路径&#xff…

【mod分享】波斯王子遗忘之沙高清重置,纹理,字体,贴图全部重置,特效增强,支持光追

各位好&#xff0c;今天小编给大家带来一款新的高清重置MOD&#xff0c;本次高清重置的游戏叫《波斯王子&#xff1a;遗忘之沙》。 《波斯王子&#xff1a;遗忘之沙》是由育碧&#xff08;Ubisoft&#xff09;开发并发行的一款动作类游戏&#xff0c;于2010年5月18日发行。游戏…

Linux执行source /etc/profile命令报错:权限不够问(已解决)

1.问题 明明以root账号登录Linux系统&#xff0c;在终端执行命令source /etc/profile时 显示权限不够 如下图&#xff1a; 2.问题原因 可能在编辑 /etc/profile 这个文件时不小心把开头的 井号 ‘#’ 给删除了 如图&#xff1a; 这里一定要有# 3.解决办法 进入/etc/pro…

用你的手机/电脑运行文生图方案

随着ChatGPT和Stable Diffusion的发布&#xff0c;最近一两年&#xff0c;生成式AI已经火爆全球&#xff0c;已然成为移动互联网后一个重要的“风口”。就图片/视频生成领域来说&#xff0c;Stable Diffusion模型发挥着极其重要的作用。由于Stable Diffusion模型参数量是10亿参…

PHP爬虫:获取商品销量数据的利器

在电子商务的激烈竞争中&#xff0c;掌握商品销量数据是商家洞察市场动态、制定销售策略的关键。通过PHP爬虫技术&#xff0c;我们可以高效地获取这些数据&#xff0c;为商业决策提供支持。 PHP爬虫的优势 PHP作为一种流行的服务器端脚本语言&#xff0c;拥有跨平台运行、丰富…

【C++篇】类与对象的秘密(上)

目录 引言 一、类的定义 1.1类定义的基本格式 1.2 成员命名规范 1.3 class与struct的区别 1.4 访问限定符 1.5 类的作用域 二、实例化 2.1 类的实例化 2.2 对象的大小与内存对齐 三、this 指针 3.1 this指针的基本用法 3.2 为什么需要this指针&#xff1f; 3.3 t…

数据结构——链表,哈希表

文章目录 链表python实现双向链表复杂度分析 哈希表&#xff08;散列表&#xff09;python实现哈希表哈希表的应用 链表 python实现 class Node:def __init__(self, item):self.item itemself.next Nonedef head_create_linklist(li):head Node(li[0])for element in li[1…

SQL Server 2019数据库“正常,已自动关闭”

现象&#xff1a; SQL Server 2019中&#xff0c;某个数据库在SQL Server Management Studio&#xff08;SSMS&#xff09;中的状态显示为“正常&#xff0c;已自动关闭”。 解释&#xff1a; 如此显示&#xff0c;是由于该数据库的AUTO_ CLOSE选项被设为True。 在微软的官…

JavaSE——IO流1:FileOutputStream(字节输出流)、FileInputStream(字节输入流)

目录 一、IO流概述 二、IO流的分类 三、字节输出流与字节输入流 (一)字节输出流——FileOutputStream 1.FileOutputStream书写步骤 2.FileOutputStream书写细节 3.FileOutputStream写数据的3种方式 4.FileOutputStream的换行和续写 (二)字节输入流——FileInputStream…

如何给手机换ip地址

在当今数字化时代&#xff0c;IP地址作为设备在网络中的唯一标识&#xff0c;扮演着举足轻重的角色。然而&#xff0c;有时出于隐私保护、网络访问需求或其他特定原因&#xff0c;我们可能需要更改手机的IP地址。本文将详细介绍几种实用的方法&#xff0c;帮助您轻松实现手机IP…

若依框架中spring security的完整认证流程,及其如何使用自定义用户表进行登录认证,学会轻松实现二开,嘎嘎赚块乾

1&#xff09;熟悉之前的SysUser登录流程 过滤器链验证配置 这里security过滤器链增加了前置过滤器链jwtFilter 该过滤器为我们自定义的&#xff0c;每次请求都会经过jwt验证 ok我们按ctrl alt B跳转过去来看下 首先会获取登录用户LoginUser 内部通过header键&#xff0c;获…