Java线程池和原子性

news2024/11/17 8:42:47

文章目录

  • 前言
  • 1 线程池
    • 1.1 线程池概述
      • 1.1.1 线程池存在的意义
      • 1.1.2 Executors默认线程池
    • 1.2 线程状态介绍
      • 1.2.1 线程状态源码
      • 1.2.2 线程状态含义
      • 1.2.3 线程状态转换图
  • 2 原子性
    • 2.1 volatile关键字
    • 2.2 synchronized解决
    • 2.3 原子性
    • 2.4 AtomicInteger类
    • 2.5 悲观锁和乐观锁


前言

“清琴浊酒莺花日,雨笠烟蓑蟹稻乡。棠荫渐高身渐隐,已将心事托渔郎”   —张英 《赠何匡山次梅村先生韵》
在这里插入图片描述

1 线程池

1.1 线程池概述

       线程池也是可以看做成一个容器,在该容器中存储很多个线程。

1.1.1 线程池存在的意义

       系统创建一个线程的成本是比较高的,频繁的创建和销毁线程对系统的资源消耗非常大,所为了提高性能,我们就可以采用线程池。线程池在启动的时,会创建大量空闲线程,当我们向线程池提交任务的时,线程池就会启动一个线程来执行该任务。等待任务执行完毕以后,线程并不会死亡,而是再次返回到线程池中称为空闲状态。等待下一次任务的执行。

1.1.2 Executors默认线程池

       1.常见方法

// 1.创建一个默认的线程池
static ExecutorService newCachedThreadPool()
// 2.创建一个指定最多线程数量的线程池
static newFixedThreadPool(int nThreads)

       2.代码实现 :

package com.syh;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;


public class MyThread {
    public static void main(String[] args) throws InterruptedException {

        // 1,创建一个默认的线程池对象
        ExecutorService executorService = Executors.newCachedThreadPool();
        executorService.submit(()->{
            System.out.println(Thread.currentThread().getName() + "在执行了");
        });
        // 2,等待2秒
        Thread.sleep(2000);
        executorService.submit(()->{
            System.out.println(Thread.currentThread().getName() + "在执行了");
        });
        executorService.shutdown();
    }
}

1.2 线程状态介绍

       线程对象在不同的时期有不同的状态,一共有6种状态

1.2.1 线程状态源码

public class Thread {
    
    public enum State {
    
        /* 新建 */
        NEW , 

        /* 可运行状态 */
        RUNNABLE , 

        /* 阻塞状态 */
        BLOCKED , 

        /* 无限等待状态 */
        WAITING , 

        /* 计时等待 */
        TIMED_WAITING , 

        /* 终止 */
        TERMINATED;
    
	}
    
    // 获取当前线程的状态
    public State getState() {
        return jdk.internal.misc.VM.toThreadState(threadStatus);
    }
    
}

1.2.2 线程状态含义

线程状态具体含义
NEW一个尚未启动的线程的状态。也称之为初始状态、开始状态。线程刚被创建,但是并未启动。还没调用start方法。MyThread t = new MyThread()只有线程象,没有线程特征。
RUNNABLE当我们调用线程对象的start方法,那么此时线程对象进入了RUNNABLE状态。那么此时才是真正的在JVM进程中创建了一个线程,线程一经启动并不是立即得到执行,线程的运行与否要听令与CPU的调度,那么我们把这个中间状态称之为可执行状态(RUNNABLE)也就是说它具备执行的资格,但是并没有真正的执行起来而是在等待CPU的度。
BLOCKED当一个线程试图获取一个对象锁,而该对象锁被其他的线程持有,则该线程进入Blocked状态;当该线程持有锁时,该线程将变成Runnable状态。
WAITING一个正在等待的线程的状态。也称之为等待状态。造成线程等待的原因有两种,分别是调用Object.wait()、join()方法。处于等待状态的线程,正在等待其他线程去执行一个特定的操作。例如:因为wait()而等待的线程正在等待另一个线程去调用notify()或notifyAll();一个因为join()而等待的线程正在等待另一个线程结束。
TIMED_WAITING一个在限定时间内等待的线程的状态。也称之为限时等待状态。造成线程限时等待状态的原因有三种,分别是:Thread.sleep(long),Object.wait(long)、join(long)。
TERMINATED一个完全运行完成的线程的状态。也称之为终止状态、结束状态

