Ошибка с ошибкой NoClassDefFoundError в веб-среде Spring / Wicket / Derby / Jetty - PullRequest
2 голосов
/ 08 апреля 2011

Я пытаюсь создать простое приложение JDBC Spring Template, веб-фреймворк, который я использую, - wicket и под веб-сервером jetty 6 (через плагин Jetty Maven). Также я создаю приложение с помощью Eclipse.

По какой-то причине я получаю NoClassDefFoundError с классом Derby jdbc. Я предполагаю, что получу класс не найден, исключение не найдено, поэтому я предполагаю, что происходит что-то еще. Класс derby является частью classpath, каталога WEB-INF / lib. Как вы думаете, в чем проблема?

Мои мысли о проблеме: Это не ошибка «jar not found in the classpath», а скорее проблема с Java или подпружиненной динамической загрузкой этого класса и при его загрузке.

Я использую Eclipse в качестве инструмента разработки, но, вероятно, это не является частью проблемы. Я все еще использую Maven в командной строке и получаю ту же проблему.

Вот ошибка:

WicketMessage: Не удается создать экземпляр страницы с помощью конструктора public wicketspring.easy.HomePage ()

Основная причина:

java.lang.NoClassDefFoundError: Could not initialize class org.apache.derby.jdbc.EmbeddedDriver
     at java.lang.Class.forName0(Native Method)
     at java.lang.Class.forName(Class.java:169)
     at org.apache.commons.dbcp.BasicDataSource.createDataSource(BasicDataSource.java:1130)
     at org.apache.commons.dbcp.BasicDataSource.getConnection(BasicDataSource.java:880)
     at org.springframework.jdbc.datasource.DataSourceUtils.doGetConnection(DataSourceUtils.java:113)
     at org.springframework.jdbc.datasource.DataSourceUtils.getConnection(DataSourceUtils.java:79)
     at org.springframework.jdbc.core.JdbcTemplate.execute(JdbcTemplate.java:577)
     at org.springframework.jdbc.core.JdbcTemplate.execute(JdbcTemplate.java:619)
     at wicketspring.easy.jdbc.JdbcWicketSpringHandler.data(JdbcWicketSpringHandler.java:39)
     at WICKET_wicketspring.easy.jdbc.JdbcWicketSpringHandler$$FastClassByCGLIB$$f1187cb6.invoke(<generated>)
     at net.sf.cglib.proxy.MethodProxy.invoke(MethodProxy.java:191)
     at org.apache.wicket.proxy.LazyInitProxyFactory$CGLibInterceptor.intercept(LazyInitProxyFactory.java:319)
     at WICKET_wicketspring.easy.jdbc.JdbcWicketSpringHandler$$EnhancerByCGLIB$$e8f0e174.data(<generated>)
     at wicketspring.easy.HomePage.<init>(HomePage.java:91)
     at wicketspring.easy.HomePage.<init>(HomePage.java:47)

Вот приложениеContext.xml для Spring:

<beans>
    <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
        <property name="driverClassName" value="org.apache.derby.jdbc.EmbeddedDriver" />
        <property name="url"><value>jdbc:derby:wicketspringdb</value></property>
        <property name="username"><value></value></property>
        <property name="password"><value></value></property>
    </bean> 
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource" />
    </bean> 
</beans>
<beans> 
    <import resource="classpath:common.xml"/>
    <bean id="jdbcHandler" class=wicketspring.easy.jdbc.JdbcWicketSpringHandler">
        <property name="jdbcTemplate" ref="jdbcTemplate" />
    </bean>

</beans>


...
Another stack trace system out.

Page.java:74) - At [2b] -- java.lang.NoClassDefFoundError: Could not initialize class org.apache.derby.jdbc.EmbeddedDriver
java.lang.NoClassDefFoundError: Could not initialize class org.apache.derby.jdbc.EmbeddedDriver
        at wicketspring.easy.HomePage.<init>(HomePage.java:72)
        at wicketspring.easy.HomePage.<init>(HomePage.java:47)
        at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
        at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:
39)
        at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorIm
pl.java:27)

...

Вот код Java, код компилируется, и я могу распечатать класс, но не могу его создать. Странно?

Java-код:

package wicketspring.easy;

