对于CPU而言,常见的瓶颈主要有两种:
服务器的压力很小,但是CPU的利用率却很高,这样的性能瓶颈相对比较容易定位(好比我只是说了你一句,你就哭了,你的弱点立马就暴露出来了);
给服务器施加的压力很大,但是CPU的利用率总是很低,也就是压力上不去,这类瓶颈最常见,定位起来也最困难(类似我对你用尽了各种手段,你就是不说实话,内心太强大了)。
影响压力上不去的因素很多,但是对于性能测试工程师而言,最具价值的瓶颈肯定是由于代码问题引起的了(定位到是代码问题导致的压力上不去,特有成就感),而导致这类问题出现的最常见的一个知识点,肯定就是锁了,如果对于Java多线程的锁机制不甚了解,分析定位代码的瓶颈和性能调优几乎是不可能的。
锁——确保多线程访问的数据一致性。
目前Java多线程的锁基本就是通过如下两种方式实现:
synchronized关键字
java.util.concurrent.locks包下的类(用的也蛮多,稍微有点复杂,课堂上讲吧)
今天就为大家分享synchronized关键字实现的加锁方式。
认识synchronized
假设这样一个场景,张三和李四都需要维修各自的电脑,但是只有一把螺丝刀工具,意味着同一时刻只能有一个人使用这把工具修他们的电脑,那么使用Java代码该如何实现呢?我们把张三和李四比喻成两个线程,把他们修电脑的过程用Java语言实现一下,开始!

上面代码执行后对线程做dump,摘取部分信息如下:

解读锁
wait()和sleep()
它们有一个共同点是,都会把当前的线程阻塞住( 时长为函数参数指定的时间), 我们称之为睡眠或者等待。但二者实际上是完全不同的两个函数, 二者有着最为本质的区别:
wait()  当线程执行到wait()方法上, 当前线程会释放监视锁, 此时其它线程可以占有该锁,一旦wait()方法执行完成, 当前线程又继续尝试获取该锁, 直到执行完该锁的作用域。 可以说wait()是多线程场合下用得最多的一个方法。 结合notify(),可以实现两个线程之间的通信,一个线程可以通过这种方法通知另一个线程继续执行, 完成线程之间的配合。 wait()和锁的示意图如下:

在wait(5000)这5秒( 5000毫秒) 期间, 当前线程会释放它占有的锁, 此时其它线程有机会获得该锁。 当wait(5000)执行完成后, 当前线程继续获得该锁的使用权。 满足如下条件之一, wait()方法退出:
达到了等待的时间之后, 自动退出。 如wait(5000),5秒后wait方法退出。
其它的线程调用了该锁的notify()方法。 如果多个线程在等待同一个锁, 只有一个线程会被通知到。
sleep()  与锁操作无关, 如果该方法恰好在一个锁的保护范围之内, 当前线程即使在执行sleep()的时候, 仍然继续保持监视锁。 该方法实际上仅仅是完成等待或者睡眠的语义。 示意图如下:

从上面的代码Thread.sleep(5000)可以看出, sleep()方法并不是锁上面的一个方法, 而是线程的一个静态方法。 也就是说该方法实际上是和锁操作无关的。 如果sleep()方法恰好在一个锁的保护范围之内, 那么当前线程即使执行到该sleep方法, 也不会产生特别的锁操作(持有锁或者释放锁), 如果原来持有, 现在仍然持有。 如果原来没有持有, 那么现在仍然不持有。
还记得第六篇文章解读的线程堆栈吗?包含的直接信息有:
线程的个数、 每个线程调用的方法堆栈、 当前锁的状态。 线程的个数可以直接数出来; 线程调用的方法堆栈, 从下向上看,即表示当前的线程调用了哪个类上的哪个方法。 而锁的状态看起来稍微有一点技巧。 与锁相关的三个重要信息如下:
当一个线程占有一个锁的时候, 线程堆栈中会打印—locked <0x24a23208>
当一个线程正在等待其它线程释放该锁, 线程堆栈中会打印—waiting to lock <0x24a23208>
当一个线程占有一个锁, 但又执行到该锁的wait()上, 线程堆栈中首先打印locked,然后又会打印—waiting on <0x24a23208>
源码:




运行上面的程序,打印堆栈信息如下:



从上面这个例子中, 可以很清晰地看出, 在线程堆栈中与锁相关的三个最重要的特征字:locked,waiting to lock,waiting on;了解这三个特征字, 就能够对锁进行分析了。
一般情况下, 当一个(些)线程在等待一个锁时, 应该有一个线程占用这个锁, 即如果有的线程在等待一个锁, 该锁必然被另一个线程占有了, 也就是说, 从打印的堆栈中如果能看到waiting to lock <0x04d02238>,应该也应该能找到一个线程locked <0x04d02238>。