1.2.3 线程状态转换图

在这里插入图片描述

2 原子性

2.1 volatile关键字

1. 问题 :

当A线程修改了共享数据时,B线程没有及时获取到最新的值,如果还在使用原先的值,就会出现问题

  • 堆内存是唯一的,每一个线程都有自己的线程栈。

  • 每一个线程在使用堆里面变量的时候,都会先拷贝一份到变量的副本中。

  • 在线程中,每一次使用是从变量的副本中获取的。

2. Volatile解决 : 强制线程每次在使用的时候,都会看一下共享区域最新的值,代码示例如下:

package com.syh;

public class Money{
    public static volatile int money = 1000;
}
package com.syh;

public class MyThread1 extends Thread{
    @Override
    public void run() {
        Money.money -= 100;
        System.out.println("李默然:" + Money.money);
    }
}
package com.syh;

public class MyThread2 extends Thread{
    @Override
    public void run() {
        System.out.println("林北辰:" + Money.money);
    }
}
package com.syh;

public class Test {
    public static void main(String[] args) {
        MyThread1 t1 = new MyThread1();
        t1.start();

        MyThread2 t2 = new MyThread2();
        t2.start();
    }
}

2.2 synchronized解决

1. synchronized解决步骤 :

  • 线程获得锁

  • 清空变量副本

  • 拷贝共享变量最新的值到变量副本中

  • 执行代码

  • 将修改后变量副本中的值赋值给共享数据

  • 释放锁

2. 代码实现 :

package com.syh;

public class Money{
    public static Object lock = new Object();
    public static int money = 1000;
}
package com.syh;

public class MyThread1 extends Thread{
    @Override
    public void run() {
        synchronized (Money.lock){
            Money.money -= 100;
            System.out.println("李默然: " + Money.money);
        }
    }
}
package com.syh;

public class MyThread2 extends Thread{
    @Override
    public void run() {
        synchronized (Money.lock){
            System.out.println("林北辰:" + Money.money);
        }
    }
}
package com.syh;

public class Test {
    public static void main(String[] args) {
            MyThread1 t1 = new MyThread1();
            t1.start();

            MyThread2 t2 = new MyThread2();
            t2.start();
        }
}

2.3 原子性

       原子性就是要么所有的操作都执行,要么所有的操作都不执行,多个操作是一个不可以分割的整体。

2.4 AtomicInteger类

java从JDK1.5开始提供了AtomicInteger类,这个包中的原子操作类提供了一种用法简单,性能高效,线程安全地更新一个变量的方式,常用方法如下:

public AtomicInteger()// 1.初始化一个默认值为0的原子型Integer
public AtomicInteger(int initialValue)// 2.初始化一个指定值的原子型Integer

int get():   			 				 // 3.获取值
int getAndIncrement():      			 // 4.以原子方式将当前值加1,注意,这里返回的是自增前的值。
int incrementAndGet():     				 // 5.以原子方式将当前值加1,注意,这里返回的是自增后的值。
int addAndGet(int data): // 6.以原子方式将输入的数值与实例中的值(AtomicInteger里的value)相加,并返回结果。
int getAndSet(int value):   			 // 7.以原子方式设置为newValue的值,并返回旧值。

代码实现 :

package com.syh;

import java.util.concurrent.atomic.AtomicInteger;

public class MyAtom {
    public static void main(String[] args) {
        AtomicInteger ac = new AtomicInteger();
        System.out.println(ac);

        AtomicInteger ac2 = new AtomicInteger(10);
        System.out.println(ac2);
    }
}

2.5 悲观锁和乐观锁

synchronized和CAS的区别是什么?

1. 相同点
       在多线程情况下,都可以保证共享数据的安全性。

