整体业务模块:
1,发红包模块:处理发红包的逻辑业务;
2,抢红包模块:分成点红包和拆红包模块;
3,数据库模块:发红包记录,抢红包记录,红包详情;
4,redis模块:缓存红包个数和金额;

第一步先构建数据库:创建3个表分别是
发红包记录表:主键,用户id,红包总金额,人数,全局唯一标识串,是否有效(1是是,0是否,默认是1),创建时间;
抢红包记录表:主键,红包记录id,每个红包随机金额,是否有效,创建时间;
红包详情表:主键,用户id,红包全局唯一标识串,抢红包时间,抢红包的金额,是否有效;
第二步搭建开发环境,利用mybatis的逆向工程生成表对应的实体类,操作数据的mapper和配置文件mapper.xml;

逆向工程:由于插件问题,idea一直报错,查看pom文件发现插件mybatis-generator插件出问题。找不到这个插件。换了版本依然如此,以前写的好好的逆向工程项目突然也运行不了了,可能是网络原因。使用阿里镜像依然如此。。最后使用eclipse 安装插件mybatis-generator插件 ,help–Eclipse Marketplace–搜索mybatis generator 安装插件,安装成功重启后,可以在new -others可以看到mybatis generator configuration file,这样说明插件安装成功(然后配置generatorConfig.xml文件,右键运行);

注意:WARNING: Project src does not exist Mybatis逆向工程;
与idea不同 generatorConfig.xml文件中对应的targetProject填写项目名称,idea是路径
<sqlMapGenerator targetPackage=“com.ssm.mapper” targetProject=“src\main\java”
改写为
<sqlMapGenerator targetPackage=“com.ssm.mapper” targetProject=“项目名”
表中自增ID,这个时候领域模型如果想要获取该ID的值,就需要在相应的mapper文件中添加
<insert id=“insertSelective” parameterType=“com.guo.entity.RedRecord” useGeneratedKeys=“true” keyProperty=“id”

