造成开销的操作包含:
1. 线程之间的协调(比如:锁、触发信号以及内存同步等)
2. 添加�的上下文切换
3. 线程的创建和销毁
4. 线程的调度

一、对性能的思考

1 性能与可伸缩性

执行速度涉及下面两个指标:
某个指定的任务单元须要“多快”才干处理完毕、计算资源一定的情况下,能完毕“多少”工作。

可伸缩性:
当添加�计算资源时(比如:CPU、内存、存储容器或I/O带宽),程序的吞吐量或者处理能力能对应地添加�。


2 评估各种性能权衡因素
避免不成熟的优化。首先使程序正确,然后再提高执行速度---假设它还执行得不够快。
以測试为基准,不要推測。
提出问题:比如“更快”的含义是什么?提升多少效率?

二、并发三大定律

Amdahl 定律

Gene Amdahl 发如今计算机体系架构设计过程中,某个部件的优化对整个架构的优化和改善是有上限的。这个发现后来成为知名的 Amdahl 定律。

(即使你有10个老婆,也不能一个月把孩子生下来。)



Gustafson 定律

Gustafson如果随着处理器个数的添加�,并行与串行的计算总量也是能够添加�的。Gustafson定律觉得加速系数差点儿跟处理器个数成正比,如果现实情况符合Gustafson定律的如果前提的话,那么软件的性能将能够随着处理个数的添加�而添加�。

(当你有10个老婆,就会要生很多其它的孩子。)



Sun-Ni 定律

充分利用存储空间等计算资源,尽量增大问题规模以产生更好/更精确的解。

(你要设法让每一个老婆都在干活,别让她们闲着。 )


1 演示样例:在各种框架中隐藏的串行部分

2 Amdahl定律的应用


多线程中串行部分是性能提升的瓶颈,比如:
多线程在同一个队列中取出任务,由于须要保证线程安全肯定在队列上加锁,此时就是多线程中串行部分。
多线程一般是处理一些计算,而计算结果可能须要多个线程间进行共享,这也是多线程中串行部分。

三、线程引入的开销

1 上下文切换

大多数通用的处理器中,上下文切换的开销相当于5000~10000个时钟周期(几微妙)
UNIX系统的vmstat、mpstat命令和Windows系统的perfmon工具都能报告上下文切换次数以及和内核中运行时间所占比例等信息。


2 内存同步
在 synchronized 和 volatile 提供的可见性保证中可能会使用一些特殊指令,即内存栅栏(Memory Barrier)。
public String getStoogeNames(){
      Vector<String> stooges = new Vector<String>();
      stooges.add( "Moe" );
      stooges.add( "Larry" );
      stooges.add( "Curly" );
       return stooges.toString();
}

在运行getStoogeNames中,至少将Vector上的锁获取/释放4此,3次add操作与1次toString操作。
JVM会把在一起操作进行合并,可能仅须要获取1次add锁与1次toString锁。


3 堵塞

四、降低锁的竞争

在并发程序中,对可伸缩性的最主要威胁就是独占方式的资源锁。

有3中方式能够减少锁的竞争程度:

1. 降低锁的持有时间。

2. 减少锁的请求频率。

3. 使用带有协调机制的独占锁,这些机制同意更高的并发性。


1 缩小锁的范围(“快进快出”)
假设一个方法中,仅有一个变量是须要多线程间共享的,不须要在方法上加入�synchronized,由于这样会直接锁住整个方法导致其多线程间穿行运行,能够通过方法中仅锁住对共享变量操作的部分来缩小锁的范围提高性能。


2 减小锁的粒度(锁分解)
在一个分装中,假设分别提供的多个方法是分别对多个数据源操作,最严谨的方式是在全部方法上都把当前类作为锁定条件,可是能够通过在每一个数据源上加入�一个独立的锁。


3 锁分段
上一个中是一个类中涉及到多个数据源,假设仅有一个数据源(比如:Map)怎样提高性能?
在数据源上加入�分段锁,比如把map的个数除以4,4份中每一份是用一个单独的锁来锁定。


4 避免热点域
前面提到的一般是针对一个变量、一个数据集合、多个数据集合提高性能的办法,可是有些情况下一个方法内涉及到多个变量或者同一个变量的多个操作,能够通过降低这样的情况出现的次数提升性能。


5 一些替代独占锁的方法
通过放弃独占锁来提升性能。如并发容器,ReadWriteLock,不可变对象以及原子变量。


6 监測CPU的利用率
CPU没有得到充分利用的原因:

. 负载不充足

. I/O密集

. 外部限制

. 锁竞争


7 向对象池说“不”

对象分配操作的开销比同步的开销更低。



五、演示样例:比較Map的性能

六、降低上下文切换的开销

參考资料:

并发三大定律 摘录自《温绍锦 - Java并发程序设计教程》