2.不同点

  • 悲观锁:synchronized总是从最坏的角度出发,认为每次获取数据的时候,别人都有可能修改。所以在每次操作共享数据之前,都会上锁。

  • 乐观锁:cas是从乐观的角度出发,假设每次获取数据别人都不会修改,所以不会上锁。只不过在修改共享数据的时候,会检查一下,别人有没有修改过这个数据。

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

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

相关文章

TikTok五分钟开户快速步骤流程!

1、注册您的账户 首先,访问TikTok广告管理器的注册页面(https://ads.tiktok.com/i18n/signup/)以创建账户。您可以选择使用电子邮件或手机号码进行注册。输入您的电子邮件和密码后,您需要同意TikTok的广告条款,然后点击…

【路径规划】用于识别封闭多边形竞技场内两点之间的最短路径

摘要 本项目展示了一种在包含障碍物的封闭多边形区域内识别两点之间的最短路径的算法。该算法不依赖于离散化的地图(如网格地图),而是直接通过几何计算处理路径和障碍物之间的关系,生成沿着障碍物边缘的最优路径。实验结果表明&a…

软考中级网络工程师选择题

部分参考 软考中级网络工程师全面学习笔记第2版(5万字)配套视频及课件_软考中级网络工程师资料-CSDN博客 1.计算机网络概述 操作系统 OSI七层模型:物联网淑慧试用 TCP/IP:网网(网际层)传应 高频考点: 中央处理器CP…

数仓思想、数仓建模、维度建模理论、数仓面试题

文章目录 一、数仓建模1、数仓建模的意义2、数仓建模方法论1)ER模型:2)六范式: 3、维度模型 二、维度建模理论之事实表1、事务型事实表2、周期型快照事实表3、累计型快照事实表 三、维度建模理论之维度表1、设计步骤2、设计要点3、…

C++ 9.25

手动实现栈、和队列 stack #include <iostream> using namespace std; class Stack { private: int* arr; // 存储栈元素的数组 int top; // 栈顶索引 int capacity; // 栈的容量 public: Stack(int size) { arr new int[size]; c…

uni-app+vue3开发微信小程序使用本地图片渲染不出来报错[渲染层网络层错误]Failed to load local image resource

我把图片放在assets里面页面通过相对路径引入。结果一直报错。 最后我把图片放在static文件夹下面。然后修改路径指向static就可以了 或者是我们必须先import 这个图片然后在使用 import banner1 from ../../assets/images/banner/banner1.png; <image :src"banner…

戴尔迅速应对疑似信息泄露事件,强化数据保护措施|硬盘文件数据销毁 文件销毁 数据销毁

在全球数字化转型的浪潮中&#xff0c;数据已成为企业最宝贵的资产之一&#xff0c;而数据安全则是这一进程中的基石。近期&#xff0c;戴尔公司遭遇了一起疑似大规模信息泄露事件&#xff0c;涉及约4900万用户的信息&#xff0c;这一事件迅速引发了全球范围内的广泛关注。面对…

【开源】 mRemoteNG 一键搞定!推荐一款强大的.NET多协议远程连接管理器

今天给大家推荐一款.NET开发的多协议、选项卡式远程连接管理器mRemoteNG。 mRemoteNG 是 mRemote 的一个分支&#xff1a;一个开源的、标签式的、多协议的、用于 Windows 的远程连接管理器。 mRemoteNG是一个开源的Windows远程连接管理器&#xff0c;它支持多种协议&#xff0c…

交通锥检测系统源码分享

交通锥检测检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Computer Visio…

Python 使用selenium 4.25 进行爬虫(1)

都说python做爬虫比较好&#xff0c;于是我跟着大家的脚步学习python进行爬虫&#xff0c;但是调试了半天&#xff0c;出现各种各样的问题&#xff0c;最终都得到实现了&#xff0c;下面我们来看具体的代码&#xff1a; from selenium import webdriver from selenium.webdriv…

模拟银行操作

