发送分布式事务消息

发送分布式事务消息

目前支持的域包括公网测试、华东1、华北2、华东2、华南1。

MQ事务消息交互流程如下:

MQ事务消息交互流程

发送事务消息包含以下两个步骤:

  1. 发送半消息及执行本地事务

     
    1. package com.alibaba.webx.TryHsf.app1;
    2. import com.aliyun.openservices.ons.api.Message;
    3. import com.aliyun.openservices.ons.api.PropertyKeyConst;
    4. import com.aliyun.openservices.ons.api.SendResult;
    5. import com.aliyun.openservices.ons.api.transaction.LocalTransactionExecuter;
    6. import com.aliyun.openservices.ons.api.transaction.TransactionProducer;
    7. import com.aliyun.openservices.ons.api.transaction.TransactionStatus;
    8. import java.util.Properties;
    9. import java.util.concurrent.TimeUnit;
    10. publicclassTransactionProducerClient{
    11. privatefinalstaticLogger log =ClientLogger.getLog();// 用户需要设置自己的log, 记录日志便于排查问题
    12. publicstaticvoidmain(String[] args)throwsInterruptedException{
    13. finalBusinessService businessService =newBusinessService();// 本地业务Service
    14. Properties properties =newProperties();
    15. properties.put(PropertyKeyConst.ProducerId,"");// 您在控制台创建的Producer ID
    16. properties.put(PropertyKeyConst.AccessKey,"");// 阿里云身份验证,在阿里云服务器管理控制台创建
    17. properties.put(PropertyKeyConst.SecretKey,"");// 阿里云身份验证,在阿里云服务器管理控制台创建
    18. //公有云生产环境:http://onsaddr-internal.aliyun.com:8080/rocketmq/nsaddr4client-internal
    19. //公有云公测环境:http://onsaddr-internet.aliyun.com/rocketmq/nsaddr4client-internet
    20. //杭州金融云环境:http://jbponsaddr-internal.aliyun.com:8080/rocketmq/nsaddr4client-internal
    21. //杭州深圳云环境:http://mq4finance-sz.addr.aliyun.com:8080/rocketmq/nsaddr4client-internal
    22. properties.put(PropertyKeyConst.ONSAddr,
    23. "http://onsaddr-internal.aliyun.com:8080/rocketmq/nsaddr4client-internal");//此处以公有云生产环境为例
    24. TransactionProducer producer =ONSFactory.createTransactionProducer(properties,
    25. newLocalTransactionCheckerImpl());
    26. producer.start();
    27. Message msg =newMessage("Topic","TagA","Hello MQ transaction===".getBytes());
    28. // 输入您在控制台创建的Topic
    29. SendResult sendResult = producer.send(msg,newLocalTransactionExecuter(){
    30. @Override
    31. publicTransactionStatusexecute(Message msg,Object arg){
    32. // 消息ID(有可能消息体一样,但消息ID不一样, 当前消息ID在控制台无法查询)
    33. String msgId = msg.getMsgID();
    34. // 消息体内容进行crc32, 也可以使用其它的如MD5
    35. long crc32Id =HashUtil.crc32Code(msg.getBody());
    36. // 消息ID和crc32id主要是用来防止消息重复
    37. // 如果业务本身是幂等的, 可以忽略, 否则需要利用msgId或crc32Id来做幂等
    38. // 如果要求消息绝对不重复, 推荐做法是对消息体body使用crc32或md5来防止重复消息
    39. Object businessServiceArgs =newObject();
    40. TransactionStatus transactionStatus =TransactionStatus.Unknow;
    41. try{
    42. boolean isCommit =
    43. businessService.execbusinessService(businessServiceArgs);
    44. if(isCommit){
    45. // 本地事务成功、提交消息
    46. transactionStatus =TransactionStatus.CommitTransaction;
    47. }else{
    48. // 本地事务失败、回滚消息
    49. transactionStatus =TransactionStatus.RollbackTransaction;
    50. }
    51. }catch(Exception e){
    52. log.error("Message Id:{}", msgId, e);
    53. }
    54. System.out.println(msg.getMsgID());
    55. log.warn("Message Id:{}transactionStatus:{}", msgId, transactionStatus.name());
    56. return transactionStatus;
    57. }
    58. },null);
    59. // demo example 防止进程退出(实际使用不需要这样)
    60. TimeUnit.MILLISECONDS.sleep(Integer.MAX_VALUE);
    61. }
    62. }
  2. 提交事务消息状态

    当本地事务执行完成(执行成功或执行失败),需要通知服务器当前消息的事务状态。通知方式有以下两种:

    • 执行本地事务完成后提交
    • 执行本地事务一直没提交状态,等待服务器回查消息的事务状态

    事务状态有以下三种:

    • TransactionStatus.CommitTransaction 提交事务,允许订阅方消费该消息
    • TransactionStatus.RollbackTransaction 回滚事务,消息将被丢弃不允许消费
    • TransactionStatus.Unknow 无法判断状态,期待MQ Broker向发送方再次询问该消息对应的本地事务的状态
     
    1. import com.alibaba.rocketmq.client.producer.LocalTransactionState;
    2. publicclassLocalTransactionCheckerImplimplementsLocalTransactionChecker{
    3. privatefinalstaticLogger log =ClientLogger.getLog();
    4. finalBusinessService businessService =newBusinessService();
    5. @Override
    6. publicTransactionStatuscheck(Message msg){
    7. //消息ID(有可能消息体一样,但消息ID不一样, 当前消息属于Half 消息,所以消息ID在控制台无法查询)
    8. String msgId = msg.getMsgID();
    9. //消息体内容进行crc32, 也可以使用其它的方法如MD5
    10. long crc32Id =HashUtil.crc32Code(msg.getBody());
    11. //消息ID、消息本 crc32Id主要是用来防止消息重复
    12. //如果业务本身是幂等的, 可以忽略, 否则需要利用msgId或crc32Id来做幂等
    13. //如果要求消息绝对不重复, 推荐做法是对消息体使用crc32或md5来防止重复消息.
    14. //业务自己的参数对象, 这里只是一个示例, 实际需要用户根据情况来处理
    15. Object businessServiceArgs =newObject();
    16. TransactionStatus transactionStatus =TransactionStatus.Unknow;
    17. try{
    18. boolean isCommit = businessService.checkbusinessService(businessServiceArgs);
    19. if(isCommit){
    20. //本地事务已成功、提交消息
    21. transactionStatus =TransactionStatus.CommitTransaction;
    22. }else{
    23. //本地事务已失败、回滚消息
    24. transactionStatus =TransactionStatus.RollbackTransaction;
    25. }
    26. }catch(Exception e){
    27. log.error("Message Id:{}", msgId, e);
    28. }
    29. log.warn("Message Id:{}transactionStatus:{}", msgId, transactionStatus.name());
    30. return transactionStatus;
    31. }
    32. }

工具类

 
  1. import java.util.zip.CRC32;
  2. publicclassHashUtil{
  3. publicstaticlongcrc32Code(byte[] bytes){
  4. CRC32 crc32 =new CRC32();
  5. crc32.update(bytes);
  6. return crc32.getValue();
  7. }
  8. }

事务回查机制说明

  1. 发送事务消息为什么必须要实现回查Check机制?

    当步骤(1)中Half消息发送完成,但本地事务返回状态为TransactionStatus.Unknow时,或者应用退出导致本地事务未提交任何状态时,从MQ Broker的角度看,这条Half状态的消息的状态是未知的,因此MQ Broker会定期要求发送方能Check该Half状态消息,并上报其最终状态。

  2. Check被回调时,业务逻辑都需要做些什么?

    MQ事务消息的check方法里面,应该写一些检查事务一致性的逻辑。MQ发送事务消息时需要实现LocalTransactionChecker接口,用来处理MQ Broker主动发起的本地事务状态回查请求;因此在事务消息的Check方法中,需要完成两件事情:

    (1) 检查该Half消息对应的本地事务的状态(commited or rollback)

    (2) 向MQ Broker提交该Half消息本地事务的状态

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值