【Java核心知识】线程基础知识

news2025/1/21 9:20:15

文章目录

  • 线程
    • 线程与进程的区别
    • 创建线程的方法
      • 方法一:继承Thread类
      • 方法二:实现Runnable接口
      • 方法三:使用Callable和FutureTask创建带返回值的线程
      • 方法四:通过线程池创建线程
    • 线程的基本操作
      • 线程的状态
      • 守护线程

线程

线程与进程的区别

一个进程由程序段数据段进程控制块三部分组成。程序段也称为代码段,是进程的程序指令;数据段是进程的操作数据在内存中的位置;进程控制端(PCB)包含进程的描述信息(如进程ID和进程名称,进程状态。进程优先级)、控制信息(如程序起始地址)、资源信息(内存信息,设备信息,文件句柄)、进程上下文(CPU寄存器的值、程序计数器PC的值),是进程存在的唯一标识

各个进程之间不共享内存,但是线程之间会共享进程的方法区内存空间、堆内存、系统资源。

进程是操作系统分配资源的最小单位,线程CPU调度的最小单位。

每一个线程都有自己独立的栈内存JDK1.8默认是1MB栈内存和堆内存是不一样的,栈内存不会被GC回收。栈内存分配的基本单位是栈帧,线程每进入一个方法,就会分配一个栈帧,栈帧保存着方法中的局部变量、方法的返回值以及方法的其他信息。当方法退出后,栈内存就会弹出该栈帧,进而释放内存空间。

创建线程的方法

方法一:继承Thread类

  • 继承Thread类,创建一个新的线程类,同时重写run方法。

  • 实例化该类,调用start方法启动线程

public class MyThread extends Thread{
    //线程的编号
    static int threadNo = 1;

    public MyThread() {
        super("DemoThread-" + threadNo++);
    }

    public void run() {
        for (int i = 1; i < 2; i++) {
            System.out.println(getName() + "轮次" + i);
        }
        System.out.println(getName() + "运行结束");
    }
}
@Test
public void testCreateThread() throws InterruptedException {
    Thread thread = null;
    //方法一:使用Thread子类创建和启动线程
    for (int i = 0; i < 2; i++) {
        thread = new MyThread();
        thread.start();
    }
    Thread.sleep(3000);
    System.out.println(Thread.currentThread().getName() + " 运行结束.");
}

方法二:实现Runnable接口

  • 新建类实现Runnable接口
  • 将实现Runnable接口的类传入Thread类中,作为Thread类的目标方法
public class Thread2 implements Runnable{
    @Override
    public void run() {
        System.out.println("我是通过实现Runnable接口创建的线程");
    }
}
@Test
public void testThread2() throws InterruptedException {
    Thread thread = new Thread(new Thread2(), "线程2");
    thread.start();
    Thread.sleep(1000);
}

或者可以直接传入Runnable接口匿名实现类,还可以使用lambda表达式

@Test
public void testThread2() throws InterruptedException {
    // Thread thread = new Thread(new Thread2(), "线程2");
    // 使用lambda表达式
    // Thread thread = new Thread(() -> System.out.println("我是匿名实现类"));
    Thread thread = new Thread(new Runnable() {
        @Override
        public void run() {
            System.out.println("我是匿名实现类");
        }
    });
    thread.start();
    Thread.sleep(1000);
}

如果使用第一种方法创建线程,创建线程时需要new Thread,如果采用继承方法,那么每个线程的数量都是各自享有的,不能共同合作;如果采用实现Runnable接口,创建线程时,可以多个线程共享一个Runnable接口实例,从而多个线程共享数据,相互合作。

实现Runnable接口的方法本质上是Thread的run方法源码会判断自己的target域是否为null,如果不为空,则执行target.run()方法,而target域就是我们传入的Runnable接口实例

方法三:使用Callable和FutureTask创建带返回值的线程

前面实现Runnable接口不能有返回值,如果新的线程需要有返回值,需要实现Callable接口。但是Callable接口跟Runnable接口没什么关系,所以不能作为new Thread(Runnable run)的传入参数。所以为了桥接Callable接口和Runnable接口,定义了一个新的接口RunnableFuture接口,该接口继承了Runnable接口使其可以作为Thread类的传入参数)和Future<>接口(1、取消异步任务,2、判断任务是否执行完毕,3、返回异步任务执行结果)。

FutureTask类实现了RunnableFuture接口,可以作为Thread的传入参数,同时他又拥有一个callable实例,用来执行任务。FutureTask内部还有另一个非常重要的Object 类型的成员——outcome实例属性,用来保存返回结果。

所以整个执行流程为:

  • 首先为了声明一个异步任务,我们需要实现Callable接口的call方法,这是第一步,声明了任务。

  • 可是如何新建线程执行该任务呢?Callable接口实现类并不能作为Thread的target;于是我们把实现Callable接口的任务类传递给了FutureTask类,FutureTask实现了RunnableFuture接口,该接口继承了Runnable和Future接口,可以作为Thread类的target。

