проблема с использованием аннотации Hibernate с JodaTime - PullRequest
3 голосов
/ 13 августа 2010

Я возвращаюсь, потому что у меня все еще есть проблема с использованием JodaTime.После предыдущих комментариев я изменил свой pom, и исправление @ Type исправлено.Вот мой новый pom:

<properties>
    <org.springframework.version>3.0.3.RELEASE</org.springframework.version>
    <hibernate.version>3.6.0.Beta1</hibernate.version>
</properties>
<dependencies>
     ...
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>${hibernate.version}</version>
        <type>jar</type>
        <scope>compile</scope>
    </dependency>

    <!-- Spring Dependencies -->
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-orm</artifactId>
        <version>3.0.2.RELEASE</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-core</artifactId>
        <version>${org.springframework.version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-web</artifactId>
        <version>${org.springframework.version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>${org.springframework.version}</version>
    </dependency>
    <dependency>
        <groupId>commons-dbcp</groupId>
        <artifactId>commons-dbcp</artifactId>
        <version>20030825.184428</version>
    </dependency>
    <dependency>
        <groupId>commons-pool</groupId>
        <artifactId>commons-pool</artifactId>
        <version>20030825.183949</version>
    </dependency>
    <dependency>
        <groupId>commons-validator</groupId>
        <artifactId>commons-validator</artifactId>
        <version>1.3.1</version>
        <classifier>sources</classifier>
    </dependency>
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>5.1.5</version>
    </dependency>

    <!-- Joda Time dependencies -->
    <dependency>
        <artifactId>joda-time</artifactId>
        <groupId>joda-time</groupId>
        <version>1.6</version>
    </dependency>
    <dependency>
        <groupId>joda-time</groupId>
        <artifactId>joda-time-jsptags</artifactId>
        <version>1.0.2</version>
    </dependency>
    <dependency>
        <groupId>joda-time</groupId>
        <artifactId>joda-time-hibernate</artifactId>
        <version>1.2</version>
        <type>jar</type>
        <scope>compile</scope>
    </dependency>
</dependencies>

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

 org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'sessionFactory' defined in class path resource [spring/spring-dao.xml]: Invocation of init method failed; nested exception is java.lang.IncompatibleClassChangeError: Implementing class
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1412)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:519)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:456)
  ....  
fr.cc2i.intervention.dao.test.WebInterventionTest.oneTimedSetUp(WebInterventionTest.java:33)
    .....
    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 java.lang.Class.getDeclaredConstructors0(Native Method)
    at java.lang.Class.privateGetDeclaredConstructors(Class.java:2389)
    at java.lang.Class.getConstructor0(Class.java:2699)
    at java.lang.Class.getDeclaredConstructor(Class.java:1985)
    at org.springframework.beans.BeanUtils.instantiateClass(BeanUtils.java:104)
    at 
           .....

Согласно этому веб-сайту: http://www.mail-archive.com/joda-interest@lists.sourceforge.net/msg00609.html , jodatime может вызвать ошибку.

Может кто-нибудь объяснить мне, что такое: java.lang.incompatibleChangeError : Implementing Class?и как решить эту проблему?

Моя конфигурация пружины:

<bean id="sessionFactory"
    class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
    <property name="dataSource"> <ref local="dataSource" /></property> 
    <property name="hibernateProperties">
        <props>
            <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
            <prop key="hibernate.show_sql">false</prop>
            <prop key="hibernate.current_session_context_class">thread</prop>
            <prop key="hibernate.transaction.factory_class">org.hibernate.transaction.JDBCTransactionFactory</prop>

            <!-- Pour les requetes SQL -->
            <prop key="hibernate.query.substitutions">true 1, false 0, yes 'Y', no 'N'</prop>

            <!-- manipuler avec attention       -->
            <prop key="hibernate.hbm2ddl.auto">create</prop>
            <prop key="hibernate.hbm2ddl.auto">update</prop>

        </props>
    </property>
    <property name="annotatedClasses">
        <list>
            <value>fr.cc2i.intervention.dao.beans.Client</value>
            <value>fr.cc2i.intervention.dao.beans.Intervention</value>
            <value>fr.cc2i.intervention.dao.beans.Technicien</value>
            <value>fr.cc2i.intervention.dao.beans.Contrat</value>               
        </list>
    </property>
</bean>

Мой класс сущности:

    package fr.cc2i.intervention.dao.beans;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;

import org.hibernate.annotations.Type;
import org.joda.time.Period;

/**
 * Classe définissant un contrat
 * My code is written in French ;)
 * @author lindows
 *
 */
@Entity
@Table(name="T_Contrat")
public class Contrat {

    private long id;

    private String reference;

    private Period heures_totales;

    private Period heures_restantes;


    public Contrat(){}

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    public long getId() {
        return id;
    }

