单线程与多线程的理解与学习(入门到深入)

在这里插入图片描述

文章目录

      • 一、在Java中,有多种方式可以创建线程。以下是几种常用的方法:
      • 二、线程的调度
        • 线程的调度分为两种调度模型
          • 分时调度模型
          • 抢占式调度模型
      • 三、线程传值
      • 四、什么是线程同步
      • 五、线程安全
      • 六、线程的同步机制
      • 七、线程控制

一、在Java中,有多种方式可以创建线程。以下是几种常用的方法:

  1. 继承Thread类:创建一个继承自Thread类的子类,并重写run()方法,在run()方法中定义线程的执行逻辑。然后通过创建子类对象,调用start()方法来启动线程。示例代码如下:
class MyThread extends Thread {
    public void run() {
        // 线程的执行逻辑
    }
}
// 创建并启动线程
MyThread thread = new MyThread();
thread.start();
  1. 实现Runnable接口:创建一个实现了Runnable接口的类,并实现其run()方法,在run()方法中定义线程的执行逻辑。然后通过创建Runnable实现类的对象,再创建Thread对象,并将Runnable实现类的对象作为参数传递给Thread的构造方法。最后调用Thread对象的start()方法来启动线程。示例代码如下:
class MyRunnable implements Runnable {
    public void run() {
        // 线程的执行逻辑
    }
}
// 创建并启动线程
MyRunnable myRunnable = new MyRunnable();
Thread thread = new Thread(myRunnable);
thread.start();
  1. 使用匿名内部类:通过创建匿名内部类来实现Runnable接口,并实现其run()方法。然后创建Thread对象,并将匿名内部类对象作为参数传递给Thread的构造方法。最后调用Thread对象的start()方法来启动线程。示例代码如下:
Runnable runnable = new Runnable() {
    public void run() {
        // 线程的执行逻辑
    }
};
// 创建并启动线程
Thread thread = new Thread(runnable);
thread.start();
  1. 使用线程池:通过Java提供的Executor框架可以方便地创建线程池和管理线程。使用线程池可以更好地控制线程的数量和资源消耗。示例代码如下:
ExecutorService executor = Executors.newFixedThreadPool(5); // 创建一个固定大小的线程池,最多同时执行5个线程
Runnable runnable = new Runnable() {
    public void run() {
        // 线程的执行逻辑
    }
};
executor.execute(runnable); // 提交任务给线程池执行
executor.shutdown(); // 关闭线程池

以上是几种常见的创建线程的方式,根据具体的需求选择合适的方式来创建线程。注意,在使用多线程时,要注意线程安全和资源共享的问题。

二、线程的调度

假如我们的电脑只有一个CPU,那么CPU在某一时刻只能执行一个指令,其他程序只有分到CPU的时间片(也就是使用权),才能够执行指令。

线程的调度分为两种调度模型

分时调度模型

所有线程轮流使用CPU的使用时间,平均分配每个线程占用CPU的时间片。

抢占式调度模型

优先让优先级的高的线程使用CPU,如果优先级相同,随机选择一个线程,优先级高的线程所获取的CPU时间片会相对多一些。

线程调度是操作系统或者编程语言的任务调度器决定在多个线程之间分配处理器执行时间的过程。线程调度的目标是合理利用CPU资源,提高系统的吞吐量和响应速度。

在Java中,线程调度是由Java虚拟机(JVM)的线程调度器负责的。JVM的线程调度器按照一定的策略来决定哪个线程获得执行时间,具体的调度策略可能因不同的JVM实现而有所不同。

Java线程调度器使用抢占式调度方式。抢占式调度意味着较高优先级的线程可以抢占正在执行的线程的CPU执行时间,以确保高优先级任务的及时执行。线程的优先级用整数表示,范围从1到10,默认为5。可以使用Thread类的setPriority()方法设置线程的优先级。

Java线程调度器基于时间片轮转的调度算法,每个线程被分配一个时间片段,在时间片段结束后,线程调度器会切换到下一个线程继续执行。时间片的长度可以通过操作系统或者JVM的设置进行调整。

