Java线程池和原子性

文章目录

  • 前言
  • 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{@Overridepublic void run() {Money.money -= 100;System.out.println("李默然:" + Money.money);}
}
package com.syh;public class MyThread2 extends Thread{@Overridepublic 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{@Overridepublic void run() {synchronized (Money.lock){Money.money -= 100;System.out.println("李默然: " + Money.money);}}
}
package com.syh;public class MyThread2 extends Thread{@Overridepublic 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.初始化一个指定值的原子型Integerint 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.mzph.cn/diannao/54808.shtml

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

相关文章

Vue 响应式监听 Watch 最佳实践

一. 前言 上一篇文章我们学习了 watch 的基础知识,了解了它的基本使用方法及注意事项,本篇文章我们继续了解在Vue 中 响应式监听 watch 的妙用。了解 watch 的基础使用请参考上一篇文章: 详解 Vue 中 Watch 的使用方法及注意事项https://bl…

【Docker】02-数据卷

1. 数据卷 数据卷(volume) 是一个虚拟目录,是容器内目录与宿主机目录之间映射的桥梁。 2. 常见命令 docker volume createdocker volume lsdocker volume rmdocker volume inspect 查看某个数据卷的详情docker volume prune 清除数据卷 **数据卷挂载&#xff1a…

bench.sh:一行命令测试Linux服务器基准测试

简介 bench.sh 是一个 Linux 系统性能基准测试工具。它的测试结果如下图:给出服务器的整体配置信息,IO 性能,网络性能。很多人使用它测试 vps 性能。 ​​ 一键运行 服务器在国外可以使用以下命令运行测试 wget -qO- bench.sh | bash复制…

Ansible 剧本的执行

Ansible 剧本(playbook)是一种用于定义和自动化 IT 任务的工具,它使用 YAML 格式编写,能够描述一系列的任务和操作,以实现对服务器、网络设备等的配置管理和部署。 以下是关于 Ansible 剧本执行的一些关键方面&#xf…

从《GTA5》的反外挂斗争看网络安全的重要性

摘要: 在网络游戏的世界里,外挂(作弊软件)一直是破坏游戏公平性和玩家体验的一大难题。作为一款深受全球玩家喜爱的游戏,《GTA5》(Grand Theft Auto V)在线模式也不例外地遭遇了外挂问题。本文将…

微信小程序配置prettier+eslint

虽然微信开发者工具是基于vscode魔改的.但是由于版本过低,导致很多插件也用不上新版本.所以在微信开发者工具限制的版本下使用的prettier,eslint也是有版本要求. 本文主要就是记录一下需要的版本号 1.微信开发者工具安装插件 2.package.json中添加以下依赖及安装依赖 "de…

JVM(HotSpot):方法区(Method Area)

文章目录 一、内存结构图二、方法区定义三、内存溢出问题四、常量池与运行时常量池 一、内存结构图 1.6 方法区详细结构图 1.8方法区详细结构图 1.8后,方法区是JVM内存的一个逻辑结构,真实内存用的本地物理内存。 且字符串常量池从常量池中移入堆中。 …

云服务器连接不上是什么原因引起的?

云服务器连接不上是什么原因引起的?云服务器连接不上是一个常见的问题,常见的原因有网络连接、账户权限、安全组设置、服务器状态、端口占用、远程登录未开启、云服务器已关闭或到期、防护软件限制、DNS劫持、资源负载过高。以下是一些主要原因及解决方法…

微信小程序 - 最新详细安装使用 Vant weapp UI 框架环境搭建详细教程

前言 自从 2024 年开始,小程序做了很多改变和升级, 导致网上很多搭建教程文章的教程失效了,本文来做最新的教程。 第一步 为了更贴合新手,我这里创建了一个纯净无任何业务代码的小程序项目。

AI语音工具——Fish Speech:使用简单,可训练专属语音模型!

1. 引言 随着人工智能技术的迅速发展,AI语音合成与识别技术在智能助手、虚拟主播、自动化客服等领域展现出巨大的潜力。今天,专属定制化语音模型不再是大公司的专利,Fish Speech让普通用户也能轻松训练自己的语音模型。 Fish Speech 是一款…

SpringBoot-全局处理异常,时间格式,跨域,拦截器,监听器

1.全局异常处理 使用ControllerAdvice与ExceptionHandler注解 /*** 全局异常处理程序** author * date */ ControllerAdvice ResponseBody public class GlobalExceptionHandler {ExceptionHandler(Exception.class)public JsonResult handleException(Exception e) {e.print…

Wireshark学习使用记录

wireshark 是一个非常好用的抓包工具,使用 wireshark 工具抓包分析,是学习网络编程必不可少的一项技能。 原理 Wireshark使用的环境大致分为两种:一种是电脑直连互联网的单机环境,另外一种就是应用比较多的互联网环境,也就是连接…

Windows下安装Neo4j流程

Neo4j简介 Neo4j 是一个基于图形结构的 NoSQL 数据库,专门用于存储和管理图数据。与传统的关系型数据库不同,Neo4j 使用 图(graph)的形式来表示数据,其中数据点(称为 节点)通过 边(…

检测死锁的机制

PostgreSQL(PG)检测死锁的机制主要基于等待图(WFG, wait-for graph)和超时设置。以下是详细的死锁检测过程: 等待图构建:当一个事务在尝试获取资源时,如果未能立即获得所需的锁,并且…

使用lodash库实现防抖和节流

在 JavaScript 中,lodash 是一个非常流行的实用工具库,提供了许多常用的函数,其中就包括防抖和节流的实现。lodash 提供的 debounce 和 throttle 函数可以让我们更方便地实现这两种功能。 1. 使用 lodash 的 debounce(防抖&#…

【6】AT32F437 OpenHarmony轻量系统移植教程(3)

开源地址:https://gitee.com/AT32437_OpenHarmony 学习本文档的意义 1.学习移植OpenHarmony轻量系统到AT32全系列mcu上,本文档移植的具体型号为AT32F437ZMT7 2.学习OpenHarmony轻量系统开发 6.target_config.h文件适配 在//kernel/liteos_m/kernel/…

秋招喜来,分享一份最新的面经

💼岗位职责: 负责后端服务的开发与维护;技术预研和技术难点攻关,保障系统可用性、稳定性、和可扩展性;持续优化系统架构设计,提出问题解决方案。 📋任职要求: 计算机或相关专业&a…

分拆逆势上会,兴福电子部分专利来自母公司,独立性被拷问

作者:Cindy 来源:IPO魔女 公司拟募集资金12.10亿元,保荐机构为天风证券。兴福电子2023年5月就提交了上市申请,期间经历了2轮问询,一年多时间就开启上会。 然而值得注意的是,兴福电子属于分拆上市&#xff…

亚马逊IP关联揭秘:发生ip关联如何处理

在亚马逊这一全球领先的电商平台上,IP关联是一个不可忽视的问题,尤其是对于多账号运营的卖家而言。本文将深入解析亚马逊IP关联的含义、影响以及应对策略,帮助卖家更好地理解和应对这一问题。 什么是亚马逊IP关联? 亚马逊IP关联…

Java高级Day51-apacheDBUtils

136.ApDBUtils 基本介绍 commons-dbutils 是 Apache 组织提供的一个开源JDBC工具类库,它是对JJDBC的封装,使用dbutils能极大简化jdbc编码的工作量 QueryRunner类:该类封装了SQL的执行,是线程安全的。可以实现增删改查和批处理 …