Что такое исключение IncompatibleClassChangeError в Java? - PullRequest
17 голосов
/ 21 августа 2010

Я работаю над небольшим приложением и пытаюсь использовать аннотации Hibernate для сопоставления моих сущностей.Я хотел проверить, все ли в порядке, когда я получил это исключение:

    Exception in thread "main" java.lang.ExceptionInInitializerError
 at fr.cc2i.intervention.dao.main.Main$HibernateUtil.<clinit>(Main.java:48)
 at fr.cc2i.intervention.dao.main.Main.test(Main.java:21)
 at fr.cc2i.intervention.dao.main.Main.main(Main.java:32)
Caused by: java.lang.IncompatibleClassChangeError: Implementing class
 at java.lang.ClassLoader.defineClass1(Native Method)
 at java.lang.ClassLoader.defineClassCond(ClassLoader.java:632)
 at java.lang.ClassLoader.defineClass(ClassLoader.java:616)
 at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:141)
 at java.net.URLClassLoader.defineClass(URLClassLoader.java:283)
 at java.net.URLClassLoader.access$000(URLClassLoader.java:58)
 at java.net.URLClassLoader$1.run(URLClassLoader.java:197)
 at java.security.AccessController.doPrivileged(Native Method)
 at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
 at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
 at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:301)
 at java.lang.ClassLoader.loadClass(ClassLoader.java:248)
 at fr.cc2i.intervention.dao.main.Main$HibernateUtil.<clinit>(Main.java:44)
 ... 2 more

Может кто-нибудь объяснить, что это за исключение?Это первый раз, когда я вижу это.Вот основная часть моего приложения:

 package fr.cc2i.intervention.dao.main;

import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.AnnotationConfiguration;

import fr.cc2i.intervention.dao.beans.Client;
import fr.cc2i.intervention.dao.beans.Contrat;

public class Main {

 public static void test(){
  Client c = new Client();
  c.setCode("123343");
  c.setAdresse("fkhdhdmh");
  c.setNom("dgsfhgsdfgs");
  c.setPhone("53456464");
  c.setContrat(new Contrat());

  Session session = HibernateUtil.getSession();
        session.beginTransaction();        
        session.save(c);
        session.getTransaction().commit();

 }

 /**
  * @param args
  */
 public static void main(String[] args) {
  Main.test();

 }

 public static class HibernateUtil {

 private static final SessionFactory sessionFactory;
     static {
         try {
             sessionFactory = new AnnotationConfiguration()
                     .configure().buildSessionFactory();
         } catch (Throwable ex) {
             // Log exception!
             throw new ExceptionInInitializerError(ex);
         }
     }

     public static Session getSession()
             throws HibernateException {
         return sessionFactory.openSession();
     }
 }

}

Моя конфигурация hibernate очень проста:

    <!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
 <session-factory>
  <!-- Database connection settings -->
  <property name="connection.driver_class">org.hsqldb.jdbcDriver</property>
  <property name="connection.url">jdbc:hsqldb:hsql://localhost</property>
  <property name="connection.username">sa</property>
  <property name="connection.password"></property>

  <!-- JDBC connection pool (use the built-in) -->
  <property name="connection.pool_size">1</property>

  <!-- SQL dialect -->
  <property name="dialect">org.hibernate.dialect.HSQLDialect</property>

  <!-- Enable Hibernate's automatic session context management -->
  <property name="current_session_context_class">thread</property>

  <!-- Disable the second-level cache  -->
  <property name="cache.provider_class">org.hibernate.cache.NoCacheProvider</property>

  <!-- Echo all executed SQL to stdout -->
  <property name="show_sql">true</property>

  <!-- Drop and re-create the database schema on startup -->
  <property name="hbm2ddl.auto">update</property>

  <mapping package="fr.cc2i.intervention.dao.beans.Client" />
  <mapping class="fr.cc2i.intervention.dao.beans.Contrat" />
  <mapping class="fr.cc2i.intervention.dao.beans.Intervention" />
  <mapping class="fr.cc2i.intervention.dao.beans.Technicien" />



 </session-factory>