除了线程优先级和时间片轮转算法,Java还提供了一些其他的线程调度方法,例如:

  1. yield()方法:调用yield()方法可以让当前线程主动让出CPU执行时间,使得其他具有相同优先级的线程有机会执行。
  2. sleep()方法:调用sleep()方法可以使线程暂停执行一段时间,让其他线程有机会执行。
  3. join()方法:调用join()方法可以让一个线程等待另一个线程执行完毕后再继续执行。
    线程调度是一个复杂的主题,涉及到多个因素,包括线程优先级、时间片分配、同步机制等。了解线程调度的原理和方法,可以帮助开发人员编写高效且可靠的多线程程序。

三、线程传值

在线程之间传递值可以通过以下几种方式实现:

  1. 构造函数或方法参数传递:可以在创建线程的时候,将需要传递的值通过构造函数或者方法参数传递给线程。线程在执行时就可以使用这些传递的值。示例代码如下:
class MyThread extends Thread {
    private int value;
    public MyThread(int value) {
        this.value = value;
    }
    public void run() {
        // 使用传递的值
        System.out.println("Value: " + value);
    }
}
// 创建并启动线程,传递值
int value = 10;
MyThread thread = new MyThread(value);
thread.start();
  1. 实例变量传递:可以在创建线程后,通过设置线程对象的实例变量来传递值。示例代码如下:
class MyThread extends Thread {
    private int value;
    public void setValue(int value) {
        this.value = value;
    }
    public void run() {
        // 使用传递的值
        System.out.println("Value: " + value);
    }
}
// 创建并启动线程,设置实例变量的值
MyThread thread = new MyThread();
int value = 10;
thread.setValue(value);
thread.start();
  1. 使用ThreadLocal:ThreadLocal类可以在每个线程中维护一个变量的副本,每个线程可以独立地访问自己的副本,实现了线程间的数据隔离。示例代码如下:
ThreadLocal<Integer> threadLocal = new ThreadLocal<>();
Runnable runnable = new Runnable() {
    public void run() {
        // 设置线程的变量值
        threadLocal.set(10);
        // 获取线程的变量值
        System.out.println("Value: " + threadLocal.get());
        // 清除线程的变量值
        threadLocal.remove();
    }
};
// 创建并启动线程
Thread thread = new Thread(runnable);
thread.start();

通过上述方法,可以在不同的线程间传递值,并在线程内部获取和使用这些传递的值。注意在多线程场景下,要注意线程安全和数据一致性的问题。

四、什么是线程同步

线程同步是一种机制,用于协调多个线程之间的执行顺序和共享资源的访问,以避免出现数据不一致或不确定的结果。
在多线程环境下,如果多个线程同时访问和修改共享资源,可能会导致以下问题:

  1. 竞态条件(Race Condition):多个线程同时访问和修改同一个共享变量,导致最终结果的不确定性。
  2. 数据不一致(Data Inconsistency):多个线程对共享数据进行操作,其中一个线程的操作可能会影响到其他线程的操作,导致数据的不一致性。
    线程同步的目的是通过限制多个线程的执行顺序,并提供互斥访问共享资源的机制,确保线程之间的协调和数据的一致性。
    常见的线程同步机制包括:
  3. 锁机制:使用锁(如synchronized关键字和ReentrantLock类)来保护共享资源的访问,确保同一时刻只有一个线程可以访问共享资源,从而避免竞态条件和数据不一致。
  4. 条件变量:使用条件变量(如wait()、notify()、notifyAll())来实现线程之间的等待和唤醒机制,以实现有序访问和协作。
  5. 原子操作:使用原子类(如AtomicInteger、AtomicReference等)来实现对共享变量的原子操作,保证操作的原子性,避免竞态条件和数据不一致。
  6. volatile关键字:使用volatile关键字来保证共享变量的可见性,确保多个线程对变量进行操作时能够看到最新的值。
    线程同步的实现方式取决于具体的需求和场景。需要根据情况选择合适的同步机制,并进行必要的测试和验证,以确保线程的安全和数据的一致性。同时,也需要注意避免死锁、活锁等问题,以保证程序的正常执行。