这时候如果想继续使用idea 就可以把生成的文件copy到idea项目中即可;
使用lombok:方便编辑模板代码 @Data 使用注解,省了Getter和Setter代码的编写。。(idea首先需要安装支持lombok的插件file–settings–plugins 找到lombok安装 ,pom文件导入依赖就可以使用了;注意枚举类不能使用@Data注解 使用@Getter注解就好;

红包生成使用二倍均值算法:红包的总金额,发放的人数
public class RedPacketUtil {

public static List<Integer> divideRedPacket(Integer totalAmount,Integer totalPeopleNum){

    List<Integer> amountList = new ArrayList<>();
    if(totalAmount>0&&totalPeopleNum>0){
        Integer restAmount = totalAmount;
        Integer restPeopleNum = totalPeopleNum;
        Random random = new Random();
        //循环产生,假如10个人 循环9次,最后一个拿到剩余全部金额
        for(int i =0; i<totalPeopleNum-1;i++){
            //随机金额;二倍均值法 总金额/人数乘以2,下边这么写是左闭右开,表示随机数是[1,人均金额的2倍),红包的单个值不会超过人均金额的两倍;
            int amount = random.nextInt(restAmount/restPeopleNum*2-1)+1;
            restAmount-=amount;
            restPeopleNum--;
            //每次随机生成的金额
            amountList.add(amount);
        }
        //剩余的金额
        amountList.add(restAmount);
    }
    return amountList;
}

发红包与抢红包的业务逻辑
发红包:用户输入金额与个数,确定输入, 后台验证参数,如果正确生成红包标识串,使用红包的工具类生成随机金额的列表,产生的红包金额列表和红包个数放入到redis缓存系统中,发红包的明细表,红包记录传入数据库;

@Override
    public String handOut(RedPacketForm form) throws Exception {
        if(form.getTotal()>0&&form.getAmount()>0){
            List<Integer> list = RedPacketUtil.divideRedPacket(form.getAmount(),form.getTotal());
            //使用时间戳作为红包标识字符串
            String timeStamp = String.valueOf(System.nanoTime());
            //生成一个随机金额列表的key
            String redId = new StringBuffer(keyPrefix).append(form.getUserId()).append(":").append(timeStamp).toString();
            //将随机金额列表放入缓存中
            redisTemplate.opsForList().leftPushAll(redId,list);
            String redTotalKey = redId+":total";
            redisTemplate.opsForValue().set(redTotalKey,form.getTotal());
            redService.recordRedPacket(form,redId,list);
            return redId;
        }else{
            throw new Exception("系统异常,参数不合法!");
        }
    }
**抢红包**:

抢红包细分两个逻辑 点红包 与 拆红包;
点红包就是查询缓存系统redis中红包的个数,有过个数大于0;说明存在红包 ,进行拆红包逻辑;否则就是红包被抢完了;
拆红包逻辑:先判断当前用户是否抢过红包,查询缓存,如果获取结果不为null说明抢过了,如果抢过了,则直接返回红包金额,没有抢过就执行点红包逻辑 ,从小红包中取出一个,然后更新缓存系统红包的个数减1,并将抢到红包的账号金额信息记录在数据库中;将当前抢到红包的用户设置到缓存系统中,表示当前用户已经抢过红包了(下次再拆就提示就不是null了,直接返回红包金额);

//判断缓冲中红包总数是否大于0个
  private Boolean click(String redId){
        String RedTotalKey = redId+":total";
        Object total = redisTemplate.opsForValue().get(RedTotalKey);
        if(total!=null&&Integer.valueOf(total.toString())>0){
            return true;
        }
        return false;
    }

抢红包
多线程问题:如果同一个用户多次发送请求,点红包。。有可能出现多次请求,查询红包个数>0;后续进行拆红包逻辑,没有抢过红包,缓存列别可能同时弹出多个小红包响应多个请求,造成一个人抢到多个红包;
解决: 使用redis.setIfAbsent()方法(对应redis的setNX)对拆红包添加分布式锁;根据redId和用户Id 生成锁的键;。同一用户多次请求拆红包的时候一次只能请求一个操作,只有第一次进行拆红包逻辑,再次请求缓存中已经存在数据,直接返回红包金额;这里加锁是为了防止一个一个用户抢多个红包问题;高并发会产生大量的key锁,对key锁设置过期时间;

public BigDecimal rob(Integer userId, String redId) {
        Object obj = redisTemplate.opsForValue().get(redId+userId+":rob");
        if(obj!=null) {
            return new BigDecimal(obj.toString());
        }
        boolean res = click(redId);
        if(res){
            final String keyLock = redId+userId+"lock";
            boolean lock = redisTemplate.opsForValue().setIfAbsent(keyLock,redId);
            redisTemplate.expire(keyLock,24L,TimeUnit.HOURS);
            if(lock){
                Object value = redisTemplate.opsForList().rightPop(redId);
                if(value!=null){
                    String redTotalKey = redId+":total";
                    Integer currTotal = redisTemplate.opsForValue().get(redTotalKey)!=null?
                            (Integer)redisTemplate.opsForValue().get(redTotalKey) :0;
                    redisTemplate.opsForValue().set(redTotalKey,currTotal-1);

                    BigDecimal result = new BigDecimal(value.toString()).divide(new BigDecimal(100));

                    redService.recordRobRedPocket(userId,redId,new BigDecimal(value.toString()));

                    redisTemplate.opsForValue().set(redId+userId+":rob",result,24, TimeUnit.HOURS);
                    log.info("当前用户抢到红包了:userId={} key={} 金额={}",userId,redId,result);
                    return result;
                }
            }
        }
        return null;
    }

写入数据库操作可以设置为异步的:
@EnableAysnc注解在类上@Async注解在异步执行的方法上;

@Service
@EnableAsync
public class RedService implements IRedService {

    private static final Logger log = LoggerFactory.getLogger(RedService.class);
    @Autowired
    private RedRecordMapper redRecordMapper;
    @Autowired
    private RedDetailMapper redDetailMapper;
    @Autowired
    private RedRobRecordMapper redRobRecordMapper;

    @Override
    @Async
    @Transactional(rollbackFor = Exception.class)
    public void recordRedPacket(RedPacketForm form, String redId, List<Integer> list) {
        RedRecord redRecord = new RedRecord();
        redRecord.setUserId(form.getUserId());
        redRecord.setRedPaket(redId);
        redRecord.setTotal(form.getTotal());
        redRecord.setAmount(BigDecimal.valueOf(form.getAmount()));
        redRecord.setCreateTime(new Date());

        redRecordMapper.insertSelective(redRecord);
        RedDetail detail;
        for(Integer i:list){
            detail=new RedDetail();
            detail.setRecordId(redRecord.getId());
            detail.setAmount(BigDecimal.valueOf(i));
            detail.setCreateTime(new Date());
            redDetailMapper.insertSelective(detail);
        }

    }

@Transactional(rollbackFor=Exception.class),如果类加了这个注解,那么这个类里面的方法抛出异常,就会回滚,数据库里面的数据也会回滚。如果不加这个参数那么只有运行时异常才会回滚,非运行异常就是IO,SQLException,以及自定义的异常;
  Checked exception: 继承自 Exception 类是 checked exception。代码需要处理 API 抛出的 checked exception,要么用 catch 语句,要么直接用 throws 语句抛出去。不处理编译都通过;
Unchecked exception: 也称 RuntimeException,它也是继承自 Exception。但所有 RuntimeException 的子类都有个特点,就是代码不需要处理它们的异常也能通过编译,所以它们称作 unchecked exception;