大家经常遇到这样的需求,尤其是支付中心接口的时候:
查询满足某种条件的订单,调用第三方接口成功,更改订单状态。
task1:
orders = queyrOrder(...); //查询已离店的订单 for (Map<String, Object> order : orders) { try { con.setAutoCommit( false ); String orderNo = (String) order.get( "order_no" ); // 调用支付中心返现接口 callCommission(order); //无论是网络异常还是返回结果失败,都抛异常 // 更改订单状态 cashedOrder(con, order); //更改订单状态 con.commit(); }
task 2
orders = queyrOrder(...); //查询“返现中”的订单 for (Map<String, Object> order : orders) { con.setAutoCommit( false ); // 调用返现任务 callCommission(order); // 更改订单状态 cashedOrder(con, order); con.commit(); }
往两个方面思考
自己系统本身数据一致性如果调用第三方接口 和 更新自己系统数据 之间任何一个环节 出异常了。比如:应用重启,机房网络问题等。如何保证第三方接口和自己系统的数据一致性?方案
加锁,方案有:版本号乐观锁、select for update 悲观锁。延伸问题:在原表上执行task,还是队列表?第三方接口需要什么特性才能保证数据一致性? 要考虑网络超时问题、应用发布重启、并发等问题。 回调提供结果查询幂等性(幂等接口的方案)
queueOrders = queryQueueOrders(); for (QueueOrder order : queueOrders){ callApi() startTransaction(); lock(order); updateStatus() commitTransaction(); }(提供结果查询的API接口的方案)
queueOrders = queryQueueOrders(); for (QueueOrder order : queueOrders){ try { startTransaction(); if (isBeingProcess(order)){ Result r = queryAPIResult(); if (r.success()){ return ; } } lock(order); callApi() updateStatus(); } finally { commitTransaction(); } }注意:该示例依然有“在事务中调用远程接口的问题”
http://www.cnblogs.com/NanguoCoffee/archive/2013/03/30/2990918.html
转载于:https://www.cnblogs.com/kms1989/p/5620950.html
相关资源:JAVA上百实例源码以及开源项目