java抢红包_Java实现抢红包功能

采用多线程模拟多人同时抢红包。服务端将玩家发出的红包保存在一个队列里,然后用Job定时将红包信息推送给玩家。每一批玩家的抢红包请求,其实操作的都是从队列中弹出的第一个红包元素,但当前的红包数量为空的时候,自动弹出下一个红包(如果有的话)。

关键思想:

1.抢红包涉及多人并发操作,需要做好同步保证多线程运行结果正确。

2.由于同时在线人数大,从性能方面考虑,玩家的发红包请求不必及时响应,而由服务端定时执行发红包队列。

下面是主要的代码和实现逻辑说明

1.创建一个类,表示红包这个实体概念。直接采用原子变量保证增减同步。Java的原子变量是一种精度更细的同步机制,在高度竞争的情况下,锁的性能将超过原子变量的性能,但在更真实的竞争情况,原子变量享有更好的性能。

public class SpringGift {

private String role;

private AtomicInteger gift;

public String getRole() {

return role;

}

public void setRole(String role) {

this.role = role;

}

public AtomicInteger getGift() {

return gift;

}

public void setGift(AtomicInteger gift) {

this.gift = gift;

}

public int getRemainCount(){

return this.gift.get();

}

}

2.采用多线程模拟多人同时抢红包。服务端将玩家发出的红包保存在一个队列里,然后用Job定时将红包信息推送给玩家。每一批玩家的抢红包请求,其实操作的都是从队列中弹出的第一个红包元素,但当前的红包数量为空的时候,自动弹出下一个红包(如果有的话)。

public class Test {

public static ConcurrentLinkedQueue queue;

public static SpringGift currGift;

public static AtomicInteger count = new AtomicInteger();

static class myThread implements Runnable{

public void run(){

handleEvent();

}

}

public static void main(String[] args) throws Exception {

queue = new ConcurrentLinkedQueue();

for(int i =0;i<3;i++){

SpringGift gift = new SpringGift();

gift.setRole("role"+i);

gift.setGift(new AtomicInteger(50));

queue.add(gift);

}

myThread mythread = new myThread();

for(int i=0;i<1000;i++){

new Thread(mythread).start();

}

System.err.println("总共收到"+count.get());

}

private static SpringGift getGift(){

//防止多条线程同时弹出队首

synchronized (queue) {//若没有加锁,打印的count总数不对!!!!

if(currGift == null || currGift.getRemainCount() <=0){

currGift = queue.poll();

}

}

return currGift;

}

public static void handleEvent(){

try{

SpringGift obj = getGift();

if(obj == null || obj.getRemainCount() <= 0){

System.err.println("没有了");

return ;

}

if(obj !=null && obj.getGift().getAndDecrement() >0 ){

System.err.println("抢到一个红包");

count.getAndIncrement();

}

Thread.sleep(500);//模拟处理其他操作

}catch(Exception e){

e.printStackTrace();

}

}

}

运行结果部分截图如下

d4dff71a0a05824b0684e0159a9fd9a1.png

需要注意的是,getGift()这个方法,由于是自动弹出队首元素,必须做好同步机制,否则,当多个请求同时操作某一个红包的最后一次剩余时,会造成总的红包数量不正确。

(将加锁的代码注释后,会发现打印的总数量有可能不正确了!)

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值