JPA 2.1/Hibernate 4.3 废弃警告

7

我正在使用带有Hibernate 4.3.x实现的JPA 2.1示例应用程序。

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.1" xmlns="http://xmlns.jcp.org/xml/ns/persistence"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence
                                http://xmlns.jcp.org/xml/ns/persistence/persistence_2_1.xsd">
    <persistence-unit name="unit1">

        <provider>org.hibernate.jpa.HibernatePersistenceProvider</provider>
        <class>net.roseindia.model.Product</class>
        <properties>
            <property name="hibernate.dialect" value="org.hibernate.dialect.MySQLDialect"/>
            <property name="hibernate.connection.url" value="jdbc:mysql://localhost:3306/common"/>
            <property name="hibernate.connection.driver_class" value="com.mysql.jdbc.Driver"/>
            <property name="hibernate.connection.username" value="root"/>
            <property name="hibernate.connection.password" value="root"/>
            <property name="hibernate.show_sql" value="true"/>
            <property name="hibernate.format_sql" value="true"/>
        </properties>

    </persistence-unit>
</persistence>

pom.xml文件中,我有以下依赖项。
<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-entitymanager</artifactId>
    <version>4.3.5.Final</version>
</dependency>

这个示例命令行应用程序运行正常(它非常简单),但是当我启动它时,我会收到以下警告消息。

Apr 13, 2014 1:12:43 PM org.hibernate.ejb.HibernatePersistence logDeprecation
WARN: HHH015016: Encountered a deprecated javax.persistence.spi.PersistenceProvider [org.hibernate.ejb.HibernatePersistence]; use [org.hibernate.jpa.HibernatePersistenceProvider] instead.

那么,这个问题是我的配置有误引起的(我能避免吗?),还是Hibernate实现中存在问题?

更新:

这是我使用的代码:

import net.roseindia.model.Product;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;

public class AppTest {

private static final String PERSISTENCE_UNIT_NAME = "unit1";
private static EntityManagerFactory factory;

public class AppTest {

    private static final String PERSISTENCE_UNIT_NAME = "unit1";
    private static EntityManagerFactory factory;