设计实现一个银行新用户现金业务办理程序&#xff0c;使其模拟新用户到银行办理现金存取业务时的场景。要求如下&#xff1a; (1)有对于用户到来的欢迎动作、用户离开的提醒动作 (2)有用户的开户、存款和取款动作&#xff0c;在完成开户存款和取款操作后&#xff0c;要提示用…

Godot C# 自定义摄像机

前言 说起来&#xff0c;Unity的社区环境跟插件支持确实要比Godot好很多&#xff0c;比如我们Unity最喜欢的Cinemachine插件&#xff0c;只需要动动手指就能轻松实现很多高级的摄像机动效。 所以一转到Godot就有一种力不从心的感觉&#xff0c;于是既然动不了手指我们就动手。自…

凤凰模拟器V6中无人机如何设置“有头模式”

凤凰模拟器是一款专为航模新手设计的飞行模拟器&#xff0c;它能够模拟大疆无人机、各种穿越机、固定翼等多种飞行器&#xff0c;提供逼真的飞行体验。该软件的操作简单易懂&#xff0c;适合新手练习使用。 一般来说&#xff0c;打开凤凰模拟器&#xff0c;选择好机型&#xf…

快速上手 Hugging Face Transformers:完整模型微调训练步骤全攻略

在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;模型微调&#xff08;Fine-Tuning&#xff09;是提升预训练模型在特定任务上表现的关键步骤。本文将详细介绍如何使用 Hugging Face Transformers 库进行模型微调训练&#xff0c;涵盖数据集下载、数据预处理、训练配…

Python发送邮件教程:如何实现自动化发信?

Python发送邮件有哪些方法&#xff1f;如何利用python发送邮件&#xff1f; 无论是工作汇报、客户通知还是个人提醒&#xff0c;邮件都能快速传递信息。Python发送邮件的自动化功能就显得尤为重要。AokSend将详细介绍如何使用Python发送邮件&#xff0c;实现自动化发信&#x…

Mysql 删除表的所有数据

在 MySQL 中&#xff0c;如果你想要删除一个表中的所有数据&#xff0c;可以使用 TRUNCATE TABLE 命令或者 DELETE 语句。下面是两种方法的对比以及如何使用它们&#xff1a; 使用 TRUNCATE TABLE TRUNCATE TABLE 是一个非常快速的方法来删除表中的所有记录&#xff0c;并且它…

我的领域-关怀三次元成长的二次元虚拟陪伴 | OPENAIGC开发者大赛高校组AI创作力奖

在第二届拯救者杯OPENAIGC开发者大赛中&#xff0c;涌现出一批技术突出、创意卓越的作品。为了让这些优秀项目被更多人看到&#xff0c;我们特意开设了优秀作品报道专栏&#xff0c;旨在展示其独特之处和开发者的精彩故事。 无论您是技术专家还是爱好者&#xff0c;希望能带给…

【从0开始自动驾驶】用python做一个简单的自动驾驶仿真可视化界面

【从0开始自动驾驶】用python做一个简单的自动驾驶仿真可视化界面 废话几句废话不多说&#xff0c;直接上源码目录结构init.pysimulator.pysimple_simulator_app.pyvehicle_config.json 废话几句 自动驾驶开发离不开仿真软件成品仿真软件种类多https://zhuanlan.zhihu.com/p/3…

介绍篇| 爬虫工具介绍

什么是网络爬虫 网络爬虫工具本质上是自动化从网站提取数据的软硬件或服务。它简化了网络爬虫&#xff0c;使信息收集变得更加容易。如今是数据和智能化时代, 如何快速、自动化获取数据, 成了个人或者企业进入智能化时代的第一步. 选择最佳网络爬虫工具时的关键因素 在选择最…

Apache DolphinScheduler-1.3.9源码分析(一)

引言 随着大数据的发展&#xff0c;任务调度系统成为了数据处理和管理中至关重要的部分。Apache DolphinScheduler 是一款优秀的开源分布式工作流调度平台&#xff0c;在大数据场景中得到广泛应用。 在本文中&#xff0c;我们将对 Apache DolphinScheduler 1.3.9 版本的源码进…