001 //$HeadURL: svn+ssh://rbezema@svn.wald.intevation.org/deegree/base/branches/2.2_testing/src/org/deegree/framework/util/ProfilerInterceptor.java $ 002 // $Id: ProfilerInterceptor.java 9339 2007-12-27 12:31:52Z apoth $ 003 /*---------------- FILE HEADER ------------------------------------------ 004 005 This file is part of deegree. 006 Copyright (C) 2001-2008 by: 007 EXSE, Department of Geography, University of Bonn 008 http://www.giub.uni-bonn.de/deegree/ 009 lat/lon GmbH 010 http://www.lat-lon.de 011 012 This library is free software; you can redistribute it and/or 013 modify it under the terms of the GNU Lesser General Public 014 License as published by the Free Software Foundation; either 015 version 2.1 of the License, or (at your option) any later version. 016 017 This library is distributed in the hope that it will be useful, 018 but WITHOUT ANY WARRANTY; without even the implied warranty of 019 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 020 Lesser General Public License for more details. 021 022 You should have received a copy of the GNU Lesser General Public 023 License along with this library; if not, write to the Free Software 024 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 025 026 Contact: 027 028 Andreas Poth 029 lat/lon GmbH 030 Aennchenstr. 19 031 53115 Bonn 032 Germany 033 E-Mail: poth@lat-lon.de 034 035 Prof. Dr. Klaus Greve 036 Department of Geography 037 University of Bonn 038 Meckenheimer Allee 166 039 53115 Bonn 040 Germany 041 E-Mail: greve@giub.uni-bonn.de 042 043 044 ---------------------------------------------------------------------------*/ 045 046 package org.deegree.framework.util; 047 048 import java.lang.reflect.InvocationTargetException; 049 import java.lang.reflect.Method; 050 import java.text.DecimalFormat; 051 import java.text.NumberFormat; 052 import java.util.HashMap; 053 import java.util.Iterator; 054 055 /** 056 * Interceptor to profile the application. 057 * 058 * @author <a href="mailto:tfr@users.sourceforge.net">Torsten Friebe </A> 059 * 060 * @author last edited by: $Author: apoth $ 061 * 062 * @version 2.0, $Revision: 9339 $, $Date: 2007-12-27 13:31:52 +0100 (Do, 27 Dez 2007) $ 063 * 064 * @see <a href="http://www.dofactory.com/patterns/PatternChain.aspx">Chain of Responsibility Design 065 * Pattern </a> 066 * 067 * @since 2.0 068 */ 069 public class ProfilerInterceptor extends Interceptor { 070 071 static { 072 Runtime.getRuntime().addShutdownHook( new Thread( new Runnable() { 073 074 public void run() { 075 printStats(); 076 } 077 } ) ); 078 } 079 080 protected static HashMap<String, ProfileEntry> data = new HashMap<String, ProfileEntry>(); 081 082 /** 083 * 084 * @param nextInterceptor 085 */ 086 public ProfilerInterceptor( Interceptor nextInterceptor ) { 087 this.nextInterceptor = nextInterceptor; 088 } 089 090 /** 091 * 092 * 093 */ 094 protected static void printStats() { 095 NumberFormat zweiDezimalen = new DecimalFormat( "### ###.##" ); 096 097 Iterator iter = data.keySet().iterator(); 098 while ( iter.hasNext() ) { 099 String m = (String) iter.next(); 100 ProfileEntry entry = data.get( m ); 101 LOG.logInfo( entry.invocations + " calls " + entry.time + " ms " + " avg " 102 + zweiDezimalen.format( entry.getAverage() ) + m ); 103 } 104 } 105 106 /** 107 * 108 * @param m 109 * @return named ProfileEntry 110 */ 111 protected static ProfileEntry getEntry( String m ) { 112 if ( data.containsKey( m ) ) { 113 return data.get( m ); 114 } 115 ProfileEntry entry = new ProfileEntry(); 116 data.put( m, entry ); 117 return entry; 118 } 119 120 /** 121 * @param method 122 * @param params 123 * @return - 124 */ 125 protected Object handleInvocation( Method method, Object[] params ) 126 throws IllegalAccessException, InvocationTargetException { 127 long start = System.currentTimeMillis(); 128 Object result = nextInterceptor.handleInvocation( method, params ); 129 long duration = System.currentTimeMillis() - start; 130 ProfileEntry entry = getEntry( getTarget().getClass().getName() + "." + method.getName() ); 131 entry.time += duration; 132 entry.invocations++; 133 return result; 134 } 135 136 static class ProfileEntry { 137 138 long time; 139 140 long invocations; 141 142 public double getAverage() { 143 return (double) time / (double) invocations; 144 } 145 146 } 147 }