import java.util.List;

import org.apache.wicket.Application;
import org.apache.wicket.PageParameters;
import org.apache.wicket.markup.html.WebPage;
import org.apache.wicket.markup.html.basic.Label;
import org.apache.wicket.markup.html.list.ListItem;
import org.apache.wicket.markup.html.list.ListView;
import org.apache.wicket.model.CompoundPropertyModel;
import org.apache.wicket.proxy.IProxyTargetLocator;
import org.apache.wicket.proxy.LazyInitProxyFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import model.LoadableDetachableList;
import model.SelectionOptionBean;
import wicketspring.easy.jdbc.JdbcWicketSpringHandler;
import wicketspring.easy.service.IStateListService;
import wicketspring.easy.service.StateListServiceImpl;

import org.apache.derby.jdbc.EmbeddedDriver;

public class HomePage extends WebPage {

    private final JdbcWicketSpringHandler jdbcWicketSpringHandler;

    public HomePage(final PageParameters parameters) {

        super(parameters);        
        jdbcWicketSpringHandler = (JdbcWicketSpringHandler) LazyInitProxyFactory.createProxy(JdbcWicketSpringHandler.class,
                new IProxyTargetLocator() {
                    private static final long serialVersionUID = 1L;
                    public Object locateProxyTarget() {
                        return ((WicketApplication) Application.get()).context().getBean("jdbcHandler");
                    }
                });    

        /// WEIRD BECAUSE IT REACHES THIS POINT!!!
        LOGGER.debug("At [1] -- " + EmbeddedDriver.class);
        try {
            LOGGER.debug("At [2] -- " + new org.apache.derby.jdbc.EmbeddedDriver());
        } catch (NoClassDefFoundError ne) {
                // STACK TRACE ERROR HERE!!!!
            LOGGER.debug("At [2b] -- " + ne);
            ne.printStackTrace();
        } catch (Exception e) {
            LOGGER.debug("At [2] -- " + e);
            e.printStackTrace();
        }
        try {

            LOGGER.debug("At -- " + Class.forName("org.apache.derby.jdbc.EmbeddedDriver"));     
        } catch (NoClassDefFoundError ne) {
            LOGGER.debug("At [3b] -- " + ne);
            ne.printStackTrace();
        } catch (Exception e) {
            LOGGER.debug("At [3] -- " + e);
            e.printStackTrace();
        }

        /// ERROR FOR STACKOVERFLOW IS GENERATED FROM THIS LINE!!!!
        LOGGER.debug("At HomePage - jdbcHandler [3]: " + jdbcWicketSpringHandler.toStringJdbcTemplate());
        LOGGER.debug("At HomePage - jdbcHandler [3]: " + jdbcWicketSpringHandler.data());        

        ...

} // End of Class //

Edit: Возможно, мне не хватает jar-файла, который зависит от Spring Jdbc или dbcp.

Вот мой список WEB-INF / lib:

antlr-2.7.6.jar
aopalliance-1.0.jar
avalon-framework-4.1.3.jar
axis-1.4.jar
axis-jaxrpc-1.4.jar
cglib-nodep-2.2.jar
commons-collections-3.1.jar
commons-dbcp-1.2.2.jar
commons-logging-1.1.jar
commons-pool-1.3.jar
derby-10.6.1.0.jar
dom4j-1.6.1.jar
hibernate-core-3.5.1-Final.jar
jetty-6.1.4.jar
jetty-management-6.1.4.jar
jetty-util-6.1.4.jar
jta-1.1.jar
log4j-1.2.14.jar
logkit-1.0.1.jar
mx4j-3.0.1.jar
mx4j-tools-3.0.1.jar
servlet-api-2.5-6.1.4.jar
servlet-api-2.5.jar
slf4j-api-1.5.8.jar
slf4j-log4j12-1.4.2.jar
spring-2.5.6.jar
spring-aop-2.5.6.jar
spring-beans-2.5.6.jar
spring-context-2.5.6.jar
spring-core-2.5.6.jar
spring-jdbc-2.5.6.jar
spring-test-2.5.6.jar
spring-tx-2.5.6.jar
testRunWrapper-1.0.0.jar
wicket-1.4.13.jar
wicket-ioc-1.4.13.jar
wicket-spring-1.4.13.jar
xml-apis-1.0.b2.jar

Ответы [ 4 ]

3 голосов
/ 08 апреля 2011

Сообщение Could not initialize class org.apache.derby.jdbc.EmbeddedDriver означает, что в этот момент JVM уже пыталась и не смогла инициализировать этот класс.

JVM не сможет инициализировать класс, если будет сгенерировано исключение, которое не будет перехвачено в статическом инициализаторе класса. Единственные причины, по которым JVM будет пытаться инициализировать класс EmbeddedDriver более одного раза, это:

  • было сгенерировано некое исключение, инициализирующее класс, это исключение было обнаружено в другом месте, и программа продолжила,
  • было сгенерировано исключение, инициализирующее класс, но затем программа ввела блок finally, и в этом блоке finally JVM попыталась снова загрузить класс.

Статический инициализатор для EmbeddedDriver ( source ) вызывает метод boot(). Однако этот метод boot() вызывает немало другого кода, поэтому трудно сказать, где может быть проблема. Я посмотрел на источник из org.apache.commons.dbcp.BasicDataSource, но кажется, что номера строк в вашей трассировке стека не согласуются с источником. Я не знаю, какую версию commons-dbcp вы используете.

Если у вас нет других выходных сообщений или трассировки стека, лучше всего подключить источник Derby к вашему отладчику и просмотреть его, чтобы увидеть, что происходит.


Кроме того, вполне возможно «напечатать» класс, который не был инициализирован. Рассмотрим следующие классы:
class St1 {
    static {
        System.out.println("In static initializer");
    }
}

public class St2 {
    public static void main(String[] args) {
        System.out.println(St1.class);
        System.out.println(new St1());
    }
}

Когда я запускаю класс St2, я получаю следующий вывод:

class St1
In static initializer
St1@65690726
1 голос
/ 10 апреля 2011

Таким образом, ошибка «Не удалось инициализировать класс org.apache.derby.jdbc.EmbeddedDriver» на самом деле была основным симптомом некоторых других, менее очевидных проблем с загрузкой классов.

Я использовал Jetty в качестве сетисервер и Spring в качестве фреймворка в java6.

Я полагаю, что возникла проблема с загрузкой классов, связанная с классом MBeanServer.

И я проигнорировала ошибку, которая произошла при запуске: «Причины: java.lang.LinkageError: нарушение ограничения загрузчика: loader (экземпляр org / mortbay / jetty / webapp / WebAppClassLoader) ранее инициировал загрузку для другого типа с именем «javax / management / MBeanServer» в java.lang.ClassLoader.defineClass1 (Собственный метод) в java.lang.ClassLoader.defineClassCond (ClassLoader.java:632) в java.lang.ClassLoader.defineClass (ClassLoader.java:616) в java.security.SecureClassLoader.defineClass (SecureClassLoader)

Я искал класс в моем каталоге WEB-INF / lib.Он был включен как часть mx4j: jar.Mx4j был зависимостью от jetty-management.jar.Мне действительно не требовалось управление Jetty, поэтому я удалил эту ссылку из моего файла pom.

По сути, включение MBeanServer (из mx4j) вызвало какую-то проблему с загрузкой классов, где org.apache.derby.jdbc.EmbeddedDriver не может быть загружен должным образом.Я удалил его из своего веб-приложения, и приложение начало работать правильно.

0 голосов
/ 08 апреля 2011

Иногда эти ошибки происходят из-за того, что у вас в пути к классу дважды дерби.В современных JDK драйверы Derby загружаются автоматически, а это означает, что JDK будет искать драйверы JDBC в пути к классам и автоматически загружать их.Таким образом, вы можете проверить системный путь к классам, а также библиотеки вашего приложения;возможно, у вас есть вторая копия Derby, спрятанная где-то в пути, и исключение пытается сообщить вам, что две версии Derby находятся в конфликте.

0 голосов
/ 08 апреля 2011

Вы уверены, что ваш derby.jar содержит класс org.apache.derby.jdbc.EmbeddedDriver?Возможно, вы захотите проверить, что у вас правильная версия.

Если вы используете Maven для упаковки приложения, то странно, что имя фляги заканчивается derby.jar без присоединенного номера версии.

...