package kafka;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.Transaction;
import utils.Jpools;
import java.util.List;
public class TestTransaction {
public static void main(String[] args) {
TestTransaction testTransaction = new TestTransaction();
boolean reValue =false;
try{
reValue =testTransaction.transMethod(100);
}catch (InterruptedException e){
System.out.println("事务被打断,请重新执行");
}finally {
if(reValue){
System.out.println("使用信用卡消费成功");
}else{
System.out.println("使用信用卡消费失败");
}
}
}
/**
* watch命令表示标记一个键,如果标记一个键
* 在递交事务前如果该键值被别人修改过,那么事务就会失败
*
* 首先标记一个balance,然后检查余额是否足够,不足的话就取消标记,并不做实际的扣除
* 加入足够的话,就启动事务进行更新操作
* 如果在此期间键balance被其他人修改过的话,那么在递交事务时,就会报错
*/
private boolean transMethod(int amount) throws InterruptedException {
System.out.println("您使用信用卡预付款金额"+amount+"元");
Jedis jedis = Jpools.getJedis();
int balance =1000; //可用余额
int amtToSubract = amount;//实际扣款额度
int debt;
jedis.set("balance",String.valueOf(balance));
jedis.watch("balance");//监测key:balance
//jedis.set("balance","2000");
jedis.set("debt","0");//此句不该出现,为了模拟其他程序已经修改了该条目
balance=Integer.parseInt(jedis.get("balance"));
if(balance < amtToSubract){
jedis.unwatch();
System.out.println("可用余额不足");
return false;
}else{
System.out.println("启动扣费流程,执行transaction事务处理");
Transaction transaction =jedis.multi();
transaction.decrBy("balance",amtToSubract);
//金额会减去实际扣款的额度
transaction.incrBy("debt",amtToSubract);//信用卡的欠款会增加
List<Object> result = transaction.exec();//执行事务
if(result.size()==0){
//事务递交失败,说明在执行期间数据被修改过
System.out.println("扣费transaction事务执行中断...");
throw new InterruptedException();
}else{
balance = Integer.parseInt(jedis.get("balabce"));
debt=Integer.parseInt(jedis.get("debt"));
System.out.println("扣费transaction事务执行成功");
System.out.println("您的可用余额:"+balance);
System.out.println("您目前欠款:"+debt);
return true;
}
}
}
}