복붙노트

[SPRING] 웹 환경에서 NoClassDefFoundError 오류 문제 Spring / Wicket / Derby / Jetty

SPRING

웹 환경에서 NoClassDefFoundError 오류 문제 Spring / Wicket / Derby / Jetty

나는 간단한 JDBC Spring Template 애플리케이션을 만들려고 노력하고 있는데, 웹 프레임 워크는 wicket이며 (Jetty Maven Plugin을 통해) 부두 6 웹 서버 아래에있다. 또한 이클립스로 앱을 만들고있다.

웬일인지 Derby jdbc 클래스와 함께 NoClassDefFoundError를 얻고 있습니다. 나는 클래스가 예외를 찾지 못했다는 것을 알았습니다. 그래서 다른 것을 추측하고 있습니다. 더비 클래스는 클래스 경로의 일부인 WEB-INF / lib 디렉토리입니다. 문제는 무엇이라고 생각하십니까?

문제에 대한 내 생각 : "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)

다음은 Spring 용 applicationContext.xml입니다.

<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)

...

여기 자바 코드입니다, 코드를 컴파일하고 클래스를 인쇄 할 수 있지만 그것을 인스턴스화 할 수 없습니다. 이상한?

자바 코드 :

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 //

편집하다: Spring Jdbc 또는 dbcp의 종속성 인 jar 파일이 누락되었을 수 있습니다.

다음은 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

해결법

  1. ==============================

    1.Could not initialize class org.apache.derby.jdbc.EmbeddedDriver 메시지는 그 시점에서 JVM이 이미이 클래스를 초기화하려고 시도했음을 의미합니다.

    Could not initialize class org.apache.derby.jdbc.EmbeddedDriver 메시지는 그 시점에서 JVM이 이미이 클래스를 초기화하려고 시도했음을 의미합니다.

    예외가 발생되어 클래스의 정적 초기화 프로그램 내에서 catch되지 않으면 JVM이 클래스를 초기화하지 못합니다. JVM이 EmbeddedDriver 클래스를 두 번 이상 초기화하려고하는 유일한 이유는 다음과 같습니다.

    EmbeddedDriver (소스)의 정적 이니셜 라이저는 boot () 메서드를 호출합니다. 그러나이 boot () 메서드는 다른 코드를 꽤 많이 호출하므로 문제가 어디 있는지 알기가 어렵습니다. org.apache.commons.dbcp.BasicDataSource 소스를 살펴 봤지만 스택 추적의 행 번호가 소스와 일치하지 않는 것 같습니다. 어떤 버전의 commons-dbcp를 사용하고 있는지 모르겠습니다.

    다른 출력 메시지 나 스택 트레이스가 없다면 더비 소스를 디버거에 연결하고 디버거를 통해 단계별로 진행하십시오. 제쳐두고 초기화되지 않은 클래스를 '인쇄'하는 것이 가능합니다. 다음 클래스를 고려하십시오.

    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
    
  2. ==============================

    2.따라서 'org.apache.derby.jdbc.EmbeddedDriver 클래스를 초기화 할 수 없습니다'오류는 실제로 덜 명백한 클래스 로딩 문제의 주된 증상이었습니다.

    따라서 'org.apache.derby.jdbc.EmbeddedDriver 클래스를 초기화 할 수 없습니다'오류는 실제로 덜 명백한 클래스 로딩 문제의 주된 증상이었습니다.

    웹 서버로 Jetty를 사용하고 Java 6에서 Spring을 프레임 워크로 사용했습니다.

    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.java:141)에서 "

    WEB-INF / lib 디렉토리에서 클래스를 검색했습니다. mx4j : jar의 일부로 포함되었습니다. Mx4j는 jetty-management.jar에 대한 종속성이었습니다. POM 파일에서 해당 참조를 제거하기 위해 부두 관리가 필요하지 않았습니다.

    기본적으로 mx4j의 MBeanServer를 포함하면 org.apache.derby.jdbc.EmbeddedDriver를 제대로로드 할 수없는 클래스 로딩 문제가 발생했습니다. 내 웹 응용 프로그램에서 제거하고 응용 프로그램이 제대로 작동하기 시작했습니다.

  3. ==============================

    3.어쨌든이 오류는 클래스 패스에 Derby가 두 번 있기 때문에 발생합니다. 최신 JDK의 경우 Derby의 드라이버는 '자동로드'되어 있습니다. 즉, JDK는 클래스 경로에서 JDBC 드라이버를 찾아 자동으로로드합니다. 따라서 응용 프로그램의 라이브러리뿐만 아니라 시스템 클래스 경로도 확인할 수 있습니다. 어쩌면 경로의 어딘가에 숨겨진 Derby의 두 번째 복사본을 가지고있을 것이며 예외적으로 두 가지 버전의 Derby가 충돌하고 있음을 알려주려고합니다.

    어쨌든이 오류는 클래스 패스에 Derby가 두 번 있기 때문에 발생합니다. 최신 JDK의 경우 Derby의 드라이버는 '자동로드'되어 있습니다. 즉, JDK는 클래스 경로에서 JDBC 드라이버를 찾아 자동으로로드합니다. 따라서 응용 프로그램의 라이브러리뿐만 아니라 시스템 클래스 경로도 확인할 수 있습니다. 어쩌면 경로의 어딘가에 숨겨진 Derby의 두 번째 복사본을 가지고있을 것이며 예외적으로 두 가지 버전의 Derby가 충돌하고 있음을 알려주려고합니다.

  4. ==============================

    4.derby.jar에 org.apache.derby.jdbc.EmbeddedDriver 클래스가 포함되어 있습니까? 올바른 버전인지 확인하고 싶을 수 있습니다.

    derby.jar에 org.apache.derby.jdbc.EmbeddedDriver 클래스가 포함되어 있습니까? 올바른 버전인지 확인하고 싶을 수 있습니다.

    Maven을 사용하여 응용 프로그램을 패키징하는 경우 jar 이름이 derby.jar이되고 버전 번호가 첨부되지 않은 것이 이상합니다.

  5. from https://stackoverflow.com/questions/5597947/issue-with-noclassdeffounderror-error-in-a-web-environment-spring-wicket-derby-j by cc-by-sa and MIT license