【Dubbo】解决Dubbo无法发布被事务代理的Service的问题

问题复现

配置好事务管理器,开启事务注解启动后,我们使用 @Transactional注解 进行声明式事务控制:

@Service
@Transactional
public class HelloServiceImpl implements HelloService {
    @Override
    public String sayHello(String name) {
        return "Hello " + name;
    }
}

再使用消费者,远程调用失败,找不到该注册的服务。

 
 

剖析原因

首先,声明式事务控制会基于JDK动态代理,创建出已经被事务增强的代理对象;

其次,Dubbo会为服务接口创建代理对象,远程调用的实际上是这个代理对象;

问题就出在这里!

Dubbo要去包扫描配置下的包中,去扫描@Service注解并为该注解下的服务接口创建代理对象;但是在此之前,@Transactional注解已经给接口做出代理对象了;

虽然我们依旧可以给代理对象创建代理对象,但是基于JDK动态代理出的代理对象,其完整类名是com.sun.proxy.$Proxy42(后两位数字不固定),显然已经不在@Service注解扫描所配置的包下了!

既然第二次没扫描到,Dubbo自然就没能发布该服务。

 
 

解决方案

既然BUG的根源是由于@Transactional改变了包名甚至类名,导致@Service没能被扫描到,那么我们不妨指定声明式事务控制采用cglib动态代理——它不会改变包名。

<!-- 开启事务注解驱动,并指定使用cglib动态代理 -->
<tx:annotation-driven proxy-target-class="true"/>

cglib虽然不改变包名,使得事务代理对象能被再次扫描到,但它却改变了类名(即服务接口名)。因此我们还需要再将它注册到Zookeeper的时候,将它的类名(即服务接口名)改回来。

@Service(interfaceClass = HelloService.class)
@Transactional
public class HelloServiceImpl implements HelloService {
    @Override
    public String sayHello(String name) {
        return "Hello " + name;
    }
}

 
 
 
 

 
 
 
 

 
 
 
 

⭐️

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在分布式系统中,实现同步回滚事务通常需要使用分布式事务管理器,例如Atomikos、Narayana等。在Dubbo中,可以使用Dubbo XA扩展来实现分布式事务Dubbo XA扩展是基于JTA规范的,它提供了一种在分布式环境下实现同步回滚事务的方式。具体实现方式如下: 1. 首先,需要在Dubbo服务提供者和消费者的配置文件中启用Dubbo XA扩展: ``` <dubbo:provider xa="true" /> <dubbo:consumer xa="true" /> ``` 2. 在Dubbo服务提供者中,需要使用`@Transactional`注解将服务方法标记为事务方法: ``` @Service public class UserServiceImpl implements UserService { @Autowired private UserMapper userMapper; @Override @Transactional public void addUser(User user) { userMapper.addUser(user); } } ``` 3. 在Dubbo服务消费者中,需要使用`TransactionContext`对象来实现同步回滚事务: ``` @Service public class UserServiceImpl implements UserService { @Autowired private UserMapper userMapper; @Reference(version = "1.0.0") private UserDubboService userDubboService; @Override @Transactional public void addUser(User user) { userMapper.addUser(user); // 同步回滚事务 TransactionContext transactionContext = DubboTransactionContext.getContext(); transactionContext.setXid(XidUtils.generateXid()); transactionContext.setAttachment("user", user); try { userDubboService.addUser(user); transactionContext.setStatus(TransactionContext.Status.COMMITTING); } catch (Exception e) { transactionContext.setStatus(TransactionContext.Status.ROLLBACKING); throw e; } } } ``` 在消费者调用`userDubboService.addUser(user)`之前,需要先设置`TransactionContext`对象的XID和附加属性。如果调用成功,将`TransactionContext`对象的状态设置为COMMITTING,否则将状态设置为ROLLBACKING。 通过以上步骤,就可以实现Dubbo调用中的同步回滚事务了。需要注意的是,Dubbo XA扩展仅支持同步回滚事务,不支持异步回滚。如果需要异步回滚,可以考虑使用TCC或MQ等方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值