五、线程安全

线程安全是指在多线程环境下,对共享资源的访问操作能够正确、可靠地执行,不会出现数据不一致或不确定的结果。
在多线程编程中,线程安全是一个重要的概念,需要特别注意以下几个方面:

  1. 原子性(Atomicity): 原子操作是指一个操作是不可中断的,要么全部执行成功,要么全部不执行。例如,对于一个整型变量的自增操作,如果不是线程安全的,可能会出现多个线程同时读取并修改值,导致结果不正确。可以使用原子类(Atomic classes)或加锁(synchronized、Lock)来保证原子操作。
  2. 可见性(Visibility): 可见性是指当一个线程修改了共享变量的值,其他线程能够立即看到该变化。使用volatile关键字可以保证变量的可见性。此外,使用synchronized或Lock机制也可以确保线程间的可见性。
  3. 有序性(Ordering): 有序性是指程序按照一定的顺序来执行。在多线程环境下,由于线程的调度和指令重排等因素,可能会导致程序执行顺序的不确定性。可以使用volatile关键字或同步机制来保证指令的有序性。
  4. 线程安全的数据结构和类:Java提供了一些线程安全的数据结构和类,如ConcurrentHashMap、CopyOnWriteArrayList等,它们在多线程环境下能够安全地进行操作,并且不需要额外的同步。
    要保证线程安全,可以采取以下几种方式:
  • 使用线程安全的数据结构和类。
  • 使用锁(synchronized、Lock)来保护共享资源的访问。
  • 使用volatile关键字来保证变量的可见性。
  • 避免使用可变的共享对象,尽量使用不可变对象。
  • 合理设计线程安全的算法和逻辑。
    需要根据具体的场景和需求选择合适的线程安全解决方案,并进行必要的测试和验证,以确保程序在多线程环境下的正确性和稳定性。

六、线程的同步机制

线程的同步机制是一种用于控制多个线程之间访问共享资源的机制,确保线程之间的正确交互和数据一致性。常见的线程同步机制包括:

  1. synchronized关键字:synchronized关键字可以用于方法和代码块。当一个线程进入synchronized修饰的方法或代码块时,它会获得对象的锁,其他试图获取锁的线程将被阻塞,直到该线程释放锁。synchronized关键字确保了同一时刻只有一个线程可以访问被锁定的代码,从而保证了线程的安全性。
    示例代码:
public synchronized void synchronizedMethod() {
    // 同步方法
}
public void method() {
    synchronized (this) {
        // 同步代码块
    }
}
  1. ReentrantLock类:ReentrantLock是Java提供的一个可重入锁,它提供了更多的灵活性和功能。与synchronized不同,ReentrantLock需要手动地进行加锁和释放锁的操作,可以更灵活地控制锁的获取和释放。它还提供了可定时的、可中断的锁等功能,使得线程同步更加灵活。
    示例代码:
ReentrantLock lock = new ReentrantLock();
public void method() {
    lock.lock();
    try {
        // 同步代码
    } finally {
        lock.unlock();
    }
}
  1. volatile关键字:volatile关键字用于修饰共享变量,保证了变量的可见性。当一个线程修改了volatile变量的值,其他线程可以立即看到该变化。然而,volatile关键字无法保证复合操作的原子性,也无法解决竞态条件问题。
    示例代码:
private volatile int sharedVariable;
public void method() {
    // 修改共享变量
    sharedVariable = 10;
    // 读取共享变量
    int value = sharedVariable;
}
  1. wait()和notify()/notifyAll()方法:wait()方法用于线程间的等待,notify()和notifyAll()方法用于线程间的唤醒。这些方法通常与synchronized关键字一起使用,用于实现线程间的协作和同步。通过wait()方法,线程可以主动释放对象的锁,并进入等待状态,直到其他线程调用notify()或notifyAll()方法来唤醒它。
    示例代码:
Object lock = new Object();
public void producer() throws InterruptedException {
    synchronized (lock) {
        // 生产数据
        lock.notify(); // 唤醒消费者线程
    }
}
public void consumer() throws InterruptedException {
    synchronized (lock) {
        lock.wait(); // 等待生产者线程唤醒
        // 消费数据
    }
}

这些线程同步机制可以根据具体的需求和场景进行选择和使用,确保线程之间的正确交互和共享资源的安全访问。同时,也需要注意避免死锁、竞态条件等问题,以确保线程的正常执行。

七、线程控制

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

8.4Java EE——基于注解的AOP实现

Spring AOP的注解 元素 描述 Aspect 配置切面 Pointcut 配置切点 Before 配置前置通知 After 配置后置通知 Around 配置环绕方式 AfterReturning 配置返回通知 AfterThrowing 配置异常通知 下面通过一个案例演示基于注解的AOP的实现&#xff0c;案例具体实现…

全志F1C200S嵌入式驱动开发(调整cpu频率和dram频率)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】 f1c200s默认的cpu频率是408M,默认的dram频率是156M。这两个数值,坦白说,都算不上特别高的频率。因为我们的晶振是24M输入,所以408/24=17,相当于整个cpu的频率只是晶振倍频了17…

node.js 爬虫图片下载

主程序文件 app.js 运行主程序前需要先安装使用到的模块&#xff1a; npm install superagent --save axios要安装指定版,安装最新版会报错&#xff1a;npm install axios0.19.2 --save const {default: axios} require(axios); const fs require(fs); const superagent r…

别在找git报错的解决方案啦,多达20条git错误解决方案助你学习工作

1. 找不到Git命令 $ sudo apt-get update $ sudo apt-get install git2. 无法克隆远程仓库 $ git clone https://github.com/username/repo.git3. 无法拉取或推送到远程仓库 $ git pull origin master $ git add . $ git commit -m "Resolve conflicts" $ git pus…

StableDiffusion 换脸实现

先看效果&#xff1a; 想要换的脸&#xff1a; 想要把脸放到的目标图片&#xff1a; 实现方案&#xff1a; StableDiffusionroop&#xff08;本次实验基于roopV0.02版本&#xff09; 1/安装SD&#xff0c;模型选择 DreamShaper,Sampler使用 Euler a 2/安装roop插件 roop插…

【隐式动态求解】使用非线性纽马克方法的隐式动态求解研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

【ARM Coresight 系列文章 10.2 - ARM Coresight STM Trace packets】

文章目录 Trace protocolpacket的种类Error packetsVERSION Packets同步 packet 上篇文章&#xff1a;ARM Coresight 系列文章 10.1 - ARM Coresight STM 介绍及使用 下篇文章&#xff1a;ARM Coresight 系列文章 10.3 - ARM Coresight STM 寄存器介绍 及STM DMA 传输介绍 Trac…

【数据结构】树状数组和线段树

树状数组和线段树 下文为自己的题解总结&#xff0c;参考其他题解写成&#xff0c;取其精华&#xff0c;做以笔记&#xff0c;如有描述不清楚或者错误麻烦指正&#xff0c;不胜感激&#xff0c;不喜勿喷&#xff01; 树状数组 需求&#xff1a; 能够快速计算区间和保证在修改…

CSS背景虚化

.mark{background-color: rgba(0,0,0,.1);-webkit-backdrop-filter: blur(3px);backdrop-filter: blur(3px); }backdrop-filter CSS 属性可以让你为一个元素后面区域添加图形效果&#xff08;如模糊或颜色偏移&#xff09;。 因为它适用于元素背后的所有元素&#xff0c;为了看…

到底什么是前后端分离

目录 Web 应用的开发主要有两种模式&#xff1a; 前后端不分离 前后端分离 总结 Web 应用的开发主要有两种模式&#xff1a; 前后端不分离 前后端分离 理解它们的区别有助于我们进行对应产品的测试工作。 前后端不分离 在早期&#xff0c;Web 应用开发主要采用前后端不…

【C#】并行编程实战:异步流

