今天突发奇想的对j2ee项目的性能监控来了兴趣,索性找找资料研究一下!
可能很多朋友在性能优化方面都有很多困扰:程序在哪方面存在缺陷?sql哪里有瓶颈?哪个请求最频繁?哪个请求耗时最长……从某种意义上说,解决问题并不是特别困难的事儿,但如何发现问题的问题会经常困扰着我。
今天给大家介绍一款j2ee项目性能监测工具:jwebap!通过简单的研究,发现这东西还是挺好用的,哈哈!
一、jwebap简介 参见
leadyu的博客
Jwebap是一个用于J2EE工程(EJB以及WebModule系统)进行性能监控的组件,它有几个特点:
1)基于ASM实现类的静态增强,可以无缝的部署于J2EE系统,对系统的开销几乎可以忽略
2) 部署和使用非常的简单,整个Jwebap的部署只需要部署jwebap_core_**.jar以及需要使用的各种plugin_**.jar,然后配置 jwebap.xml和web.xml就可以完成所有的部署,比起绝大多数的profiling容易的多。 同时Jwebap提供Web Console进行整个Jwebap的管理和数据展现。在API层提供一套默认的视图框架供plugin开发者使用,可以只用Jar包就开发出相当漂亮的 Web界面。
3)Jwebap的开发分为两个部分Jwebap-core部分,Jwebap-plugin部分。core部分基于jdk14 提供了类静态增强,轨迹生命管理,Plugin管理,视图框架等等,在这个基础上开发plugin。我觉得,好的profiling应该能够根据不同的人 群按需使用,同时在功能不断复杂和强大的过程中仍然能够保证较轻的架子。
理论部分原作者已经介绍的很详细了,这里就不再赘述。直接上干货!
二、jwebap的部署
1、将jwebap_0.6.1.jar、tracer_0.6.1.jar以及依赖包commons-beanutils.jar、commons-betwixt-0.8.jar、commons-digester-1.7.jar、commons-logging.jar、commontemplate-0.8.1.jar、jxl.jar导入到项目的classpath下。(一般webapp项目放在web-inf下的lib中即可)。
2、将jwebap.xml导入到classpath下。
jwebap.xml
<?xml version="1.0" encoding="UTF-8"?>
<jwebap>
<!--plugins-->
<plugin name="Tracer" ref="${ABSOLUTE_PATH}/tracer_0.6.1.jar"/>
<!--dispatcher-->
<dispatcher name="RedirectDispatcher" mapping="" type="org.jwebap.ui.controler.DefaultRedirectDispatcher" />
<dispatcher name="ActionDispatcher" mapping="/console/*" type="org.jwebap.ui.controler.ActionDispatcher" />
<dispatcher name="ResourceDispatcher" mapping="/resources/*" type="org.jwebap.ui.controler.ResourceDispatcher" />
<!--action-mapping-->
<action-mapping>
<action path="/" type="org.jwebap.ui.action.EmptyAction" template="resources/view/index.ctl" />
<action path="/deploy/plugins" type="org.jwebap.ui.action.EmptyAction" template="resources/view/plugin_deploy.ctl" />
<action path="/deploy/plugins/list" type="org.jwebap.ui.action.PluginListAction" />
<action path="/deploy/plugins/remove" type="org.jwebap.ui.action.PluginRemoveAction" />
<action path="/deploy/plugins/new" type="org.jwebap.ui.action.PluginFormAction" template="resources/view/plugin_deploy_new.ctl" />
<action path="/deploy/plugins/add" type="org.jwebap.ui.action.PluginAddAction" />
<action path="/deploy/plugins/detail" type="org.jwebap.ui.action.EmptyAction" template="resources/view/plugin_detail.ctl" />
<action path="/deploy/plugins/components/list" type="org.jwebap.ui.action.ComponentListAction" />
<action path="/deploy/plugins/components/detail" type="org.jwebap.ui.action.ComponentFormAction" template="resources/view/component_detail.ctl" />
<action path="/deploy/plugins/components/save" type="org.jwebap.ui.action.ComponentSaveAction" />
</action-mapping>
</jwebap>
3、修改项目下的web.xml,添加jwebap相关的配置
<context-param>
<param-name>jwebap-config</param-name>
<param-value>/WEB-INF/jwebap.xml</param-value>
</context-param>
<listener>
<listener-class>org.jwebap.startup.JwebapListener</listener-class>
</listener>
<filter>
<filter-name>PageDetectFilter</filter-name>
<filter-class>org.jwebap.plugin.tracer.http.DetectFilter</filter-class>
<init-param>
<param-name>excludeUrls</param-name>
<param-value>/detect;/detect/*;*.js;*.jpg;*.htm;*.html;*.gif;*.png;*.css;*.swf</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>PageDetectFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<servlet>
<servlet-name>detect</servlet-name>
<servlet-class>org.jwebap.ui.controler.JwebapServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>detect</servlet-name>
<url-pattern>/detect/*</url-pattern>
</servlet-mapping>
4、修改tracer_0.6.1.jar中的plugin.xml文件(在jar包的META-INF文件夹下)
……
<component name="MethodComponent" type="org.jwebap.plugin.tracer.method.MethodComponent">
<component-param>
<name>trace-filter-active-time</name>
<value>-1</value>
<description>(ms) timings filter's over time</description>
</component-param>
<component-param>
<name>trace-max-size</name>
<value>1000</value>
<description>max over-time trace size</description>
</component-param>
<component-param style="longtext">
<name>detect-clazzs</name>
<value>
com.gamecenter.controller.report.*;<!-- 改为需要监控的包或者类 -->
</value>
<description>
package name and class name that monitored by MethodComponent,
e.g.: 'test.*;test.Test' , divided by ';'
</description>
</component-param>
</component>
<component name="JdbcComponent" type="org.jwebap.plugin.tracer.jdbc.JdbcComponent">
<component-param>
<name>trace-filter-active-time</name>
<value>-1</value>
<description>(ms) timings filter's over time</description>
</component-param>
<component-param>
<name>trace-max-size</name>
<value>1000</value>
<description>max over-time trace size</description>
</component-param>
<component-param style="longtext">
<name>connection-listener</name>
<value>
org.jwebap.plugin.tracer.http.ServletOpenedConnectionListener;org.jwebap.plugin.tracer.method.MethodOpenedConnectionListener
</value>
<description>Connection Listener</description>
</component-param>
<component-param style="longtext">
<name>driver-clazzs</name>
<value>com.jolbox.bonecp.BoneCPDataSource</value><!-- 配置数据库连接池驱动,我这里用的是bonecp连接池 -->
<description>
1)Local datasource: set your ConnectionManagerClass,or the connection pool 's datasource. If you have more than one class ,divided by ';'.
c3p0:com.mchange.v2.c3p0.ComboPooledDataSource;
dbcp: org.apache.commons.dbcp.BasicDataSource
Also,other class. Jwebap will inject driver-clazzs,and detect any connection and datasource object it's method renturn.
Note: 'driver-clazzs =jdbc driver' is deprecated. Beacause of connection pool, set 'driver-clazzs =jdbc driver', jwebap will find out all connection is leaked.
2)JNDI datasource: If your application uses jndi datasource, you can set the class which manages connections in your application as driver, e.g.: 'com.china.telecom.ConnectionManager'.
Else if you use spring to get jndi datasource ,
you also can set driver-clazzs=org.springframework.jndi.JndiObjectFactoryBean.JdbcComponent
will inject this class to proxy all connection the class's method return.
org.springframework.orm.ibatis.SqlMapClientFactoryBean
</description>
</component-param>
</component>
至此,jwebap的配置就完成了!让我们赶快启动应用来测试一下吧!
应用正常启动后,在浏览器中输入 http://ip:port/servername/detect/console/ 你会看到jwebap的性能检测界面,大功告成!
三、遇到的问题:
1、JdbcComponent 中driver-clazzs的设置。这里直接设置成数据库连接池就好了。但是设置完成以后可能会出现以下错误:
org.jwebap.toolkit.bytecode.InjectException: com.jolbox.bonecp.BoneCPConfig注入失败.
……………………
Caused by: org.jwebap.toolkit.bytecode.asm.DefineBytecodeException: com/jolbox/bonecp/BoneCPConfig注入错误:loader (instance of org/apache/catalina/loader/WebappClassLoader): attempted duplicate class definition for name: "com/jolbox/bonecp/BoneCPConfig"
这个问题之前我也尝试了很多方法,最后才发现这个重复加载是由于spring和jwebap都加载了boncp类造成的,而我这里采取的解决办法就是将
<listener>
<listener-class>
org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>
放在
<listener>
<listener-class>
org.jwebap.startup.JwebapListener
</listener-class>
</listener>
的后面即可。至于为什么,我现在还不是很清楚,欢迎哪位大师指点一二!
2、如果JdbcComponent 中driver-clazzs的设置没有报错,但是控制台下的jdbc Trances就是没有监测到数据,那么你的driver-clazzs可能没有和你项目中所使用的数据库连接池(或驱动)匹配上,再换个数据源试试吧!