</hibernate-configuration>

Вот зависимость hibernate maven, которую я использую:

<properties>
        <org.springframework.version>3.0.3.RELEASE</org.springframework.version>
        <hibernate.version>3.6.0.Beta1</hibernate.version></properties>

    <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-entitymanager</artifactId>
            <version>${hibernate.version}</version>
            <type>jar</type>
            <scope>compile</scope>
        </dependency>

Может кто-нибудь помочь мне, пожалуйста ??

Ответы [ 6 ]

25 голосов
/ 21 августа 2010

Это означает, что в какой-то момент interface был изменен на class, но разработчик исходного интерфейса не был изменен и перекомпилирован для учета этого (несовместимого) изменения.

Например,рассмотрим следующие типы:

interface Fooable {
  void foo();
}

class FooImpl implements Fooable {
  public void foo() {
     /* Do something... */
  }
}

Теперь предположим, что Fooable изменено и перекомпилировано, но FooImpl равно , а не :

abstract class Fooable {
  public abstract void foo();
}
4 голосов
/ 21 августа 2010

Поддержка JDK 1.4 была прекращена в версии 3.6. Таким образом, аннотации Hibernate были объединены обратно в Core, и больше нет необходимости иметь как Configuration, так и AnnotationConfiguration, а последние не следует использовать в 3.6.x (на самом деле, вы, вероятно, должны использовать JPA EntityManager, а не основной API, но это уже другая история).

Но мой реальный совет - использовать ветку Hibernate 3.5.x , а не 3.6.x. Как мы уже видели, команда Hibernate вносит большие изменения в 3.6, которая находится на вершине по-прежнему в бета-версии, и если вы не будете внимательно следить за изменениями, вы столкнетесь с некоторыми сюрпризами и не найдете много ресурсов в Интернете. еще. Просто используйте Hibernate 3.5 (3.5.5-Final на момент написания этой статьи).

3 голосов
/ 21 августа 2010

Дважды проверьте, совместимы ли все ваши библиотеки.Попробуйте то же самое со стабильной версией hibernate, есть вероятность, что бета-версия является дефектной, или бета-версия POM hibernate 3.6.0 имеет несовместимые зависимости.библиотеки с официальных серверов.


ОК - ваша проблема не ошибка, а новая функция .В примечаниях к выпуску Hibernate 3.6.0 Beta2 отмечается серьезное изменение по сравнению с предыдущей бета-версией: AnnotationConfiguration полностью устарела .Поэтому вам следует (а) обновить до последней бета-версии (Beta3), если вы хотите сохранить версию 3.6.0, и (б) не продолжать использовать AnnotationConfiguration.

2 голосов
/ 21 августа 2010

Правильно, erickson правильно определил проблему.

Вероятно, это вызвано наличием двух конфликтующих версий одного и того же определения типа в вашем пути к классам.Например, library-version2.jar определяет некоторые вещи, но у вас также есть библиотека-version5.jar в вашем пути к классам.

Во время выполнения это приводит к описанию боли и страданий Эриксона.

1 голос
/ 05 мая 2011

Я пишу это, чтобы помочь любому, кто найдет эту ошибку.Иногда jar-файлы, поступающие из Spring Roo, генерируют библиотеки pom.xml и Tomcat Server Runtime и конфликтуют с этой ошибкой во время модульного тестирования.При создании технологии внешнего интерфейса, такой как Flex, библиотеки времени выполнения Сервера не требуется представлять в пути к классам, поэтому просто удалите библиотеки времени выполнения Сервера из пути к классам.

0 голосов
/ 21 августа 2010

Какой сервер приложений вы используете? Может быть дубликат или конфликтующий баночка с гибернацией.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...