赞
踩
工具: vmstat
检查应用性能时,应该首先审查CPU时间。
代码优化的目的是提升而不是降低(更短时间段内的)CPU的使用率。
在试图深入优化应用前,应该先弄清楚为何CPU使用率低。
工具: iostat
对于所有应用来说,监控磁盘使用率非常重要。即便不直接写磁盘的应用,系统交换仍会影响它们的性能。
写入磁盘的应用遇到瓶颈,是因为写入数据的效率不高(吞吐率太低),或者是因为写入太多数据(吞吐率太高)。
工具: netstat nicstat
对于基于网络的应用来说,务必要监控网络以确保它不是瓶颈。
往网络写数据的应用遇到瓶颈,可能是因为写数据的效率太低(吞吐率太低),或者是因为写入了太多数据(吞吐率太高)。
jcmd: 打印java进程涉及的基本类,线程和VM信息。
jconsole: 提供JVM活动的图形化视图,包括线程的使用,类的使用和GC活动。
jhat: 读取内存堆转储,并有助于分析。
jmap: 提供堆转储和其他JVM内存使用的信息。
jinfo: 查看JVM的系统属性,可以动态设置的一些系统属性。
jstack: 转储Java进程的栈信息。
jstat: 提供GC和类装载活动的信息。
jvisualvm: 监视JVM的GUI工具,可以用来剖析运行的应用,分析JVM堆转储。
Visual VM是一个功能强大的多合一故障诊断和性能监控的可视化工具,它集成了多种性能统计工具的功能,使用Visual VM可以替代jstat、jmap、jhat、jstack等工具。在命令行输入jvisualvm即可启动visualvm。
打开Visual VM之后,左边导航栏会显示出当前机器所有Java进程:
点击你想监控的程序即可对该程序进行监控,Visual VM的性能监控页一共有以下几个tab页:
概述页会显示程序的基本使用情况,比如,进程ID,系统属性,启动参数等。
通过监视页面,可以监视应用程序的CPU、堆、永久区、类加载器和线程数的整体情况,通过页面上的Perform GC和Heap Dump按钮还可以手动执行Full GC和生成堆快照。
线程页面会提供详细的线程信息,单击Thread Dump按钮可以导出当前所有线程的堆栈信息,如果Visual VM在当前线程中找到死锁,则会以十分显眼的方式在Threads页面给予提示。
抽样器可以对CPU和内存两个性能进行抽样,用于实时地监控程序。CPU采样器可以将CPU占用时间定位到方法,内存采样器可以查看当前程序的堆信息。下面是一个频繁调用的Java程序,我们会对改程序进行采样:
public class MethodTime { static java.util.Random r=new java.util.Random(); static Map<String,String> map=null; static{ map=new HashMap<String,String>(); map.put("1", "Java"); map.put("2", "C++"); map.put("3", "Delphi"); map.put("4", "C"); map.put("5", "Phython"); } public String getNameById(String id){ try { Thread.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } return map.get(id); } public List<String> getNamesByIds(String ids){ List<String> re=new ArrayList<String>(); String[] strs=ids.split(","); for(String id:strs){ re.add(getNameById(id)); } return re; } public List<String> getNamesByIdsBad(String ids){ List<String> re=new ArrayList<String>(); String[] strs=ids.split(","); for(String id:strs){ //A bad code getNameById(id); re.add(getNameById(id)); } return re; } public class NamesByIdsThread implements Runnable{ @Override public void run() { try{ while(true){ int c=r.nextInt(4); String ids=""; for(int i=0;i<c;i++) ids=Integer.toString((r.nextInt(4)+1))+","; getNamesByIds(ids); } }catch(Exception e){ } } } public class NamesByIdsBadThread implements Runnable{ @Override public void run() { try{ while(true){ int c=r.nextInt(4); String ids=""; for(int i=0;i<c;i++) ids=Integer.toString((r.nextInt(4)+1))+","; getNamesByIdsBad(ids); } }catch(Exception e){ } } } public static void main(String args[]){ MethodTime instance=new MethodTime(); new Thread(instance.new NamesByIdsThread()).start(); new Thread(instance.new NamesByIdsBadThread()).start(); } }
通过Visual VM的采样功能,可以找到改程序中占用CPU时间最长的方法:
默认Visual VM不统计内置对象的函数调用,比如java.*包中的类,如果要统计这些内置对象,单机右上角的设置进行调配。Visual VM虽然可以统计方法的调用时间,但是无法给出方法调用堆栈,Jprofile不仅可以给出方法调用时间,还可以给出方法调用堆栈,较Visual VM更强大。
右击左导航的应用程序,会出现以下菜单:
单机应用程序快照,可以分析当前应用程序的快照,单击堆Dump能够对当前的堆信息进行分析。Visual VM的更多使用方法,可以查看Oracle的官方文档文档链接
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。