3

我编写了一个部署在 Tomcat 6 中的自定义 MBean。它的任务之一是查询数据库值。我通过使用 JNDI 加载数据库资源来做到这一点——该资源在 Tomcat 的 server.xml 中定义。

问题是,当我创建它的实例时,javax.naming.InitialContext它会抛出 a ClassNotFoundExceptionas it can't find org.apache.naming.java.javaURLContextFactory。此类catalina.jar在公共类加载器中并由其加载。包含我的 MBean 代码的 jar 由共享类加载器加载。

关于如何解决这个问题的任何想法?

注意:我的 MBean 是由我在tomcat/conf/web.xml. 我还在一个 webapp 中定义了它,web.xml这没什么区别。我不能真正移动我的 jar 以便由公共类加载器加载,因为它依赖于共享类加载器加载的类。

提前致谢,

将要

4

1 回答 1

6

它看起来是一个奇怪的类加载或安全/权限问题。以下是一种解决方法。

主要思想:由于ServletContextListener可以在new InitialContext()没有ClassNotFoundException在侦听器中获取它的情况下调用,并在注册 MBean 之前将其传递给 MBean 对象的构造函数。我使用了一个简单的 Web 应用程序,我没有修改tomcat/conf/web.xml.

中的资源配置tomcat/conf/context.xml

<Context>
...
    <Resource name="jdbc/TestDB" auth="Container" type="javax.sql.DataSource"
        maxActive="100" maxIdle="30" maxWait="10000"
        username="root" password="..." driverClassName="com.mysql.jdbc.Driver"
        url="jdbc:mysql://localhost:3306/javatest?autoReconnect=true"/>
...
<Context>

web.xml资源配置:

<resource-ref>
    <description>DB Connection</description>
    <res-ref-name>jdbc/TestDB</res-ref-name>
    <res-type>javax.sql.DataSource</res-type>
    <res-auth>Container</res-auth>
</resource-ref>

其中ServletContextListener注册了 MBean:

import java.lang.management.ManagementFactory;

import javax.management.MBeanServer;
import javax.management.ObjectName;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;

@WebListener
public class ContextListener implements ServletContextListener {

    private ObjectName objectName;

    public void contextInitialized(final ServletContextEvent sce) {
        System.out.println("---> bean context listener started");

        final MBeanServer mbeanServer = 
            ManagementFactory.getPlatformMBeanServer();
        try {
            final InitialContext initialContext = new InitialContext();
            final Context envContext = 
                (Context) initialContext.lookup("java:/comp/env");

            objectName = new ObjectName("com.example:type=Hello");
            final Hello helloMbean = new Hello(envContext);
            mbeanServer.registerMBean(helloMbean, objectName);
            System.out.println("---> registerMBean ok");
        } catch (final Exception e) {
            e.printStackTrace();
        }
    }

    public void contextDestroyed(final ServletContextEvent sce) {
        System.out.println("---> bean context listener destroyed");
        final MBeanServer mbeanServer = 
            ManagementFactory.getPlatformMBeanServer();
        try {
            mbeanServer.unregisterMBean(objectName);
            System.out.println("---> unregisterMBean ok");
        } catch (final Exception e) {
            e.printStackTrace();
        }
    }
}

MBean接口:

public interface HelloMBean {
    void sayHello();
}

MBean 实现:

import java.sql.Connection;

import javax.naming.Context;
import javax.sql.DataSource;

public class Hello implements HelloMBean {

    private final Context envContext;

    public Hello(final Context envContext) {
        this.envContext = envContext;
        System.out.println("new hello " + envContext);
    }

    @Override
    public void sayHello() {
        System.out.println("sayHello()");

        try {
            final DataSource ds = 
                (DataSource) envContext.lookup("jdbc/TestDB");
            final Connection conn = ds.getConnection();
            System.out.println("   conn: " + conn);
            // more JDBC code
        } catch (final Exception e) {
            e.printStackTrace();
        }
    }
}

MBean Descriptor How To说 a是必需的mbeans-descriptor.xml,但没有它也能很好地工作。我可以使用jconsole. 调用sayHello()通过jconsole打印以下内容:

conn: jdbc:mysql://localhost:3306/javatest?autoReconnect=true, \
UserName=root@localhost, MySQL-AB JDBC Driver

资料来源:

于 2011-10-02T23:12:04.947 回答