从直觉上讲,RunnableCallable接口的地位一致,Future接口是为了管理Callable接口返回值锁增加的功能增强接口,它们两个总是成对出现。关系如下:

在这里插入图片描述

@Test
public void testThread3() throws ExecutionException, InterruptedException {
    // 将Callable接口实例传入FutureTask类
    FutureTask<Long> futureTask = new FutureTask<>(() -> {
        long startTime = System.currentTimeMillis();
        System.out.println(Thread.currentThread().getName() + " 线程运行开始.");
        Thread.sleep(1000);
        for (int i = 0; i < 200; i++) {
            int j = i * 10000;
        }
        long used = System.currentTimeMillis() - startTime;
        return used;
    });
    Thread thread = new Thread(futureTask);
    thread.start();
    // 等子线程执行完毕后返回
    Long res = futureTask.get();
    System.out.println(res);
}

如果子线程没执行完毕,主线程就调用get()函数获取执行结果,那么主线程会阻塞直到子线程执行完毕

方法四:通过线程池创建线程

可以将一个Runnable接口匿名类或者一个Callable接口匿名类提交给线程池对象来创建线程。

线程池对象由线程池工厂创建。

  • execute方法不带返回值
  • submit方法带返回值:可以看到submit()方法只需要传入Callable接口,不需要创建FutureTask对象,这是因为submit方法内部会创建FutureTask对象。
@Test
public void testExecutor() throws ExecutionException, InterruptedException {
    ExecutorService executorService = Executors.newCachedThreadPool();
    // 通过execute方法向线程池提交一个runnable接口匿名类,不带返回值
    executorService.execute(() -> {
        System.out.println("我是Runnable接口线程");
    });
    Future<Integer> future = executorService.submit(() -> {
        System.out.println("我是Callable或者Runnable接口线程");
        return 2;
    });
    // 获取返回结果
    System.out.println(future.get());
}

线程的基本操作

  • Thread.sleep():使线程睡眠,从运行态转为阻塞态
  • Thread.interrupt():将线程的interrupt标志设为true,可以通过isInterrupt()方法判断是否中断。interrupt()方法只是会改变线程中断状态,不会真正中断一个线程,仍需要用户调用isInterrupted()方法进行监控后处理。
  • Thread.join():join()方法是实例方法,需要使用被合并线程的句柄(或者指针、变量)去调用, 主线程必须等待被合并线程执行完毕后才能继续执行
  • Thread.yield():让线程由运行态转为就绪状态,并不会阻塞该线程

线程的状态

  • 新建:新建状态
  • 就绪:等待线程调度
  • 运行:运行中
  • 阻塞:不满足某种条件,被挂起
  • 结束:线程结束

守护线程

  • 守护线程必须在调用start方法前设置
  • 当main函数退出后,守护线程也会退出。如果JVM进程终止,守护线程也会被强制关闭
  • 守护线程创建的线程也是守护线程

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

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

相关文章

基于Java的学生宿舍管理系统,springboot+jsp,MySQL数据库,前后端分离,学生+宿舍管理员+系统管理员三种身份,完美运行,有一万字论文。

基于Java的学生宿舍管理系统&#xff0c;springbootjsp&#xff0c;MySQL数据库&#xff0c;前后端分离&#xff0c;学生宿舍管理员系统管理员三种身份&#xff0c;完美运行&#xff0c;有一万字论文。 学生&#xff1a;个人信息管理、保修、假期&#xff08;入校、离校&#x…

【网络】路由配置实践

网络实践-路由篇 本文使用vmware虚拟机进行路由表配置实践&#xff0c;通过配置路由表连接两个不同的网络 网络规划&#xff1a; 准备三台centos7虚拟机&#xff0c;其中一台作为路由设备router&#xff0c;其余两台分别代表处在不通网络中的pc1和pc2。因此router虚拟机需要分…

IntelliJ IDEA的远程开发(Remote Development)

DEA的远程开发功能&#xff0c;可以将本地的编译、构建、调试、运行等工作都放在远程服务器上执行&#xff0c;而本地仅运行客户端软件进行常规的开发操作即可&#xff0c;官方给出的逻辑图如下&#xff0c;可见通过本地的IDE和服务器上的IDE backend将本地电脑和服务器打通&am…

iOS设备管理软件iMazing2024

文件系统是一种用于在存储设备上组织文件和文件夹的系统。所有的存储设备&#xff08;如计算机、USB存储器、手机等&#xff09;都有着自己的文件系统。 iMazing文件系统功能将iOS设备中封闭而复杂的文件管理系统整合在一起&#xff0c;让您管理文件像在电脑中使用Windows文件…

肖sir __linux__面试题和考核05

面试题 1、查看linux中第11行到第20行的数据&#xff08;比如文档a 有30行&#xff09; 方法1&#xff1a;tail -n 11 mm |head -n10 n 表示从第10行开始&#xff0c;取前10行 方法2&#xff1a;head -n -10 mm| tail -n 10 表示从末尾第10行开始&#xff0c;最后10行 方法3&am…

