spring 使用 tx:annotation-driven 防止自动装配 bean

声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow 原文地址: http://stackoverflow.com/questions/17426406/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me): StackOverFlow

提示:将鼠标放在中文语句上可以显示对应的英文。显示中英文
时间:2020-09-08 06:03:31  来源:igfitidea点击:

Using tx:annotation-driven prevents Autowiring a bean

springspring-mvcosgitransactionaleclipse-virgo

提问by Alvaro Cavalcanti

I'm developing a module on an OSGi application, using Spring MVC and Virgo Webserver.

我正在使用 Spring MVC 和 Virgo Webserver 在 OSGi 应用程序上开发一个模块。

On my module I have a Controller, that access a Manager, which has a list of handlers that are responsible for handling report generation.

在我的模块上,我有一个控制器,它访问一个管理器,它有一个负责处理报告生成的处理程序列表。

Everything was doing fine until I had to call a transactional method from an external service. Since none of my classes were transactional I had to add the references to the transation manager and annotation-driven. Then, my Manager stopped being notified.

一切都很好,直到我不得不从外部服务调用事务方法。由于我的类都不是事务性的,因此我必须添加对事务管理器和annotation-driven. 然后,我的经理不再收到通知。

I understand that when using annotation-drivenall my beans must implement a public interface in order for the proxying mechanism to work. And as far as I know, all the classes are (one of them wasn't, but then I changed it).

我知道在使用annotation-driven我所有的 bean 时必须实现一个公共接口,以便代理机制工作。据我所知,所有课程都是(其中一个不是,但后来我改变了它)。

My configuration files are:

我的配置文件是:

bundle-context.xml:

捆绑上下文.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">
<context:annotation-config />
<tx:annotation-driven transaction-manager="transactionManager"/>
<bean id="reportManager" class="reportmodule.manager.impl.ReportManagerImpl"/>
<bean id="mvpepReportHandler" class="reportmodule.manager.impl.MVPEPReportHandler"/>
<bean id="reportConfigDao" class="reportmodule.repository.impl.ReportConfigurationHibernateDAOImpl"/>
<bean id="oSGIChangeReportHandler" class="reportmodule.osgi.impl.OSGIChangeReportHandlerImpl"/>
<bean id="reportController"
class="reportmodule.controller.impl.ReportControllerImpl"/>
<bean id="reportControllerHandlerMapping"
class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
<property name="mappings">
<value>/module/reportController/**=reportController</value>
</property>
<property name="alwaysUseFullPath" value="true"></property>
</bean>
</beans>

and my bundle-osgi.xml is as follows:

我的 bundle-osgi.xml 如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:osgi="http://www.springframework.org/schema/osgi"
xmlns:osgi-compendium="http://www.springframework.org/schema/osgi-compendium"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/osgi
http://www.springframework.org/schema/osgi/spring-osgi.xsd
http://www.springframework.org/schema/osgi-compendium
http://www.springframework.org/schema/osgi-compendium/spring-osgi-compendium-1.2.xsd">

<osgi:reference id="transactionManager" interface="org.springframework.transaction.PlatformTransactionManager" />
<osgi:reference id="sessionFactory" interface="org.hibernate.SessionFactory" />
<osgi:reference id="smaCoreUtilService" interface="core.util.service.SmaCoreUtilService" />

<osgi:service ref="reportControllerHandlerMapping"
interface="org.springframework.web.servlet.HandlerMapping"
context-class-loader="service-provider"
auto-export="interfaces"/>

<osgi:service interface="reportmodule.api.manager.ReportManager" ref="reportManager" auto-export="interfaces"/>

<osgi:service interface="reportmodule.api.manager.ReportHandler" ref="mvpepReportHandler" auto-export="interfaces"/>

<osgi:service interface="reportmodule.repository.ReportConfigurationDAO" ref="reportConfigDao" auto-export="interfaces"/>

<osgi:service interface="reportmodule.osgi.OSGIChangeReportHandler" ref="oSGIChangeReportHandler" auto-export="interfaces"/>

<osgi:list cardinality="0..N" id="reportHandler" interface="reportmodule.api.manager.ReportHandler" greedy-proxying="true">
<osgi:listener ref="oSGIChangeReportHandler" bind-method="register" unbind-method="unregister"/>    
</osgi:list>

</beans>

So, after all the services are being published the oSGIChangeReportHandler.register is called (I'm able to debbug it):

因此,在发布所有服务后,将调用 oSGIChangeReportHandler.register(我可以对其进行调试):

@Service(value="oSGIChangeReportHandler")
public class OSGIChangeReportHandlerImpl implements OSGIChangeReportHandler {

    private ReportManager reportManager;

    /**
     * @param reportManager the reportManager to set
     */
    @Autowired
    public void setReportManager(ReportManager reportManager) {
        this.reportManager = reportManager;
    }

    @SuppressWarnings("rawtypes")
    public void register(ReportHandler reportHandler, Map properties) {
        reportManager.addReportHandler(reportHandler);
    }

    @SuppressWarnings("rawtypes")
    public void unregister(ReportHandler reportHandler, Map properties) {
        reportManager.removeReportHandler(reportHandler);
    }


}

