Dubbo学习(四):解决Dubbo无法发布被事务代理的Service问题


在前面博客的Dubbo的入门案例(Dubbo学习(三):Dubbo快速入门(含负载均衡))中,可以看到通过Dubbo提供的标签配置就可以进行包扫描,扫描到@Service注解的类就可以被发布为服务。

如果在服务提供者类上加入@Transactional事务控制注解后,服务就发布不成功了。

原因是事务控制的底层原理是为服务提供者类创建代理对象,而默认情况下Spring是基于JDK动态代理方式创建代理对象,而此代理对象的完整类名为com.sun.proxy.$Proxy42(最后两位数字不是固定的),导致Dubbo在发布服务前进行包匹配时无法完成匹配,进而没有进行服务的发布。


一、问题展示(dubbo-provider工程)

说明: 操作入门案例的服务提供者dubbo-provider工程。

(1)在pom.xml文件中增加maven坐标

<dependency>
  <groupId>mysql</groupId>
  <artifactId>mysql-connector-java</artifactId>
  <version>5.1.47</version>
</dependency>
<dependency>
  <groupId>com.alibaba</groupId>
  <artifactId>druid</artifactId>
  <version>1.1.6</version>
</dependency>
<dependency>
  <groupId>org.mybatis</groupId>
  <artifactId>mybatis-spring</artifactId>
  <version>1.3.2</version>
</dependency>

(2)在applicationContext-service.xml配置文件中加入数据源、事务管理器、开启事务注解的相关配置

<!--数据源-->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" destroy-method="close">
    <property name="username" value="root" />
    <property name="password" value="1234" />
    <property name="driverClassName" value="com.mysql.jdbc.Driver" />
    <property name="url" value="jdbc:mysql://localhost:3306/mytest" />
</bean>
<!-- 事务管理器  -->
<bean id="transactionManager"
      class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"/>
</bean>
<!--开启事务控制的注解支持-->
<tx:annotation-driven transaction-manager="transactionManager"/>

(3)在HelloServiceImpl类上加入spring事务注解:@Transactional

package net.xiaof.service.impl;

import com.alibaba.dubbo.config.annotation.Service;
import net.xiaof.service.HelloService;
import org.springframework.transaction.annotation.Transactional;

/**
 * @author zhangxh
 * @Description:
 * @date 2021-01-02
 *
 * “@service”注解的类会被发布为服务
 */
@Transactional
@Service //【注意】:此注解非spring注解,为com.alibaba.dubbo.config.annotation.Service
public class HelloServiceImpl implements HelloService {

    @Override
    public String sayHello(String name) {
        return "hello " + name;
    }

}

(4)启动服务提供者和服务消费者,并访问

访问:http://localhost:8084/dubbo/hello?name=zhangsan

报如下错: 没有可用的服务提供者
在这里插入图片描述

查看dubbo管理控制台发现服务并没有发布,如下:
访问地址:http://localhost:9000/dubbo-admin-2.6.0/

用户:root
密码:root

在这里插入图片描述

(5)断点调试查看Dubbo执行过程

在IDEA中使用快捷键“ctrl+shift+R”,搜索类AnnotationBean:

在这里插入图片描述

在AnnotationBean类的postProcessAfterInitialization方法上打断点。

在这里插入图片描述

Debug方式启动服务提供者应用:

在这里插入图片描述

断点调试自行查看(可参考下图):

在这里插入图片描述


二、解决方法(dubbo-provider工程)

通过上面的断点调试可以看到,在HelloServiceImpl类上加入事务注解后,Spring会为此类基于JDK动态代理技术创建代理对象,创建的代理对象完整类名为com.sun.proxy.$Proxy35,导致Dubbo在进行包匹配时没有成功(因为我们在发布服务时扫描的包为com.itheima.service),所以后面真正发布服务的代码没有执行。

解决方法操作步骤:

(1)指定使用cglib代理方式为Service类创建代理对象

修改applicationContext-service.xml配置文件,开启事务控制注解支持,必须指定proxy-target-class属性,值为true。其作用是使用cglib代理方式为Service类创建代理对象。

<!--开启事务控制的注解支持-->
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

(2)@Service注解指定接口类型

修改HelloServiceImpl类,在该类上dubbo源的@Service注解中加入interfaceClass属性,值为HelloService.class,作用是指定服务的接口类型

注意:
必须指定interfaceClass属性,否则会导致发布的服务接口为SpringProxy(这是spring的cglib代理默认实现的接口),而不是我们创建的HelloService接口。

package net.xiaof.service.impl;

import com.alibaba.dubbo.config.annotation.Service;
import net.xiaof.service.HelloService;
import org.springframework.transaction.annotation.Transactional;

/**
 * @author zhangxh
 * @Description:
 * @date 2021-01-02
 *
 * “@service”注解的类会被发布为服务
 * 【注意】:@Service注解非spring注解,为com.alibaba.dubbo.config.annotation.Service
 */
@Transactional
@Service(interfaceClass = HelloService.class,protocol = "dubbo")
public class HelloServiceImpl implements HelloService {

    @Override
    public String sayHello(String name) {
        return "hello " + name;
    }

}

(3)启动服务提供者和服务消费者,并访问(成功

访问:http://localhost:8084/dubbo/hello?name=zhangsan

在这里插入图片描述

(4)可查看zookeeper管控台

访问:http://localhost:9000/dubbo-admin-2.6.0/

用户:root
密码:root

在这里插入图片描述


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在分布式系统中,实现同步回滚事务通常需要使用分布式事务管理器,例如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等方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值