ThreadLocal原理剖析

一.ThreadLocal的含义 ThreadLocal也就是线程本地变量&#xff0c;创建了一个ThreadLocal变量&#xff0c;访问这个变量的每个线程都会有这个变量的一个本地拷贝&#xff0c;多个线程操作这个变量的时候&#xff0c;实际上是操作自己本地内存里面的变量&#xff0c;从而起到线…

基于单片机的简易智能电动车设计

一、项目介绍 智能交通工具在现代社会中起着越来越重要的作用&#xff0c;电动车作为一种环保、便捷的交通工具&#xff0c;受到了广泛的关注和应用。本设计基于单片机技术&#xff0c;设计一款简易智能电动车&#xff0c;实现基本的控制和功能&#xff0c;并提供良好的用户体…

kubectl入门

一.kubectl的三种资源管理方式&#xff1a; 二. kubectl资源介绍&#xff1a; 1.namespace&#xff1a;实现多套环境的资源隔离或者多租户的资源隔离。k8s中的pod默认可以相互访问&#xff0c;如果不想让两个pod之间相互访问&#xff0c;就将其划分到不同ns下。 2.pod&#x…

搭建Python开发环境

开发环境 Python是一种跨平台 的编程语言&#xff0c;这意味着它能够在所有的主流操作系统中运行&#xff0c;开发者可以Windows、Linux、Mac中开发和学习Python&#xff0c;甚至在Android手机中也可以运行Python代码。 在所有安装了Pythonr现代计算机上&#xff0c;都能够运…

华为OD:敏感字段加密

题目描述&#xff1a; 给定一个由多个命令字组成的命令字符串&#xff1a; 1、字符串长度小于等于127字节,只包含大小写字母,数字,下划线和偶数个双引号&#xff1b; 2、命令字之间以一个或多个下划线_进行分割&#xff1b; 3、可以通过两个双引号”"来标识包含下划线…

自定义创建项目

基于VueCli自定义创建项目 1.Eslint代码规范 代码规范:一套写代码的约定规则。 比如 赋值符号的左右是否需要空格 一句话结束是否要加&#xff1b; 正规的团队 需要统一的编码风格 https://standardjs.com/rules-zhcn.html 规则查找 https://zh-hans.eslint.org/docs/late…

CESM2代码下载

这半年忙着毕业写论文&#xff0c;好久好久好久不更新了∠( ω)&#xff0f; &#xff0c;今天准备开个新坑 ๑乛◡乛๑&#xff0c;学习一下CESM&#xff08;Community Earth System Model&#xff09;&#xff0c;它是一个完全耦合的全球气候模型&#xff0c;可用于地球过去、…

千纸鹤APP云验证系统源码 APK注入引流弹窗

千纸鹤APP云验证系统是一款全面的验证系统&#xff0c;包括网络验证、APK注入、注册机、引流弹窗、更新弹窗等功能。该系统提供完整的源代码&#xff0c;方便开发者二次开发和定制化需求。 可以对用户进行多种验证&#xff0c;包括账号密码验证、短信验证码验证等。该系统还提供…

jmeter 常数吞吐量定时器

模拟固定吞吐量的定时器。它可以控制测试计划中各个请求之间的时间间隔&#xff0c;以达到预期的吞吐量。 参数包括&#xff1a; Target Throughput&#xff1a;目标吞吐量&#xff08;每分钟请求数&#xff09;Calculate Throughput based on&#xff1a;吞吐量计算基准&…

机器人中的数值优化(七)——修正阻尼牛顿法

本系列文章主要是我在学习《数值优化》过程中的一些笔记和相关思考&#xff0c;主要的学习资料是深蓝学院的课程《机器人中的数值优化》和高立编著的《数值最优化方法》等&#xff0c;本系列文章篇数较多&#xff0c;不定期更新&#xff0c;上半部分介绍无约束优化&#xff0c;…

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书南京师范大学图书馆

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书南京师范大学图书馆

程序员工作技巧

提高工作技巧的秘方 案例 让我们猜测一下 绵羊想表达的什么&#xff1f; 并不够准确 崩了&#xff0c;不能用了整个业务瘫痪了研发没有责任感 语义规则/模棱俩可 相对语言 量化数据表达&#xff1a;疼苦指数&#xff0c;拥堵指数&#xff0c;准确。 尽量减少标签化评价 标签…

iPhone 隔空投送使用指南:详细教程

本文介绍了如何在iPhone上使用隔空投送,包括如何在iOS 11到iOS 14的iPhone上启用它、发送文件以及接受或拒绝AirDrop发送给你的文件。对于iOS 7以上的旧款iPhone,提供了另一种方法。 如何打开隔空投送 你可以通过以下两种方式之一启动隔空投送功能:在“设置”应用程序或控…

[dasctf]misc05

盲水印 png里藏jpg&#xff0c;bwm.py可以提取含flag的图片