在基于Spring框架的应用中,自己实现了基于commons-pool2的连接池,结果在Spring Boot的启动中,提示此类已注册,无法建立此MXBean,启动失败。
org.springframework.jmx.export.UnableToRegisterMBeanException: Unable to register MBean [mirana.cloud.vmware.ConnectionPool@27d10ff] with key 'connectionPool'; nested exception is javax.management.InstanceAlreadyExistsException: MXBean already registered with name org.apache.commons.pool2:type=GenericObjectPool,name=pool
……
Caused by: javax.management.InstanceAlreadyExistsException: MXBean already registered with name org.apache.commons.pool2:type=GenericObjectPool,name=pool
... 17 common frames omitted
查阅资料,MXBean必须注册在不同的名称,而Spring命名MXBean的方式为包名+类名,这样就出现了名称重复的现象。
解决办法一:取消MXBean的注册
对于自己定义的连接池,取消MXBean的注册,只需要激活GenericObjectPoolConfig既可以,方法如下:
<bean id="connectionPool" class="com.cnitsec.mirana.cloud.vmware.ConnectionPool">
<constructor-arg name="factory">
<bean class="mirana.cloud.vmware.ConnectionPoolFactory">
</bean>
</constructor-arg>
<constructor-arg name="config">
<bean class="org.apache.commons.pool2.impl.GenericObjectPoolConfig">
<!-- 取消JMX的激活 -->
<property name="jmxEnabled" value="false"/>
</bean>
</constructor-arg>
</bean>
解决办法二:更改继承方式为组合方式
继承的实现方法:
public class ConnectionPool extends GenericObjectPool<Connection> {
/**
* @param factory
* @param config
*/
public ConnectionPool(PooledObjectFactory<Connection> factory, GenericObjectPoolConfig config) {
super(factory, config);
}
public Connection borrow() {
try {
return super.borrowObject();
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
}
更改后的组合实现方法:
public class ConnectionPool{
private final GenericObjectPool<Connection> pool;
/**
* @param factory
* @param config
*/
public ConnectionPool(PooledObjectFactory<Connection> factory, GenericObjectPoolConfig config) {
pool = new GenericObjectPool<>(factory, config);
}
public Connection borrow() {
try {
return pool.borrowObject();
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
}
构造函数与接口引用都没有变,可以无缝结合,还可以观测MXBean的实时情况,如下:
解决办法三:更改Spring MXBean的命名策略
具体请参见“org.springframework.jmx.export.MBeanExporter”中的“setNamingStrategy”方法,具体实现方法略。
结论
灵活使用JAVA的基本方法与策略,加上追踪对方的源码文件,基本可以解决绝大多数的问题。