在本篇文章中我們在SpringCloud
環境下通過使用Seata
來模擬用戶購買商品
時由于用戶余額不足導致本次訂單提交失敗,來驗證下在MySQL
數據庫內事務是否會回滾
。
本章文章只涉及所需要測試的服務列表
以及Seata
配置部分。
用戶提交訂單購買商品大致分為以下幾個步驟:
- 減少庫存
- 扣除金額
- 提交訂單
1. 準備環境
-
Seata Server
如果對
Seata Server
部署方式還不了解,請訪問:http://blog.yuqiyu.com/seata-init-env.html -
Eureka Server
服務注冊中心,如果對
Eureka Server
部署方式還不了解,請訪問http://blog.yuqiyu.com/eureka-server.html
2. 準備測試服務
為了方便學習的同學查看源碼,我們本章節源碼采用Maven Module
(多模塊)的方式進行構建。
我們用于測試的服務所使用的第三方依賴都一致,各個服務的pom.xml
文件內容如下所示:
?
<dependencies><!--Web--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!--openfeign接口定義--><dependency><groupId>org.minbox.chapter</groupId><artifactId>openfeign-service</artifactId><version>0.0.1-SNAPSHOT</version></dependency><!--公共依賴--><dependency><groupId>org.minbox.chapter</groupId><artifactId>common-service</artifactId><version>0.0.1-SNAPSHOT</version></dependency><!--seata--><dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-seata</artifactId></dependency><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId></dependency><!--Eureka Client--><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-eureka-client</artifactId></dependency><dependency><groupId>com.zaxxer</groupId><artifactId>HikariCP</artifactId></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId></dependency><dependency><groupId>org.minbox.framework</groupId><artifactId>api-boot-starter-mybatis-enhance</artifactId></dependency>
</dependencies>
2.1 Openfeign接口定義模塊
由于我們服務之間采用的Openfeign
方式進行相互調用,所以創建了一個模塊openfeign-service
來提供服務接口的定義
。
- 賬戶服務提供的接口定義
賬戶服務
對外所提供的Openfeign
接口定義如下所示:
?
/*** 賬戶服務接口** @author 恒宇少年*/
@FeignClient(name = "account-service")
@RequestMapping(value = "/account")
public interface AccountClient {/*** 扣除指定賬戶金額** @param accountId 賬戶編號* @param money 金額*/@PostMappingvoid deduction(@RequestParam("accountId") Integer accountId, @RequestParam("money") Double money);
}
-
商品服務提供的接口定義
商品服務
對外所提供的Openfeign
接口定義如下所示:
?
/*** 商品服務接口定義** @author 恒宇少年*/@FeignClient(name = "good-service")@RequestMapping(value = "/good")public interface GoodClient {/*** 查詢商品基本信息** @param goodId {@link Good#getId()}* @return {@link Good}*/@GetMappingGood findById(@RequestParam("goodId") Integer goodId);/*** 減少商品的庫存** @param goodId {@link Good#getId()}* @param stock 減少庫存的數量*/@PostMappingvoid reduceStock(@RequestParam("goodId") Integer goodId, @RequestParam("stock") int stock);}
2.2 公共模塊
公共模塊common-service
內所提供的類是共用的
,各個服務都可以調用,其中最為重要的是將Seata
所提供的數據源代理(DataSourceProxy
)實例化配置放到了這個模塊中,數據庫代理相關配置代碼如下所示:
?
/*** Seata所需數據庫代理配置類** @author 恒宇少年*/
@Configuration
public class DataSourceProxyAutoConfiguration {/*** 數據源屬性配置* {@link DataSourceProperties}*/private DataSourceProperties dataSourceProperties;public DataSourceProxyAutoConfiguration(DataSourceProperties dataSourceProperties) {this.dataSourceProperties = dataSourceProperties;}/*** 配置數據源代理,用于事務回滾** @return The default datasource* @see DataSourceProxy*/@Primary@Bean("dataSource")public DataSource dataSource() {HikariDataSource dataSource = new HikariDataSource();dataSource.setJdbcUrl(dataSourceProperties.getUrl());dataSource.setUsername(dataSourceProperties.getUsername());dataSource.setPassword(dataSourceProperties.getPassword());dataSource.setDriverClassName(dataSourceProperties.getDriverClassName());return new DataSourceProxy(dataSource);}
}
該配置類在所需要的服務中使用@Import
注解進行導入使用。
2.3 賬戶服務
-
服務接口實現
賬戶服務
用于提供接口的服務實現,通過實現openfeign-service
內提供的AccountClient
服務定義接口來對應提供服務實現,實現接口如下所示:
?
/*** 賬戶接口實現** @author 恒宇少年*/@RestControllerpublic class AccountController implements AccountClient {/*** 賬戶業務邏輯*/@Autowiredprivate AccountService accountService;@Overridepublic void deduction(Integer accountId, Double money) {accountService.deduction(accountId, money);}}
- 服務配置(application.yml)
?
# 服務名spring:application:name: account-service# seata分組cloud:alibaba:seata:tx-service-group: minbox-seata# 數據源datasource:url: jdbc:mysql://localhost:3306/testusername: rootpassword: 123456type: com.zaxxer.hikari.HikariDataSourcedriver-class-name: com.mysql.cj.jdbc.Driver# eurekaeureka:client:service-url:defaultZone: http://service:nodev2@10.180.98.83:10001/eureka/
通過spring.cloud.alibaba.seata.tx-service-group
我們可以指定服務所屬事務的分組,該配置非必填,默認為spring.application.name
配置的內容加上字符串-fescar-service-group
,如:account-service-fescar-service-group
,詳見com.alibaba.cloud.seata.GlobalTransactionAutoConfiguration
配置類源碼。
在我本地測試環境的
Eureka Server
在10.180.98.83
服務器上,這里需要修改成你們自己的地址,數據庫連接信息也需要修改成你們自己的配置。
- 導入Seata數據源代理配置
?
/*** @author 恒宇少年*/@SpringBootApplication@Import(DataSourceProxyAutoConfiguration.class)public class AccountServiceApplication {/*** logger instance*/static Logger logger = LoggerFactory.getLogger(AccountServiceApplication.class);public static void main(String[] args) {SpringApplication.run(AccountServiceApplication.class, args);logger.info("賬戶服務啟動成功.");}}
通過@Import
導入我們common-service
內提供的Seata
數據源代理配置類DataSourceProxyAutoConfiguration
。
2.4 商品服務
-
服務接口實現
商品服務提供商品的查詢以及庫存扣減接口服務,實現
openfeign-service
提供的GoodClient
服務接口定義如下所示:
?
/*** 商品接口定義實現** @author 恒宇少年*/@RestControllerpublic class GoodController implements GoodClient {/*** 商品業務邏輯*/@Autowiredprivate GoodService goodService;/*** 查詢商品信息** @param goodId {@link Good#getId()}* @return*/@Overridepublic Good findById(Integer goodId) {return goodService.findById(goodId);}/*** 扣減商品庫存** @param goodId {@link Good#getId()}* @param stock 減少庫存的數量*/@Overridepublic void reduceStock(Integer goodId, int stock) {goodService.reduceStock(goodId, stock);}}
- 服務配置(application.yml)
?
spring:application:name: good-servicecloud:alibaba:seata:tx-service-group: minbox-seatadatasource:url: jdbc:mysql://localhost:3306/testusername: rootpassword: 123456type: com.zaxxer.hikari.HikariDataSourcedriver-class-name: com.mysql.cj.jdbc.Drivereureka:client:service-url:defaultZone: http://service:nodev2@10.180.98.83:10001/eureka/server:port: 8081
- 導入Seata數據源代理配置
?
/*** @author 恒宇少年*/@SpringBootApplication@Import(DataSourceProxyAutoConfiguration.class)public class GoodServiceApplication {/*** logger instance*/static Logger logger = LoggerFactory.getLogger(GoodServiceApplication.class);public static void main(String[] args) {SpringApplication.run(GoodServiceApplication.class, args);logger.info("商品服務啟動成功.");}}
2.5 訂單服務
-
服務接口
訂單服務
提供了下單的接口,通過調用該接口完成下單功能,下單接口會通過Openfeign
調用account-service
、good-service
所提供的服務接口來完成數據驗證,如下所示:
?
/*** @author 恒宇少年*/@RestController@RequestMapping(value = "/order")public class OrderController {/*** 賬戶服務接口*/@Autowiredprivate AccountClient accountClient;/*** 商品服務接口*/@Autowiredprivate GoodClient goodClient;/*** 訂單業務邏輯*/@Autowiredprivate OrderService orderService;/*** 通過{@link GoodClient#reduceStock(Integer, int)}方法減少商品的庫存,判斷庫存剩余數量* 通過{@link AccountClient#deduction(Integer, Double)}方法扣除商品所需要的金額,金額不足由account-service拋出異常** @param goodId {@link Good#getId()}* @param accountId {@link Account#getId()}* @param buyCount 購買數量* @return*/@PostMapping@GlobalTransactionalpublic String submitOrder(@RequestParam("goodId") Integer goodId,@RequestParam("accountId") Integer accountId,@RequestParam("buyCount") int buyCount) {Good good = goodClient.findById(goodId);Double orderPrice = buyCount * good.getPrice();goodClient.reduceStock(goodId, buyCount);accountClient.deduction(accountId, orderPrice);Order order = toOrder(goodId, accountId, orderPrice);orderService.addOrder(order);return "下單成功.";}private Order toOrder(Integer goodId, Integer accountId, Double orderPrice) {Order order = new Order();order.setGoodId(goodId);order.setAccountId(accountId);order.setPrice(orderPrice);return order;}}
- 服務配置(application.yml)
?
spring:application:name: order-servicecloud:alibaba:seata:tx-service-group: minbox-seatadatasource:url: jdbc:mysql://localhost:3306/testusername: rootpassword: 123456type: com.zaxxer.hikari.HikariDataSourcedriver-class-name: com.mysql.cj.jdbc.Drivereureka:client:service-url:defaultZone: http://service:nodev2@10.180.98.83:10001/eureka/server:port: 8082
- 啟用Openfeign & 導入Seata數據源代理配置
?
/*** @author 恒宇少年*/@SpringBootApplication@EnableFeignClients(basePackages = "org.minbox.chapter.seata.openfeign")@Import(DataSourceProxyAutoConfiguration.class)public class OrderServiceApplication {/*** logger instance*/static Logger logger = LoggerFactory.getLogger(OrderServiceApplication.class);public static void main(String[] args) {SpringApplication.run(OrderServiceApplication.class, args);logger.info("訂單服務啟動成功.");}}
我們僅在order-service
調用了其他服務的Openfeign
接口,所以我們只需要在order-service
內通過@EnableFeignClients
注解啟用Openfeign
接口實現代理。
3. 服務連接Seata Server
服務想要連接到Seata Server
需要添加兩個配置文件,分別是registry.conf
、file.conf
。
-
registry.conf
注冊到
Seata Server
的配置文件,里面包含了注冊方式、配置文件讀取方式,內容如下所示:
?
registry {# file、nacos、eureka、redis、zk、consultype = "file"file {name = "file.conf"}}config {type = "file"file {name = "file.conf"}}
-
file.conf
該配置文件內包含了使用
file
方式連接到Eureka Server
的配置信息以及存儲分布式事務信息
的方式,如下所示:
?
transport {# tcp udt unix-domain-sockettype = "TCP"#NIO NATIVEserver = "NIO"#enable heartbeatheartbeat = true#thread factory for nettythread-factory {boss-thread-prefix = "NettyBoss"worker-thread-prefix = "NettyServerNIOWorker"server-executor-thread-prefix = "NettyServerBizHandler"share-boss-worker = falseclient-selector-thread-prefix = "NettyClientSelector"client-selector-thread-size = 1client-worker-thread-prefix = "NettyClientWorkerThread"# netty boss thread size,will not be used for UDTboss-thread-size = 1#auto default pin or 8worker-thread-size = 8}}## transaction log storestore {## store mode: file、dbmode = "file"## file storefile {dir = "sessionStore"# branch session size , if exceeded first try compress lockkey, still exceeded throws exceptionsmax-branch-session-size = 16384# globe session size , if exceeded throws exceptionsmax-global-session-size = 512# file buffer size , if exceeded allocate new bufferfile-write-buffer-cache-size = 16384# when recover batch read sizesession.reload.read_size = 100# async, syncflush-disk-mode = async}## database storedb {datasource = "druid"db-type = "mysql"driver-class-name = "com.mysql.jdbc.Driver"url = "jdbc:mysql://10.180.98.83:3306/iot-transactional"user = "dev"password = "dev2019."}}service {vgroup_mapping.minbox-seata = "default"default.grouplist = "10.180.98.83:8091"enableDegrade = falsedisable = false}client {async.commit.buffer.limit = 10000lock {retry.internal = 10retry.times = 30}}
配置文件內service
部分需要注意,我們在application.yml
配置文件內配置了事務分組為minbox-seata
,在這里需要進行對應配置vgroup_mapping.minbox-seata = "default"
,通過default.grouplist = "10.180.98.83:8091"
配置Seata Server
的服務列表。
將上面兩個配置文件在各個服務
resources
目錄下創建。
4. 編寫下單邏輯
在前面說了那么多,只是做了準備工作,我們要為每個參與下單的服務添加對應的業務邏輯。
-
賬戶服務
在
account-service
內添加賬戶余額扣除業務邏輯類,AccountService
如下所示:
?
/*** 賬戶業務邏輯處理** @author 恒宇少年*/@Service@Transactional(rollbackFor = Exception.class)public class AccountService {@Autowiredprivate EnhanceMapper<Account, Integer> mapper;/*** {@link EnhanceMapper} 具體使用查看ApiBoot官網文檔http://apiboot.minbox.io/zh-cn/docs/api-boot-mybatis-enhance.html** @param accountId {@link Account#getId()}* @param money 扣除的金額*/public void deduction(Integer accountId, Double money) {Account account = mapper.selectOne(accountId);if (ObjectUtils.isEmpty(account)) {throw new RuntimeException("賬戶:" + accountId + ",不存在.");}if (account.getMoney() - money < 0) {throw new RuntimeException("賬戶:" + accountId + ",余額不足.");}account.setMoney(account.getMoney().doubleValue() - money);mapper.update(account);}}
-
商品服務
在
good-service
內添加查詢商品、扣減商品庫存的邏輯類,GoodService
如下所示:
?
/*** 商品業務邏輯實現** @author 恒宇少年*/@Service@Transactional(rollbackFor = Exception.class)public class GoodService {@Autowiredprivate EnhanceMapper<Good, Integer> mapper;/*** 查詢商品詳情** @param goodId {@link Good#getId()}* @return {@link Good}*/public Good findById(Integer goodId) {return mapper.selectOne(goodId);}/*** {@link EnhanceMapper} 具體使用查看ApiBoot官網文檔http://apiboot.minbox.io/zh-cn/docs/api-boot-mybatis-enhance.html* 扣除商品庫存** @param goodId {@link Good#getId()}* @param stock 扣除的庫存數量*/public void reduceStock(Integer goodId, int stock) {Good good = mapper.selectOne(goodId);if (ObjectUtils.isEmpty(good)) {throw new RuntimeException("商品:" + goodId + ",不存在.");}if (good.getStock() - stock < 0) {throw new RuntimeException("商品:" + goodId + "庫存不足.");}good.setStock(good.getStock() - stock);mapper.update(good);}}
5. 提交訂單測試
我們在執行測試之前在數據庫內的seata_account
、seata_good
表內對應添加兩條測試數據,如下所示:
?
-- seata_good
INSERT INTO `seata_good` VALUES (1,'華為Meta 30',10,5000.00); -- seata_account
INSERT INTO `seata_account` VALUES (1,10000.00,'2019-10-11 02:37:35',NULL);
增加SEATA恢復表
DROP SCHEMA IF EXISTS zeroa;CREATE SCHEMA zeroa;USE zeroa;CREATE TABLE `undo_log` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`branch_id` bigint(20) NOT NULL,`xid` varchar(100) NOT NULL,`context` varchar(128) NOT NULL,`rollback_info` longblob NOT NULL,`log_status` int(11) NOT NULL,`log_created` datetime NOT NULL,`log_modified` datetime NOT NULL,`ext` varchar(100) DEFAULT NULL,PRIMARY KEY (`id`),UNIQUE KEY `ux_undo_log` (`xid`,`branch_id`)) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;DROP SCHEMA IF EXISTS zerob;CREATE SCHEMA zerob;USE zerob;CREATE TABLE `undo_log` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`branch_id` bigint(20) NOT NULL,`xid` varchar(100) NOT NULL,`context` varchar(128) NOT NULL,`rollback_info` longblob NOT NULL,`log_status` int(11) NOT NULL,`log_created` datetime NOT NULL,`log_modified` datetime NOT NULL,`ext` varchar(100) DEFAULT NULL,PRIMARY KEY (`id`),UNIQUE KEY `ux_undo_log` (`xid`,`branch_id`)) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
5.1 啟動服務
將我們本章所使用good-server
、order-service
、account-service
三個服務啟動。
5.2 測試點:正常購買
我們添加的賬戶余額測試數據夠我們購買兩件商品,我們先來購買一件商品驗證下接口訪問是否成功,通過如下命令訪問下單接口:
?
~ curl -X POST http://localhost:8082/order\?goodId\=1\&accountId\=1\&buyCount\=1
下單成功.
通過我們訪問/order
下單接口,根據響應的內容我們確定商品已經購買成功。
通過查看order-service
控制臺內容:
?
2019-10-11 16:52:15.477 INFO 13142 --- [nio-8082-exec-4] i.seata.tm.api.DefaultGlobalTransaction : [10.180.98.83:8091:2024417333] commit status:Committed
2019-10-11 16:52:16.412 INFO 13142 --- [atch_RMROLE_2_8] i.s.core.rpc.netty.RmMessageListener : onMessage:xid=10.180.98.83:8091:2024417333,branchId=2024417341,branchType=AT,resourceId=jdbc:mysql://localhost:3306/test,applicationData=null
2019-10-11 16:52:16.412 INFO 13142 --- [atch_RMROLE_2_8] io.seata.rm.AbstractRMHandler : Branch committing: 10.180.98.83:8091:2024417333 2024417341 jdbc:mysql://localhost:3306/test null
2019-10-11 16:52:16.412 INFO 13142 --- [atch_RMROLE_2_8] io.seata.rm.AbstractRMHandler : Branch commit result: PhaseTwo_Committed
我們可以看到本次事務已經成功Committed
。
再去驗證下數據庫內的賬戶余額
、商品庫存
是否有所扣減。
5.3 測試點:庫存不足
測試商品添加了10
個庫存,在之前測試已經銷售掉了一件商品,我們測試購買數量超過庫存數量時,是否有回滾日志,執行如下命令:
?
~ curl -X POST http://localhost:8082/order\?goodId\=1\&accountId\=1\&buyCount\=10
{"timestamp":"2019-10-11T08:57:13.775+0000","status":500,"error":"Internal Server Error","message":"status 500 reading GoodClient#reduceStock(Integer,int)","path":"/order"}
在我們good-service
服務控制臺已經打印了商品庫存不足的異常信息:
?
java.lang.RuntimeException: 商品:1庫存不足.at org.minbox.chapter.seata.service.GoodService.reduceStock(GoodService.java:42) ~[classes/:na]....
我們再看order-service
的控制臺打印日志:
?
Begin new global transaction [10.180.98.83:8091:2024417350]
2019-10-11 16:57:13.771 INFO 13142 --- [nio-8082-exec-5] i.seata.tm.api.DefaultGlobalTransaction : [10.180.98.83:8091:2024417350] rollback status:Rollbacked
通過日志可以查看本次事務進行了回滾
。
由于庫存的驗證在賬戶余額扣減之前,所以我們本次并不能從數據庫的數據來判斷事務是真的回滾。
5.4 測試點:余額不足
既然商品庫存不足我們不能直接驗證數據庫事務回滾,我們從賬戶余額不足來下手,在之前成功購買了一件商品,賬戶的余額還夠購買一件商品,商品庫存目前是9件
,我們本次測試購買5件
商品,這樣就會出現購買商品庫存充足
而余額不足
的應用場景,執行如下命令發起請求:
?
~ curl -X POST http://localhost:8082/order\?goodId\=1\&accountId\=1\&buyCount\=5
{"timestamp":"2019-10-11T09:03:00.794+0000","status":500,"error":"Internal Server Error","message":"status 500 reading AccountClient#deduction(Integer,Double)","path":"/order"}
我們通過查看account-service
控制臺日志可以看到:
?
java.lang.RuntimeException: 賬戶:1,余額不足.at org.minbox.chapter.seata.service.AccountService.deduction(AccountService.java:33) ~[classes/:na]
已經拋出了余額不足
的異常。
通過查看good-service
、order-serivce
控制臺日志,可以看到事務進行了回滾操作。
接下來查看seata_account
表數據,我們發現賬戶余額沒有改變,賬戶服務的事務回滾
驗證成功。
查看seata_good
表數據,我們發現商品的庫存也沒有改變,商品服務的事務回滾
驗證成功。
6. 總結
本章主要來驗證分布式事務框架Seata
在MySQL
下提交與回滾有效性,是否能夠完成我們預期的效果,Seata
作為SpringCloud Alibaba
的核心框架,更新頻率比較高,快速的解決使用過程中遇到的問題,是一個潛力股,不錯的選擇。
由于本章設計的代碼比較多,請結合源碼進行學習。
7. 本章源碼
請訪問<a href="https://gitee.com/hengboy/spring-cloud-chapter" target="_blank">https://gitee.com/hengboy/spring-cloud-chapter</a>查看本章源碼,建議使用git clone https://gitee.com/hengboy/spring-cloud-chapter.git
將源碼下載到本地。
- Gitee:https://gitee.com/hengboy/spring-boot-chapter
?
坑點一
如果你的項目采用是spring-cloud-alibaba-seata 0.9.0版本或以下的話,它集成了fescar-spring的0.4.2版本,如果你的seata-server服務端是采用0.5.0以上建議還是降低版本,采用0.4.2版本。因為0.4.2版本解壓是fescar-server名字,意不意外。這就是坑。而且項目引入seata依賴會與舊版本的fescar依賴沖突。
? ? ? ? ? ? <dependency>
? ? ? ? ? ? ? ? <groupId>org.springframework.cloud</groupId>
? ? ? ? ? ? ? ? <artifactId>spring-cloud-alibaba-seata</artifactId>
? ? ? ? ? ? ? ? <version>0.9.0.RELEASE</version>
? ? ? ? ? ? </dependency>
?
如果你的項目采用是spring-cloud-alibaba-seata 0.9.1(這個的seata為0.5.2)版本以上的話,那恭喜你。你可以使用seata-server的0.5.2以上的版本了。只需要在依賴這樣引入
? ? ? ?<dependency>
? ? ? ? ? ? <groupId>org.springframework.cloud</groupId>
? ? ? ? ? ? <artifactId>spring-cloud-alibaba-seata</artifactId>
? ? ? ? ? ? <version>0.9.1.BUILD-SNAPSHOT</version>
? ? ? ? ? ? <exclusions>
? ? ? ? ? ? ? ? <exclusion>
? ? ? ? ? ? ? ? ? ? <groupId>io.seata</groupId>
? ? ? ? ? ? ? ? ? ? <artifactId>seata-spring</artifactId>
? ? ? ? ? ? ? ? </exclusion>
? ? ? ? ? ? </exclusions>
? ? ? ? </dependency>
? ? ? ? <dependency>
? ? ? ? ? ? <groupId>io.seata</groupId>
? ? ? ? ? ? <artifactId>seata-all</artifactId>
? ? ? ? ? ? <version>seata-server對應的版本</version>
? ? ? ? </dependency>
?
坑點二
需要在每個服務中的resources文件中添加兩個文件file.conf和registry.conf
具體以seata-server中的file.conf和registry.conf為準。
坑點三
如果你的項目采用是spring-cloud-alibaba-seata 0.9.0版本或以下的話,因為它集成了fescar-spring的0.4.2版本,如果你是使用nacos來配置參數的話,建議使用seata-server 0.4.2,不然引入seata0.5.0以上的版本依賴會混淆,容易報以下錯誤
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'globalTransactionScanner' defined in class path resource [org/springframework/cloud/alibaba/seata/GlobalTransactionAutoConfiguration.class]: Invocation of init method failed; nested exception is java.lang.ExceptionInInitializerError
Caused by: java.lang.ExceptionInInitializerError: null
Caused by: java.lang.NullPointerException: Name is null
?? ?at java.lang.Enum.valueOf(Enum.java:236) ~[na:1.8.0_201]
?? ?at com.alibaba.fescar.core.rpc.netty.TransportProtocolType.valueOf(TransportProtocolType.java:25) ~[fescar-core-0.4.2.jar:na]
?? ?at com.alibaba.fescar.core.rpc.netty.NettyBaseConfig.<clinit>(NettyBaseConfig.java:114) ~[fescar-core-0.4.2.jar:na]
?? ?... 23 common frames omitted
?
坑點四
數據源需要引入druid依賴
/**
?* @author lgt
?*/
@Configuration
public class DatabaseConfiguration {
?? ?@Bean
?? ?@ConfigurationProperties(prefix = "spring.datasource")
?? ?public DruidDataSource druidDataSource() {
?? ??? ?return new DruidDataSource();
?? ?}
?? ?/**
?? ? * 需要將 DataSourceProxy 設置為主數據源,否則事務無法回滾
?? ? *
?? ? * @param druidDataSource The DruidDataSource
?? ? * @return The default datasource
?? ? */
?? ?@Primary
?? ?@Bean("dataSource")
?? ?public DataSource dataSource(DruidDataSource druidDataSource) {
?? ??? ?return new DataSourceProxy(druidDataSource);
?? ?}
}
ps:上面是 seata 數據源的配置,數據源采用 druid 的DruidDataSource,但實際 jdbcTemplate 執行時并不是用該數據源,而用的是 seata 對DruidDataSource的代理DataSourceProxy,所以,與 RM 相關的代碼邏輯基本上都是從DataSourceProxy這個代理數據源開始的。
坑點五
0.6.1及之前版本的啟動命令是:sh seata-server.sh 8091 file 127.0.0.1
0.7.0 及之后版本的啟動命令是:sh seata-server.sh -p 8091 -h 127.0.0.1 -m file
0.5.2及之前的版本的數據庫和之后版本數據庫是不一樣,詳細以github的文件一致
————————————————
版權聲明:本文為CSDN博主「sbit_」的原創文章,遵循CC 4.0 BY-SA版權協議,轉載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/sbit_/article/details/96112393
作者:恒宇少年
鏈接:https://www.jianshu.com/p/0a92b7c97c65
來源:簡書
著作權歸作者所有。商業轉載請聯系作者獲得授權,非商業轉載請注明出處。
?