0

0

Java中什么是伪共享?有什么解决方案?

betcha

betcha

发布时间:2025-01-07 17:45:35

|

1111人浏览过

|

来源于php中文网

原创

伪共享:隐秘的性能杀手伪共享是指多个线程同时访问不同变量,但这些变量位于同一缓存行中。这会导致缓存行失效,降低性能。解决方案包括:缓存行填充:通过填充字段来增加变量之间的距离,避免其位于同一缓存行中。使用更细粒度的锁:只锁定真正需要访问的数据。重新设计数据结构:以减少伪共享的可能性。

Java中什么是伪共享?有什么解决方案?

Java里的伪共享:幽灵般的性能杀手

你是否曾经遇到过这种情况:你的多线程Java程序性能奇差,代码逻辑看似完美无缺,CPU占用率却居高不下?罪魁祸首,可能就是潜伏在内存深处的“伪共享”这个幽灵。

这篇文章会带你揭开伪共享的神秘面纱,深入剖析其成因,并提供一些行之有效的解决方案。读完之后,你将能够识别并解决由伪共享引起的性能瓶颈,编写出更高效的多线程Java代码。

伪共享的本质

简单来说,伪共享是指多个线程同时访问不同变量,但这些变量恰好位于同一缓存行中。由于缓存行是CPU缓存的基本单位,当一个线程修改了缓存行中的某个变量,即使其他线程访问的是不同的变量,整个缓存行也会失效,导致其他线程不得不重新从主内存加载数据,从而引发性能损失。这就好比一群人共用一个房间,一个人打翻了水杯,其他人也得跟着收拾残局。

立即学习Java免费学习笔记(深入)”;

这看似微不足道,但累积起来的影响却极其巨大。想象一下,在高并发环境下,成千上万个线程都在争抢同一个缓存行,那性能下降的程度可想而知。

缓存行与Java内存模型

要理解伪共享,我们需要对CPU缓存和Java内存模型有一些基本的了解。CPU缓存分层结构,L1、L2、L3缓存,速度依次降低,容量依次增大。缓存行是缓存的基本单位,大小通常是64字节。Java内存模型定义了主内存和工作内存,线程之间通过主内存进行数据交换。伪共享正是发生在缓存行和Java内存模型交互过程中的一个微妙问题。

一个简单的例子

思乐微信商城微分销系统
思乐微信商城微分销系统

思乐微信商城微分销系统是以.net+access/mssql进行开发的微信分销系统。基于微信朋友圈的传播,是打造以分销商为中心的全新微信分销体验。让粉丝实时有效的获取朋友圈流量并快速分享购买分佣。因为是基于微信,所以要在微信上体验才更好。关注我们的微信核心功能:1、自动提示用户关注微信,解决一般程序无关注微信公众号的过程2、只要通过链接进一次,不过好久注册,什么方面注册,只要是用微信注册的,都会算

下载

让我们来看一个简单的例子,假设有两个线程分别修改两个long类型的变量:

public class FalseSharing {

    private static final int NUM_THREADS = 4;
    private static final long ITERATIONS = 10000000L;

    private static volatile long[] longs = new long[NUM_THREADS];

    public static void main(String[] args) throws InterruptedException {
        Thread[] threads = new Thread[NUM_THREADS];
        for (int i = 0; i < NUM_THREADS; i++) {
            threads[i] = new Thread(() -> {
                long i1 = 0;
                for (long j = 0; j < ITERATIONS; j++) {
                    longs[ThreadLocalRandom.current().nextInt(NUM_THREADS)] += 1;
                    i1 += 1;
                }
            });
        }
        long start = System.currentTimeMillis();
        for (Thread t : threads) {
            t.start();
        }
        for (Thread t : threads) {
            t.join();
        }
        long end = System.currentTimeMillis();
        System.out.println("Time taken: " + (end - start) + "ms");
    }
}

在这个例子中,如果longs数组的元素恰好位于同一缓存行,那么就会发生伪共享。每个线程的更新都会导致缓存行失效,从而降低性能。

解决方案:缓存行填充

解决伪共享最有效的方法是通过缓存行填充来避免多个变量位于同一缓存行中。我们可以使用填充字段来增加变量之间的距离,确保它们位于不同的缓存行中:

public class FalseSharingSolution {

    private static final int NUM_THREADS = 4;
    private static final long ITERATIONS = 10000000L;

    private static class LongWrapper {
        long value;
        long p1, p2, p3, p4, p5, p6, p7; // Padding
    }

    private static volatile LongWrapper[] longs = new LongWrapper[NUM_THREADS];

    static {
        for (int i = 0; i < NUM_THREADS; i++) {
            longs[i] = new LongWrapper();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        // ... (rest of the code is similar to the previous example)
    }
}

在这个例子中,我们使用了LongWrapper类来包装long变量,并添加了填充字段p1p7。这些填充字段会占用足够的内存空间,确保value变量位于不同的缓存行中,从而避免伪共享。

其他解决方案与经验总结

除了缓存行填充,还有其他一些方法可以缓解伪共享问题,例如使用更细粒度的锁,或者重新设计数据结构。但是,缓存行填充通常是最简单有效的方案。

需要注意的是,伪共享并非总是显而易见的。它往往隐藏在复杂的代码中,需要仔细分析和调试才能发现。因此,在编写高性能多线程Java代码时,务必注意潜在的伪共享问题,并采取相应的措施进行优化。 经验告诉我,预先考虑潜在的伪共享问题,并进行合理的代码设计,比事后补救要高效得多。 不要等到性能问题爆发才去寻找原因,这会浪费大量的时间和精力。 记住,编写高效的代码,是一个持续学习和改进的过程。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

825

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

724

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

731

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

445

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

429

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16881

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号