首页
技术小册
AIGC
面试刷题
技术文章
MAGENTO
云计算
视频课程
源码下载
PDF书籍
「涨薪秘籍」
登录
注册
第一章:相关概念
1.1 并发和并行
1.2 进程和线程
1.3 线程调度
1.4 多线程的应用场景
第二章:线程的创建和使用
2.1 概述
2.1 JDK 5之前创建线程的方式
2.1.1 概述
2.1.2 继承 Thread 类
2.1.3 实现 Runnable 接口
2.1.4 继承方式和实现方式的区别
2.2 设置线程名称和获取线程名称
2.3 线程的优先级
第三章:线程安全
3.1 售票程序引出线程安全
3.2 线程安全的原因
3.3 线程安全的解决方法
3.4 JDK 5 之前解决线程安全
3.4.1 概述
3.4.2 同步机制的原理
3.4.3 同步机制中的锁
3.4.4 应用示例
3.5 JDK 5 之后解决线程安全
3.6 死锁
3.7 synchronized 与 Lock 的对比
第四章:线程通信
4.1 为什么需要线程通信?
4.2 等待唤醒机制
4.3 生产者和消费者问题
4.4 sleep() 和 wait() 方法的区别
4.5 生产者和消费者性能问题
4.6 Lock 接口的深入
第五章:单例设计模式
5.1 概述
5.2 单例模式
第六章:线程池
6.1 概述
6.2 线程池相关的API
6.3 实现 Callable 接口
6.4 ThreadPoolExecutor
第七章:线程的生命周期
第八章:volatile 关键字
8.1 概述
8.2 应用示例
第九章:原子性
9.1 概述
9.2 AtomicInteger
9.2.1 常用方法
9.2.2 原理
9.2.3 源码解析
9.3 synchronized 和 CAS 的区别
第十章:并发工具类
10.1 Hashtable
10.2 ConcurrentHashMap
10.3 CountDownLatch
10.4 Semaphore
当前位置:
首页>>
技术小册>>
Java语言基础8-Java多线程
小册名称:Java语言基础8-Java多线程
- 所谓的原子性指的是一次操作或多次操作,要么所有的操作全部得到了执行并且不会受到任何因素的干扰而中断,要么所有的操作都不执行,多个操作是一个不可分割的整体 。 - 示例:引入原子性问题 ```bash package com.github.demo13; /** * @author maxiaoke.com * @version 1.0 * */ public class SubRunnable implements Runnable { private int count; @Override public void run() { for (int i = 0; i < 100; i++) { /* * count++; * * ① 从共享数据中读取数据到本线程栈中 * ② 修改本线程栈中变量副本的值 * ③ 将本线程栈中变量副本的值赋值给共享数据 */ this.count++; System.out.println("已经送了" + this.count + "个冰淇淋"); } } } ``` ```bash package com.github.demo13; /** * @author maxiaoke.com * @version 1.0 * */ public class Test { public static void main(String[] args) { Runnable r = new SubRunnable(); for (int i = 0; i < 100; i++) { new Thread(r).start(); } // 有问题,不应该是10000吗,咋是9999。 // 是因为count++不是原子性操作,他在执行的过程中,可能会被其他线程打断操作。 } } ``` ![](/uploads/images/20230725/b1542fc72525afe9759ff8f362d6e87c.png) ![](/uploads/images/20230725/e9be8fa16eedf8a34c110c708d46c1a6.gif) 示例:volatile 关键字只能保证每次使用共享数据的时候是最新值,但是不能保证原子性 ```bash package com.github.demo13; /** * @author maxiaoke.com * @version 1.0 * */ public class SubRunnable implements Runnable { private volatile int count; @Override public void run() { for (int i = 0; i < 100; i++) { /* * count++; * * ① 从共享数据中读取数据到本线程栈中 * ② 修改本线程栈中变量副本的值 * ③ 将本线程栈中变量副本的值赋值给共享数据 */ this.count++; System.out.println("已经送了" + this.count + "个冰淇淋"); } } } ``` ```bash package com.github.demo13; /** * @author maxiaoke.com * @version 1.0 * */ public class Test { public static void main(String[] args) { Runnable r = new SubRunnable(); for (int i = 0; i < 100; i++) { new Thread(r).start(); } } } ``` ![](/uploads/images/20230725/5c4f2270bed147a373b9bfdda4fda3d1.png) ![](/uploads/images/20230725/8de24975c5ce3459a02224122924a4a0.gif) JDK 5 之后引入了新的包 java.util.concurrent.atomic(atomic:原子的),其中有很多类可以用来保证原子性。 ![](/uploads/images/20230725/3609c85a8250977184144ea4828f22ca.png)
上一篇:
第九章:原子性
下一篇:
9.2 AtomicInteger
该分类下的相关小册推荐:
java源码学习笔记
Java语言基础9-常用API和常见算法
Java语言基础15-单元测试和日志技术
Java语言基础13-类的加载和反射
Java语言基础4-数组详解
Java语言基础7-Java中的异常
深入拆解 Java 虚拟机
Java语言基础16-JDK8 新特性
手把手带你学习SpringBoot-零基础到实战
经典设计模式Java版
Java面试指南
Java语言基础11-Java中的泛型