《Java并发编程实战》第十一章 性能与可伸缩性 读书笔记的更多相关文章

  1. Java并发编程实战 第11章 性能与可伸缩性

    关于性能 性能的衡量标准有很多,如: 服务时间,等待时间用来衡量程序的"运行速度""多快". 吞吐量,生产量用于衡量程序的"处理能力",能够 ...

  2. 《Java并发编程实战》第十章 避免活跃性危急 读书笔记

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/love_world_/article/details/27635333 一.死锁 所谓死锁: 是指两 ...

  3. 《Java并发编程实战》第八章 线程池的使用 读书笔记

    一.在任务与运行策略之间的隐性解耦 有些类型的任务须要明白地指定运行策略,包含: . 依赖性任务.依赖关系对运行策略造成约束.须要注意活跃性问题. 要求线程池足够大,确保任务都能放入. . 使用线程封 ...

  4. Java并发编程实战---第六章:任务执行

    废话开篇 今天开始学习Java并发编程实战,很多大牛都推荐,所以为了能在并发编程的道路上留下点书本上的知识,所以也就有了这篇博文.今天主要学习的是任务执行章节,主要讲了任务执行定义.Executor. ...

  5. Java并发编程实战 第16章 Java内存模型

    什么是内存模型 JMM(Java内存模型)规定了JVM必须遵循一组最小保证,这组保证规定了对变量的写入操作在何时将对其他线程可见. JMM为程序中所有的操作定义了一个偏序关系,称为Happens-Be ...

  6. 【java并发编程实战】第一章笔记

    1.线程安全的定义 当多个线程访问某个类时,不管允许环境采用何种调度方式或者这些线程如何交替执行,这个类都能表现出正确的行为 如果一个类既不包含任何域,也不包含任何对其他类中域的引用.则它一定是无状态 ...

  7. Java并发编程实战 第8章 线程池的使用

    合理的控制线程池的大小: 下面内容来自网络.不过跟作者说的一致.不想自己敲了.留个记录. 要想合理的配置线程池的大小,首先得分析任务的特性,可以从以下几个角度分析: 任务的性质:CPU密集型任务.IO ...

  8. java并发编程实战:第二章----线程安全性

    一个对象是否需要是线程安全的取决于它是否被多个线程访问. 当多个线程访问同一个可变状态量时如果没有使用正确的同步规则,就有可能出错.解决办法: 不在线程之间共享该变量 将状态变量修改为不可变的 在访问 ...

  9. Java并发编程实战 第15章 原子变量和非阻塞同步机制

    非阻塞的同步机制 简单的说,那就是又要实现同步,又不使用锁. 与基于锁的方案相比,非阻塞算法的实现要麻烦的多,但是它的可伸缩性和活跃性上拥有巨大的优势. 实现非阻塞算法的常见方法就是使用volatil ...

随机推荐

  1. 在Silverlight中的DispatcherTimer的Tick中使用基于事件的异步请求

    需求:在silverlight用户界面上使用计时器定时刷新数据. 在 Silverlight 中的 DispatcherTimer 的 Tick 事件 中使用异步请求数据时,会出现多次请求的问题,以下 ...

  2. requirejs 小结

    1,“生搬硬套”-----js依赖的写法,依赖关系如这里的嵌套关系: /** * 模块化加载,注意各个js文件的依赖关系(嵌套) */ require(["jquery"], fu ...

  3. window国际化文案

    越来越多的程序支持多语言切换,或者能自动适应当前系统语言,让自己开发的程序支持多语言不仅可以让自己的程序被国人使用,也能让外国程序爱好者使用.VC开发多语言程序有多种方法,或读取配置文件,或使用不同资 ...

  4. 创业实战go语言制作网站(转)

    简单介绍一下之前的整个职业生涯,挺典型的,工程师 –> 资深工程师 –> 架构师 –> 项目经理 –> 部门经理,可以说,产品研发过程中的所有角色我都走了一遍,任职的公司也越来 ...

  5. Spring编程风格

    给自己使用的无需定义接口:即一个模块内部的都是封装的,定义接口并不会得到很多好处,变过几次实现?? “优先面向接口编程,而非实现” 不是必须,是优先: 给朋友(第三方)使用的定义接口:即要公开的功能, ...

  6. 逻辑回归损失函数(cost function)

    逻辑回归模型预估的是样本属于某个分类的概率,其损失函数(Cost Function)可以像线型回归那样,以均方差来表示:也可以用对数.概率等方法.损失函数本质上是衡量”模型预估值“到“实际值”的距离, ...

  7. 剑指Offer:互为变位词

    // 判断两个单词是否互为变位词: 如果两个单词中的字母相同,并且每个字母出现的次数也相同, 那么这两个单词互为变位词 #include <stdio.h> #include <st ...

  8. NET下RabbitMQ实践[WCF发布篇]

    在之前的两篇文章中,主要介绍了RabbitMQ环境配置,简单示例的编写.今天将会介绍如何使用WCF将RabbitMQ列队以服务的方式进行发布.          注:因为RabbitMQ的官方.net ...

  9. Java程序版权保护解决方案

    通常C.C++等编程语言开发的程序都被编译成目标代码,这些目标代码都是本机器的二进制可执行代码.通常所有的源文件被编译.链接成一个可执行文件.在这些可执行文件中,编译器删除了程序中的变量名称.方法名称 ...

  10. IE兼容性问题解决方案1--ajax请求不发送到后台

    相信很多小伙伴会遇到这种问题,用ajax做异步请求的时候,在IE浏览器下,并没有发送出去.但是相关程序确实执行了.为什么呢? 原来这是IE缓存方式的原因,所以呢,用下边的解决方案吧. 1.在请求的UR ...