持久化注解Bean后置处理器是否有用处?

16
根据它的JavaDoc,PersistenceAnnotationBeanPostProcessor似乎负责使用@PersistenceContext注释注入EntityManager。它似乎意味着如果在Spring应用程序上下文xml中未声明此bean,则@PersistenceContext注释将不起作用。
然而,根据我的实验,这并不是真相。

Persistence.xml

<persistence xmlns="http://java.sun.com/xml/ns/persistence"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/persistence 
        http://java.sun.com/xml/ns/persistence/persistence_1_0.xsd"
    version="1.0">
    <persistence-unit name="default" transaction-type="RESOURCE_LOCAL" />
</persistence>

Spring应用程序上下文XML

<context:component-scan base-package="com.test.dao" />

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="persistenceUnitName" value="default"/>
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="showSql" value="true"/>
            <property name="generateDdl" value="true"/>
            <property name="databasePlatform" value="org.hibernate.dialect.DerbyDialect"/>
        </bean>
    </property>
</bean>

<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
    <property name="driverClassName" value="org.apache.derby.jdbc.ClientDriver"/>
    <property name="url" value="jdbc:derby://localhost:1527/c:\derbydb\mydb"/>
    <property name="username" value="APP"/>
    <property name="password" value="APP"/>
</bean>

<tx:annotation-driven/>

<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
    <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>

<!-- 
    <bean id="persistenceAnnotation" class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />
 -->

UserDaoImpl

@Repository("userDao")
public class UserDaoImpl implements UserDao {

    @PersistenceContext
    protected EntityManager entityManager;

    @Transactional
    public void save(User user) {
            entityManager.persist(user);
    }
}

无论我注释还是取消注释persistenceAnnotation bean,结果都是一样的。留下这个bean不会有什么损失,但这个bean有什么用呢?
我正在使用Spring 3.0.5。
有人能提供一个场景,在该场景中去掉这个bean将导致失败吗?
此外,我不喜欢创建一个空的持久化单元来愚弄Spring。幸运的是,这个问题已经在Spring 3.1.0中得到了解决。

1
我猜测 <tx:annotation-driven/> 隐式地包含了 PersistenceAnnotationBeanPostProcessor,因此不需要显式声明。 - Abhinav Sarkar
有趣的观点。我以为tx:annotation-driven/与Transactional注释有关。但你可能是对的。 - Mingtao Sun
2个回答

16

PersistenceAnnotationBeanPostProcessor是由<context:component-scan />元素透明激活的。确切地说,是<context:annotation-config />元素激活了该bean,但是这个元素又会透明激活<context:component-scan />


7
正如Oliver Gierke所提到的,当使用基于注解的配置时,Spring会自动将org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor加载到应用程序上下文中。它的职责之一是搜索适当的实体EntityManagerFactory,为您提供@PersistenceContext注释的属性的EntityManager
如果您在spring配置/上下文中有多个EntityManagerFactory bean,并且您具有没有unitName属性的@PersistenceContext注释(假设您正在使用一个带有这样的bean的框架,并且您无法触及框架代码),您可能会遇到此异常:org.springframework.beans.factory.NoUniqueBeanDefinitionException
我发现了这个解决方法,以防您遇到这种情况:
<bean id="org.springframework.context.annotation.internalPersistenceAnnotationProcessor"
class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" >
<property name="defaultPersistenceUnitName" value="entityManagerFactory"/>
</bean> 

这将用具有defaultPersistenceUnitName的新PersistenceAnnotationBeanPostProcessor覆盖Spring加载的默认处理器。

谢谢!经过数小时的研究,这为我解决了问题。可惜的是,我不明白为什么在默认的EntityManagerFactory上设置主键不受支持 :(. 这将更加直观。(仍然适用于Spring 4的解决方案) - Toilal

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