Java项目性能瓶颈分析及定位(八)——Java线程堆栈分析(五)的更多相关文章

  1. Java问题定位之Java线程堆栈分析

    采用Java开发的大型应用系统越来越大,越来越复杂,很多系统集成在一起,整个系统看起来像个黑盒子.系统运行遭遇问题(系统停止响应,运行越来越慢,或者性能低下,甚至系统宕掉),如何速度命中问题的根本原因 ...

  2. Java线程堆栈分析

    不知觉间工作已有一年了,闲下来的时候总会思考下,作为一名Java程序员,不能一直停留在开发业务使用框架上面.老话说得好,机会是留给有准备的人的,因此,开始计划看一些Java底层一点的东西,尝试开始在学 ...

  3. 【转】java线上程序排错经验2 - 线程堆栈分析

    前言 在线上的程序中,我们可能经常会碰到程序卡死或者执行很慢的情况,这时候我们希望知道是代码哪里的问题,我们或许迫切希望得到代码运行到哪里了,是哪一步很慢,是否是进入了死循环,或者是否哪一段代码有问题 ...

  4. Java项目性能瓶颈定位

    文章目标 当Java项目出现性能瓶颈的时候,通常先是对资源消耗做分析,包括CPU,文件IO,网络IO,内存:之后再结合相应工具查找消耗主体的程序代码.本文主要介绍系统资源消耗的分析过程,以及常用的Ja ...

  5. jenkins构建java项目找不到命令mvn,java的解决方法

    jenkins构建java项目时出现的报错情况: $ mvn clean install FATAL: command execution failed java.io.IOException: er ...

  6. GitHub Java项目推荐|功能丰富的 Java 工具包|提高开发效率

    GitHub Java项目推荐|功能丰富的 Java 工具包|提高开发效率 功能丰富的 Java 工具包.它帮助我们实现了常用的工具方法,从而减少代码的体积,提高开发效率.该项目最初是作者工作项目中的 ...

  7. Github优秀java项目集合(中文版) - 涉及java所有的知识体系

    Java资源大全中文版 我想很多程序员应该记得 GitHub 上有一个 Awesome - XXX 系列的资源整理.awesome-java 就是 akullpp 发起维护的 Java 资源列表,内容 ...

  8. JAVA项目之苹果IAP内购JAVA服务器验证流程详解

    1.前言 本博客是经历过多个项目检验的, 绝对真实, 适应于对苹果iap内购稍微有些了解的JAVA开发人员,  认真看,  定能完美解决苹果内购问题. 苹果IAP内购支付实际上是"将客户端支 ...

  9. Java应用性能瓶颈分析思路

    1 问题描述 因产品架构的复杂性,可能会导致性能问题的因素有很多.根据部署架构,大致的可以分为应用端瓶颈.数据库端瓶颈.环境瓶颈三大类.可以根据瓶颈的不同部位,选择相应的跟踪工具进行跟踪分析. 应用层 ...

随机推荐

  1. Qt 学习之路 2(3):Hello, world!

     豆子  2012年8月22日  Qt 学习之路 2  107条评论 想要学习 Qt 开发,首先要搭建 Qt 开发环境.好在现在搭建 Qt 开发环境还是比较简单的.我们可以到 Qt 官方网站找到最新版 ...

  2. Python:raw_input 和 input用法

    转自:http://blog.csdn.net/kjing/article/details/7450146 Python input和raw_input的区别 使用input和raw_input都可以 ...

  3. java Swing包相关知识点

    1.窗体的创建及相关的常用设置 //创建一个窗体 JFrame jf=new JFrame("第一步句法分析"); //设置用户在此窗体上发起 "close" ...

  4. spring boot中 启用aspectj

    <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring- ...

  5. python中的设计模式

    单例模式:Python 的单例模式最好不要借助类(在 Java 中借助类是因为 Java 所有代码都要写在类中),而是通过一个模块来实现.一个模块的模块内全局变量.模块内全局函数,组合起来就是一个单例 ...

  6. nginx —— 理解nginx_upstream_jvm_route模块解决tomcat多节点session不一致问题

    这种方式不需要修改web工程只需要对nginx下载nginx_upstream_jvm_route插件,修改tomcat和nginx配置,就能解决session问题.由于这种方式不会把session存 ...

  7. 在SQL Server中创建用户角色及授权(使用SQL语句)

    1. 首先在 SQL Server 服务器级别,创建登陆帐户(create login) --创建登陆帐户(create login) create login dba with password=' ...

  8. inventor安装失败怎样卸载安装inventor 2017?

    AUTODESK系列软件着实令人头疼,安装失败之后不能完全卸载!!!(比如maya,cad,3dsmax等).有时手动删除注册表重装之后还是会出现各种问题,每个版本的C++Runtime和.NET f ...

  9. [转]显示农历日期的JS

    本文转自:http://blog.sina.com.cn/s/blog_47377e77010009xc.html <!DOCTYPE HTML PUBLIC "-//W3C//DTD ...

  10. Starting cloudera-scm-server: * Couldn't start cloudera-scm-server的解决办法(图文详解)

    bigdata@ubuntucmbigdata1:~$ sudo /etc/init.d/mysql start start: Job is already running: mysql bigdat ...