And although the debugger shows Proxies for both the reportManagerand reportHandleron the registermethod, the debugger does not halts on the ReportManagerImpl.addReportHandlermethod:

尽管调试器在方法上显示了reportManager和 的代理,但调试器不会在方法上停止:reportHandlerregisterReportManagerImpl.addReportHandler

@Service(value="reportManager")
@Transactional(propagation = Propagation.MANDATORY, rollbackFor = Exception.class)
public class ReportManagerImpl implements ReportManager {

    private ReportConfigurationDAO reportConfigurationDAO;

    private ArrayList<ReportHandler> reportHandlers = new ArrayList<ReportHandler>();

    /**
     * @param reportConfigurationDAO the reportConfigurationDAO to set
     */
    @Autowired
    public void setReportConfigurationDAO(ReportConfigurationDAO reportConfigurationDAO) {
        this.reportConfigurationDAO = reportConfigurationDAO;
    }

    @Override
    @Transactional
    public InputStream gerarRelatorio(ReportRequest repoReq) throws NegocioException {
        // Generates the report...
    }

    /* (non-Javadoc)
     * @see reportmodule.api.manager.ReportManager#addReportHandler(reportmodule.api.manager.ReportHandler)
     */
    @Override
    public void addReportHandler(ReportHandler handler) {
        if (handler != null) {
            this.reportHandlers.add(handler);
        }
    }

    /* (non-Javadoc)
     * @see reportmodule.api.manager.ReportManager#removeReportHandler(reportmodule.api.manager.ReportHandler)
     */
    @Override
    public void removeReportHandler(ReportHandler handler) {
        if (handler != null) {
            this.reportHandlers.remove(handler);
        }
    }

}

I must stress that when I remove the tx:annotation-driventag from the bundle-context.xmlfile, everything works fine (the handleris properly added to the list during startup).

我必须强调,当我tx:annotation-drivenbundle-context.xml文件中删除标签时,一切正常(handler在启动期间正确添加到列表中)。

So, what am I missing here?

那么,我在这里错过了什么?

采纳答案by Alvaro Cavalcanti

Problem solved!

问题解决了!

As you can see on my code above, I was defining the beans both via XMLand Annotation, thus every bean was duplicated in runtime. Then, when I added the tx:annotation-driventag the application begun intercepting the wrong bean. It was indeed notifying a bean, but an orphan bean.

正如您在上面的代码中看到的那样,我通过XML和定义了 bean Annotation,因此每个 bean 在运行时都被复制。然后,当我添加tx:annotation-driven标签时,应用程序开始拦截错误的 bean。它确实在通知一个bean,但是一个孤儿bean。

回答by WiLLStenico

Here′s an example working with tx. Look this line: xmlns:tx="http://www.springframework.org/schema/tx"

这是一个使用 tx 的示例。看这一行: xmlns:tx="http://www.springframework.org/schema/tx"

and this on schemaLocation: http://www.springframework.org/schema/tx/spring-tx.xsd

这在 schemaLocation:http: //www.springframework.org/schema/tx/spring-tx.xsd

Source: http://www.springbyexample.org/examples/hibernate-transaction-annotation-config.html

来源:http: //www.springbyexample.org/examples/hibernate-transaction-annotation-config.html