本来这章该讲的是 ASP .NET Core 中的 IIS 和 Kestrel &#xff0c;但是我看了下这个是给服务器用的。而我只是个 Unity 客户端程序&#xff0c;对于服务器的了解趋近于零。 鉴于我对服务器知识和需求的匮乏&#xff0c;这里就不讲原书&#xff08;大部分&#xff09;内容了。本…

前端面试题 —— Vue (二)

目录 一、过滤器的作用&#xff0c;如何实现一个过滤器 二、v-model 是如何实现的&#xff0c;语法糖实际是什么&#xff1f; 三、$nextTick 原理及作用 四、Vue 中给 data 中的对象属性添加一个新的属性时会发生什么&#xff1f;如何解决&#xff1f; 五、简述 mixin、ex…

【C# 数据结构】Heap 堆

【C# 数据结构】Heap 堆 先看看C#中有那些常用的结构堆的介绍完全二叉树最大堆 Heap对类进行排序实现 IComparable<T> 接口 对CompareTo的一点解释 参考资料 先看看C#中有那些常用的结构 作为 数据结构系类文章 的开篇文章&#xff0c;我们先了解一下C# 有哪些常用的数据…

【防火墙】iptables防火墙(一)

防火墙具有隔离功能 主要部署在网络边缘或者主机边缘&#xff0c;防火墙的主要作用是决定哪些数据可以被外网访问&#xff0c;哪些数据可以进入内网访问 网络层&#xff08;路由器&#xff09;&#xff1a;数据的转发 安全技术 1.入侵监测系统&#xff1a;在检测到威胁&…

城市气象数据可视化:洞察气候变化,构建智慧城市

随着城市化进程的加速&#xff0c;城市气象数据的采集和分析变得越来越重要。气象数据不仅影响着人们的生活和出行&#xff0c;还与城市的发展和规划息息相关。在数字化时代&#xff0c;如何将城市中各个气象数据进行可视化&#xff0c;让复杂的数据变得简单易懂&#xff0c;成…

全国大学生数据统计与分析竞赛2021年【本科组】-B题:用户消费行为价值分析

目录 摘 要 1 任务背景与重述 1.1 任务背景 1.2 任务重述 2 任务分析 3 数据假设 4 任务求解 4.1 任务一&#xff1a;数据预处理 4.1.1 数据清洗 4.1.2 数据集成 4.1.3 数据变换 4.2 任务二&#xff1a;对用户城市分布情况与分布情况可视化分析 4.2.1 城市分布情况可视化分析 4…

微信小程序客服系统-对接消息推送-对接模板订阅消息-嵌入webview客服链接

想要给自己的小程序增加客服系统功能 小程序客服对接导自己的系统等需求&#xff0c;可以参照我开发的客服系统&#xff0c;实现私有化部署搭建对接的微信小程序 小程序消息推送对接 首先登录小程序后台在小程序后台>开发管理>开发设置>服务器域名部分&#xff0c;配置…

使用TensorFlow训练深度学习模型实战(下)

大家好&#xff0c;本文接TensorFlow训练深度学习模型的上半部分继续进行讲述&#xff0c;下面将介绍有关定义深度学习模型、训练模型和评估模型的内容。 定义深度学习模型 数据准备完成后&#xff0c;下一步是使用TensorFlow搭建神经网络模型&#xff0c;搭建模型有两个选项…

Java-运算符

目录 一、什么是运算符 二、算术运算符 1.基本四则运算符&#xff1a;加减乘除&#xff08;、-、*、/、%&#xff09; 2.增量运算符&#xff08;、-、*、%&#xff09; 3.自增、自减运算符&#xff08;、--&#xff09; 三、关系运算符 四、逻辑运算符 1.逻辑与 && …

Vue前端渲染blob二进制对象图片的方法

近期做开发&#xff0c;联调接口。接口返回的是一张图片&#xff0c;是对二进制图片处理并渲染&#xff0c;特此记录一下。 本文章是转载文章&#xff0c;原文章&#xff1a;Vue前端处理blob二进制对象图片的方法 接口response是下图 显然&#xff0c;获取到的是一堆乱码&…