    private void setId(long id) {
        this.id = id;
    }


    public String getReference() {
        return reference;
    }

    public void setReference(String reference) {
        this.reference = reference;
    }

    @Column
    @Type(type="org.joda.time.contrib.hibernate.PersistantPeriod")
    public Period getHeures_totales() {
        return heures_totales;
    }

    public void setHeures_totales(Period heuresTotales) {
        heures_totales = heuresTotales;
    }

    @Column
    @Type(type="org.joda.time.contrib.hibernate.PersistantPeriod")
    public Period getHeures_restantes() {
        return heures_restantes;
    }

    public void setHeures_restantes(Period heuresRestantes) {
        heures_restantes = heuresRestantes;
    }

    public String toString(){
        return this.reference + " - \n Heures totales :" + this.heures_totales.toString() + 
         " Heures Restantes : " + this.heures_restantes.toString();
    }

}

Спасибо за вашу помощь

Ответы [ 3 ]

4 голосов
/ 14 августа 2010

Как я написал в комментарии, ваш pom.xml грязный, и это как-то является основной причиной проблемы. Вот полное объяснение.

Согласно анонсу Одновременных выпусков Hibernate 3.5.4 и 3.6.0. Beta1 , приведенному ниже, модуль hibernate-annotations объединен в core в Hibernate 3.6.x (стало возможным, поскольку Hibernate 3.6.x прекращает поддержку JDK 1.4):

3.6.0.Beta1

3.6 вводит некоторые новые функции, а также включает в себя большинство исправления от 3.5. Изменения конкретного проценты включают

  • Поддержка отбрасывания для JDK 1.4
  • объединение некоторых модулей в ядро; в частности, Hibernate-JMX и Зимуют-аннотаций
  • переупаковка классов в hibernate-тестировании
  • HHH-2277 : исправляет давно известное ограничение в поддержке ключ-много-к-одному.
  • HHH-5138 , HHH-5182 , HHH-5262 : совокупно адрес a ряд улучшений "Система спящего режима". Увидеть недавно отдельная и расширенная глава Типы в справочнике для подробности.
  • ЧЧЧ-5268 : поддержка java.util.UUID

Для более подробной информации о 3.6.0.Beta1, включая полный список изменений, см. страницу релиза .

И действительно, последние версии hibernate-annotations-3.6-SNAPSHOT.jar на самом деле пустых банок (из-за слияния), которые были временно собраны до полного удаления самого модуля maven. Но поскольку вы не используете hibernate-core-3.6-SNAPSHOT.jar, у вас просто нет аннотации @Type.

Теперь, вот краткое изложение моих рекомендаций / замечаний:

  • Используйте механизм разрешения транзитивных зависимостей Maven, не объявляйте все зависимости, а только "верхний уровень", т. Е. hibernate-entitymanager, если вы хотите использовать JPA.
  • Начиная с версии 3.5, Hibernate Core, Annotations, EntityManager выпущены вместе , их версии в sync .
  • Если вы хотите использовать Hibernate 3.6.x (Beta2 уже там), @Type находится в hibernate-core (это не очень важно, если вы используете переходные зависимости).
  • Я не рекомендую полагаться на SNAPSHOT, если вы точно не знаете, что делаете.

Итак, ваш pom.xml должен объявить что-то подобное для Hibernate:

<project>
  ...
  <properties>
    <!--hibernate.version>3.4.O.GA</hibernate.version-->    <!-- for JPA 1.0 -->
    <hibernate.version>3.5.4-Final</hibernate.version>      <!-- for JPA 2.0 -->
    <!--hibernate.version>3.6.0.Beta2</hibernate.version--> <!-- experimental -->
  <properties>
  ...
  <dependencies>
    <!-- Hibernate dependencies -->
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>${hibernate.version}</version>
    </dependency>
    ...    
  </dependencies>
  ...
</project>

И это все, что вам нужно.

1 голос
/ 08 октября 2010

API типов изменяется в Hibernate 3.6.0.Я был бы удивлен, если классы Joda Time Contrib работают вообще с Hibernate 3.6, хотя они должны работать с Hibernate 3.5.Могу ли я предложить вам взглянуть на мой проект, в котором есть реализации пользовательских типов для Joda Time и JSR310: http://usertype.sourceforge.net/

Спасибо и всего наилучшего,

Крис

1 голос
/ 13 августа 2010

Аннотация @Type присутствует в hibernate-annotations, версия 3.5.1-Final. Я не могу найти моментальный снимок, который вы используете 3.6.0-SNAPSHOT в репозитории jboss , поэтому я не смог протестировать это специально, но я предполагаю, что аннотация отсутствует / удалена версия снимка по какой-то причине.

Если нет особой причины, по которой вам нужен снимок, я бы придерживался выпущенных версий. Если вы вернетесь к 3.5.1-Final, это решит проблему.

...