    public static void main(String[] args) {
        factory = Persistence.createEntityManagerFactory(PERSISTENCE_UNIT_NAME);
        EntityManager em = factory.createEntityManager();

        em.getTransaction().begin();

        Product product = new Product();
        product.setProductName("JPA 2.1 Book");
        product.setProductDescription("This is the latest book on JPA 2.1");
        product.setStockQty(100.00);
        product.setPrice(95.99);
        em.persist(product);
        em.getTransaction().commit();
        em.close();
        factory.close();

    }
}

1
展示一下你是如何创建EntityManagerFactory的。在某个地方,你正在使用旧的org.hibernate.ejb.HibernatePersistence而不是新的org.hibernate.jpa.HibernatePersistenceProvider。简单搜索一下你的代码就可以找到它。 - JB Nizet
@JBNizet,我更新了运行的类。这样就够了吗? - Ashot Karakhanyan
1
听起来像是一个 bug。可能是 https://hibernate.atlassian.net/browse/HV-863。或者加载的 persistence.xml 实际上不是你向我们展示的那个。 - JB Nizet
不,我提供的是完全相同的内容,就是我正在运行的。是的,我也发现了一些类似的问题(https://community.jboss.org/thread/233644),但不适用于 hibernate 4.3.x 版本。所以我想可能是我做错了什么。不管怎样,非常感谢。 - Ashot Karakhanyan
你在使用 Spring 吗?可能是 https://jira.spring.io/browse/SPR-11240。 - GreyBeardedGeek
不,@GreyBeardedGeek,这是一个命令行简单代码。是的,我将同意这是一个错误。 - Ashot Karakhanyan
3个回答

7

对于Spring的用户来说,这是一个问题:即使Spring的开发人员解决了使用HibernateJpaVendorAdapter类的问题,实际上你仍需要告诉你的使用此类。我通过以下配置已经移除了警告(注意,我为jpaVendorAdapter指定了一个bean):

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalEntityManagerFactoryBean">
    <property name="persistenceUnitName" value="goal-control-unit" />
    <property name="jpaDialect">
        <bean class="cl.antica.goalcontrol.util.ExtendedHibernateJPADialect" />
    </property>
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"/>
    </property> 
</bean>

我已经对你的代码进行了调试,发现Persistence类自动加载了一个被弃用的类。这是createEntityManagerFactory实际的代码,当你只传递持久化单元名称时,它将被调用:

public static EntityManagerFactory createEntityManagerFactory(String persistenceUnitName, Map properties) {
    EntityManagerFactory emf = null;
    List<PersistenceProvider> providers = getProviders();
    for ( PersistenceProvider provider : providers ) {
        emf = provider.createEntityManagerFactory( persistenceUnitName, properties );
        if ( emf != null ) {
            break;
        }
    }
    if ( emf == null ) {
        throw new PersistenceException( "No Persistence provider for EntityManager named " + persistenceUnitName );
    }
    return emf;
}

默认情况下,getProviders() 方法会将 org.hibernate.ejb.HibernatePersistence 作为第一个可能的提供者(我猜这是一种备选方案)。我会采取以下方法:

import java.util.List;
import java.util.Map;

import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
import javax.persistence.PersistenceException;
import javax.persistence.spi.PersistenceProvider;
import javax.persistence.spi.PersistenceProviderResolverHolder;

import org.hibernate.ejb.HibernatePersistence;

@SuppressWarnings({"deprecation", "rawtypes"})
public class CustomPersistence extends Persistence {

    public static EntityManagerFactory createEntityManagerFactory(String persistenceUnitName) {
        return CustomPersistence.createEntityManagerFactory(persistenceUnitName, null);
    }

    public static EntityManagerFactory createEntityManagerFactory(String persistenceUnitName, Map properties) {
        EntityManagerFactory emf = null;
        List<PersistenceProvider> providers = getProviders();
        PersistenceProvider defaultProvider = null;
        for (PersistenceProvider provider : providers) {
            if (provider instanceof HibernatePersistence) {
                defaultProvider = provider;
                continue;
            }
            emf = provider.createEntityManagerFactory(persistenceUnitName, properties);
            if (emf != null) {
                break;
            }
        }
        if (emf == null && defaultProvider != null)
            emf = defaultProvider.createEntityManagerFactory( persistenceUnitName, properties );
        if ( emf == null ) {
            throw new PersistenceException( "No Persistence provider for EntityManager named " + persistenceUnitName );
        }
        return emf;
    }

    protected static List<PersistenceProvider> getProviders() {
        return PersistenceProviderResolverHolder
                .getPersistenceProviderResolver()
                .getPersistenceProviders();
    }

}

我已测试该代码,似乎足以解决问题。在您的代码中,您只需要用 CustomPersistence 替换 Persistence 类即可。希望这可以帮到您。


Shaka,通常我会与Spring一起使用它,但在这种情况下,我仅使用了JPA,并且收到了警告。我想确保警告原因不是我的配置错误。无论如何,谢谢。 - Ashot Karakhanyan
我已经弄清楚了整个草图。希望这真的能帮到你。 - Shaka Shadows
参见:https://hibernate.atlassian.net/browse/HHH-9141(已在5.0.0 beta1中修复)。 - Dave Jarvis

1

通过遵循弃用警告中的建议,我发现可以相对整洁地创建实体管理器工厂来克服这个问题。

从这里开始:

WARN: HHH015016: Encountered a deprecated javax.persistence.spi.PersistenceProvider [org.hibernate.ejb.HibernatePersistence]; use [org.hibernate.jpa.HibernatePersistenceProvider] instead.

我们可以从HibernatePersistenceProvider javadocs中看到获得与javax.persistence.EntityManagerFactory兼容的类的路径。首先需要一个HibernatePersistenceProvider实例,然后调用createEntityManagerFactory方法。这是Spring Bean布线,可以避免过时警告:
<bean id="persistenceProvider" class="org.hibernate.jpa.HibernatePersistenceProvider">
</bean>
<bean id="emf" factory-bean="persistenceProvider" factory-method="createEntityManagerFactory">
   <constructor-arg value="testpersistence"/>
   <constructor-arg><map/></constructor-arg>
</bean>

这里有一个Stackoverflow帖子链接,展示了Maven依赖。


1
这个问题似乎已经被归类为Hibernate的一个错误,并在版本5.0.0.Beta1中得到了修复,详见https://hibernate.atlassian.net/browse/HHH-9141
至少我也遇到了同样的问题,将依赖项更改为此版本后,我的警告消失了。
<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-entitymanager</artifactId>
    <version>5.0.0.Beta1</version>
</dependency>

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接