zoukankan      html  css  js  c++  java
  • 一个java的Profile工具

    场景:我们在系统运行中,需要监控某个代码段的运行时间,我们完全可以使用currentTimeMillis来做,但是做起来比较麻烦,尤其是需要阶段监控的时候,那么这个工具就出现啦~~~

    先说下想要实现的功能:1.能够对代码段进行运行时间的监控,比如代码行a->代码行b的运行时间。2.能够监控代码行嵌套的运行时间监控,比如a->b->c->d中a->d和b->c的运行时间监控(类似括号一样,形成配对的方式)。3.能够在运行范围内的jvm一些指标的监控,比如内存使用量等。

    /**
     * 性能相关的调试工具,支持,线程正式场景,做运行时间的profile,运行性能监控(不建议线上使用,因为需要开启监控线程)
     * 该工具不会抛出任何异常
     * @author Administrator
     * @version $Id: ProfileUtils.java, v 0.1 2016年9月5日 下午11:02:45 Administrator Exp $
     */
    public class Profiler {
    
        /** debug模式 */
        //    private static volatile boolean                    debug               = false;
    
        private final static String                        LOG_TEMPLATE        = "[messag=%s][startTime=%s][endTime=%s][durationTime=%sms][processors=%s][memUse=%s]";
        private final static String                        SIMPLE_LOG_TEMPLATE = "[durationTime=%sms][message=%s]";
        private final static SimpleDateFormat              DATE_FORMAT         = new SimpleDateFormat(
            "yyyy/MM/dd HH:mm:ss");
    
        /** profile日志,建议运行中别做修改,否则有些配置会导致残留线程 */
        private static ThreadLocal<ProfileConfig>          configHolder        = new ThreadLocal<ProfileConfig>() {
                                                                                   protected ProfileConfig initialValue() {
                                                                                       return new ProfileConfig(
                                                                                           false, false,
                                                                                           0);
                                                                                   };
                                                                               };
    
        /** 开始monitor的时间 */
        private static ThreadLocal<Stack<MonitorResource>> resStackHolder      = new ThreadLocal<Stack<MonitorResource>>() {
                                                                                   protected java.util.Stack<MonitorResource> initialValue() {
                                                                                       return new Stack<MonitorResource>();
                                                                                   };
                                                                               };
    
        /** 监控线程 */
        private static ThreadLocal<MonitorThread>          monitorThreadHolder = new ThreadLocal<MonitorThread>();
    
        /**
         * 开始monitor
         */
        public static void enter(Object msgObj) {
            try {
                Stack<MonitorResource> monitorResStack = resStackHolder.get();
                monitorResStack.push(new MonitorResource(msgObj, System.currentTimeMillis()));
                ProfileConfig config = configHolder.get();
                //开启监控线程
                if (config.isUseMonitorThread()) {
                    if (monitorThreadHolder.get() != null) {
                        killThread();
                    }
                    MonitorThread monitorThread = new MonitorThread(getCurrentMonitorRes(), config);
                    monitorThreadHolder.set(monitorThread);
                    monitorThread.start();
                }
            } catch (Throwable e) {
                //            if (debug) {
                //                e.printStackTrace();
                //            }
                return;
            }
        }
    
        /**
         * 结束monitor
         * @return
         */
        public static MonitorResource release() {
            try {
                Stack<MonitorResource> monitorResStack = resStackHolder.get();
                MonitorResource monitorResource = getCurrentMonitorRes();
                monitorResource.setEndTime(System.currentTimeMillis());
                ProfileConfig config = configHolder.get();
                //监控线程关闭
                if (config.isUseMonitorThread()) {
                    killThread();
                }
                return monitorResStack.pop();
            } catch (Throwable e) {
                //            if (debug) {
                //                e.printStackTrace();
                //            }
                return new MonitorResource(e.getMessage(), 0);
            }
        }
    
        /**
         * 使用新的messageObj替换原来的
         * @param messageObj
         * @return
         */
        public static MonitorResource release(Object messageObj) {
            MonitorResource monitorResource = release();
            monitorResource.setMessageObj(messageObj);
            return monitorResource;
        }
    
        /**
         * 结束monitor并且打印日志
         * @param logger
         * @return
         */
        public static MonitorResource releaseAndLog(Logger logger, Object messageObj) {
            MonitorResource resource = release(messageObj);
            LoggerUtils.info(logger, resource);
            return resource;
        }
    
        /**
         * 结束monitor并且打印日志
         * @param logger
         * @return
         */
        public static MonitorResource releaseAndLog(Logger logger) {
            MonitorResource resource = release();
            LoggerUtils.info(logger, resource);
            return resource;
        }
    
        /**
         * 设置profile配置
         * @param config
         */
        public static void setProfileConfig(ProfileConfig config) {
            configHolder.set(config);
        }
    
        /**
         * Setter method for property <tt>debug</tt>.
         * 
         * @param debug value to be assigned to property debug
         */
        //    public static void setDebug(boolean debug) {
        //        Profiler.debug = debug;
        //    }
    
        /**
         * 移除监控线程
         */
        private static void killThread() {
            try {
                MonitorThread futureTask = monitorThreadHolder.get();
                monitorThreadHolder.remove();
                futureTask.interrupt();
            } catch (Throwable e) {
                // ignore
                //            if (debug) {
                //                e.printStackTrace();
                //            }
            }
        }
    
        /**
         * 获取当前的monitorRes
         * @return
         */
        public static MonitorResource getCurrentMonitorRes() {
            try {
                Stack<MonitorResource> resStack = resStackHolder.get();
                return resStack.get(resStack.size() - 1);
            } catch (Exception e) {
                //            if (debug) {
                //                e.printStackTrace();
                //            }
                return new MonitorResource(e.getMessage(), 0);
            }
        }
    
        /**
         * 资源使用情况,比如cpu最大使用量等。
         * @author Administrator
         * @version $Id: Profile.java, v 0.1 2016年9月5日 下午11:38:39 Administrator Exp $
         */
        public static class MonitorResource {
    
            /** 当前资源的标志 */
            private Object     messageObj    = null;
    
            private long       startTime     = 0;
    
            private long       endTime       = 0;
    
            private int        processorNums = 0;
    
            private List<Long> memUse        = Lists.newArrayList();
    
            /**
             * @param messageObj
             * @param startTime
             */
            public MonitorResource(Object messageObj, long startTime) {
                super();
                this.messageObj = messageObj;
                this.startTime = startTime;
            }
    
            /**
             * Setter method for property <tt>messageObj</tt>.
             * 
             * @param messageObj value to be assigned to property messageObj
             */
            public void setMessageObj(Object messageObj) {
                this.messageObj = messageObj;
            }
    
            public String getMemUse() {
                StringBuilder stringBuilder = new StringBuilder();
                for (int i = 0; i < memUse.size(); i++) {
                    stringBuilder.append(memUse.get(i) / 1024L + "K");
                    if (i != memUse.size() - 1) {
                        stringBuilder.append(",");
                    }
                }
                return stringBuilder.toString();
            }
    
            /**
             * 获取整个profile堆栈
             * @return
             */
            public Stack<MonitorResource> getMonitorResStack() {
                return resStackHolder.get();
            }
    
            /** 
             * @see java.lang.Object#toString()
             */
            @Override
            public String toString() {
                return configHolder.get().isUseSimpleLogTemplate()
                    ? (String.format(SIMPLE_LOG_TEMPLATE, endTime - startTime, messageObj))
                    : (String.format(LOG_TEMPLATE, messageObj, DATE_FORMAT.format(new Date(startTime)),
                        DATE_FORMAT.format(new Date(endTime)), endTime - startTime, processorNums,
                        getMemUse()));
            }
    
            /**
             * 获取运行时间
             * @return
             */
            public long getDurTime() {
                return endTime - startTime;
            }
    
            public void putMemUse(long l) {
                memUse.add(l);
            }
    
            /**
             * Setter method for property <tt>endTime</tt>.
             * 
             * @param endTime value to be assigned to property endTime
             */
            public void setEndTime(long endTime) {
                this.endTime = endTime;
            }
    
            /**
             * Getter method for property <tt>messageObj</tt>.
             * 
             * @return property value of messageObj
             */
            public Object getMessageObj() {
                return messageObj;
            }
    
            /**
             * Setter method for property <tt>processorNums</tt>.
             * 
             * @param processorNums value to be assigned to property processorNums
             */
            public void setProcessorNums(int processorNums) {
                this.processorNums = processorNums;
            }
    
        }
    
        public static class ProfileConfig {
            private boolean useSimpleLogTemplate  = false;
            private boolean useMonitorThread      = false;
            private int     monitorCollectDurTime = 500;
    
            /**
             * @param useSimpleLogTemplate
             * @param useMonitorThread
             * @param monitorCollectDurTime
             */
            public ProfileConfig(boolean useSimpleLogTemplate, boolean useMonitorThread,
                                 int monitorCollectDurTime) {
                super();
                this.useSimpleLogTemplate = useSimpleLogTemplate;
                this.useMonitorThread = useMonitorThread;
                this.monitorCollectDurTime = monitorCollectDurTime;
            }
    
            /**
             * Getter method for property <tt>useSimpleLogTemplate</tt>.
             * 
             * @return property value of useSimpleLogTemplate
             */
            public boolean isUseSimpleLogTemplate() {
                return useSimpleLogTemplate;
            }
    
            /**
             * Setter method for property <tt>useSimpleLogTemplate</tt>.
             * 
             * @param useSimpleLogTemplate value to be assigned to property useSimpleLogTemplate
             */
            public void setUseSimpleLogTemplate(boolean useSimpleLogTemplate) {
                this.useSimpleLogTemplate = useSimpleLogTemplate;
            }
    
            /**
             * Getter method for property <tt>useMonitorThread</tt>.
             * 
             * @return property value of useMonitorThread
             */
            public boolean isUseMonitorThread() {
                return useMonitorThread;
            }
    
            /**
             * Setter method for property <tt>useMonitorThread</tt>.
             * 
             * @param useMonitorThread value to be assigned to property useMonitorThread
             */
            public void setUseMonitorThread(boolean useMonitorThread) {
                this.useMonitorThread = useMonitorThread;
            }
    
            /**
             * Getter method for property <tt>monitorCollectDurTime</tt>.
             * 
             * @return property value of monitorCollectDurTime
             */
            public int getMonitorCollectDurTime() {
                return monitorCollectDurTime;
            }
    
            /**
             * Setter method for property <tt>monitorCollectDurTime</tt>.
             * 
             * @param monitorCollectDurTime value to be assigned to property monitorCollectDurTime
             */
            public void setMonitorCollectDurTime(int monitorCollectDurTime) {
                this.monitorCollectDurTime = monitorCollectDurTime;
            }
    
        }
    
        private static class MonitorThread extends Thread {
    
            private static final AtomicLong threadCount = new AtomicLong();
    
            private MonitorResource         monitorResource;
    
            private final ProfileConfig     config;
    
            /**
             * 
             */
            public MonitorThread(MonitorResource resource, ProfileConfig config) {
                monitorResource = resource;
                setName("monitor-thread-" + threadCount.getAndIncrement());
                setDaemon(true);
                this.config = config;
            }
    
            /** 
             * @see java.lang.Thread#run()
             */
            @Override
            public void run() {
                monitorResource.setProcessorNums(Runtime.getRuntime().availableProcessors());
                while (true) {
                    monitorResource.putMemUse(
                        Runtime.getRuntime().maxMemory() - Runtime.getRuntime().freeMemory());
    
                    try {
                        Thread.sleep(config.getMonitorCollectDurTime());
                    } catch (InterruptedException e) {
                        //                    if (debug) {
                        //                        e.printStackTrace();
                        //                    }
                        return;
                    }
                }
            }
        }
    }

     可以看到,我们有个监控资源的概念,每个阶段都对应一个监控资源,比如a->d和b->c都对应了一个监控资源。从实现上,每次进入监控的时候,会生成一个监控资源,并且记录当前时间,并且把该监控资源压栈。同时,还会启动一个监控线程,将jvm的状态不断写入该监控资源中,所以针对a->d和b->c都生成了自己的监控线程,并且吧jvm状态写入自己的监控资源中。  当release的时候,会kill掉监控线程,并且把监控资源出栈。

    根据上面的设计可以看到,该工具的使用需要注意下面的问题:

    1.如果不启用监控线程,那么可以用于线上的场景。

    2.如果启用了监控线程,那么只适合debug分析的场景,因为如果是线上,当监控的代码块并发量大起来的时候,会以1:1的比例创建监控线程,这个时候会有风险。

    3.启用监控线程的场景比较适合于,并发量较小(创建的监控线程少),但是执行时间长的场景。这个时候可以对代码块进行执行分析。

  • 相关阅读:
    CF225E Unsolvable
    CF1100E Andrew and Taxi
    oracle数据库导入导出方法
    ORACLE无法删除当前连接用户
     为什么上传文件的表单里要加个属性enctype----摘录
    ecplise 使用快捷键
    spring工作机制及为什么要用?
    阐述struts2的执行流程。
    Hibernate工作原理及为什么要用?
    Mybatis 如何自动生成bean dao xml 配置文件 generatorconfig.xml (main()方法自动生成更快捷)
  • 原文地址:https://www.cnblogs.com/color-my-life/p/5855103.html
Copyright © 2011-2022 走看看