一: 首先是版本问题
阿里巴巴的dubbo的版本2.6.0及其以下,直接使用声明式事务,是无法发布服务的
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>dubbo</artifactId>
<version>2.6.0</version>
</dependency>
但是使用apache的dubbo,那么就能直接使用声明式事务管理,发布服务正常, 因为apache对dubbo做了优化
<dependency>
<groupId>org.apache.dubbo</groupId>
<artifactId>dubbo</artifactId>
<version>2.7.0</version>
</dependency>
二: 解决方案
- 在配置事务管理器的时候必须指定以cglib的动态代理创建对象
<!--配置发布服务需要扫描的包,包下的@Service注解的类都会被发布-->
<dubbo:annotation package="com.zuoyueer.service"/>
<!--事务管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!--事务注解驱动,proxy-target-class="true"指定cglib动态代理创建容器中的对象-->
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
- 在事务的@Service注解中必须指定interfaceClass属性,注解的包别倒错了
package com.zuoyueer.service.impl;
import com.alibaba.dubbo.config.annotation.Service;
import com.zuoyueer.service.HelloService;
import org.springframework.transaction.annotation.Transactional;
/**
* @author Zuoyueer
* Date: 2019/12/2
* Time: 10:40
* @projectName Framework
* @description: 服务层实现类, 模拟, 不写实现代码
*/
@Transactional
//必须指定接口,不然即使发布了服务,服务的路径也不对
@Service(interfaceClass = HelloService.class)
public class HelloServiceImpl implements HelloService {
//TODO
}
三: 原因分析
在HelloServiceImpl类上加入事务注解后,Spring会为此类基于JDK动态代理技术创建代理对象,创建的代理对象完整类名为com.sun.proxy.$Proxy35(最后两位数字不是固定的),导致Dubbo在进行包匹配时没有成功(因为我们在发布服务时扫描的包为com.zuoyuer.service),所以后面真正发布服务的代码没有执行。
修改HelloServiceImpl类,在Service注解中加入interfaceClass属性,值为HelloService.class,作用是指定服务的接口类型.
否则会导致发布的服务接口为SpringProxy,而不是HelloService接口