Redis实战(6)-数据结构Set实战之获取随机乱序唯一的试卷题目

摘要:缓存中间件Redis拥有许多丰富、重要且有趣的数据结构,集合Set便是其中的一个佼佼者,其核心特性跟JavaSE集合体系中的Set几乎一毛一样,即“无序”且“唯一”,当我们向集合Set伸手要一个元素时,其底层会随机地给我们发一个元素!本文我们将继续给各位小伙伴介绍并实战另外一种典型的业务场景~从“考试系统”中获取随机、乱序且唯一的试卷题目列表。

内容:“考试”对于很多小伙伴来说应该并不陌生,像小学升初中考试、高中升大学考试、大学期间的各种各样的考试、如今许多人报名学车“科目一”的考试 以及 各种培训考证涉及的考试等等,或多或少相信大家都有经历过!

一份试卷,其核心就在于“题目”,对于监考方以及出题方而言,如何降低学员考试期间的“作弊”率则是最令人头疼的问题,不知什么时候,有位“人才”想出了一种颇有成效的方法,那就是“尽量让每位考生拿到的试卷题目是一样的(当然啦,题目总数是一样)”

现在大部分的“在线考试系统”也几乎是采取了这种方式,即考生成功登录“考试系统”后,展现在每个考生面前的试卷题目几乎是不一样的 又或者 题目是一样的,但是不同考生的题目顺序却完全是不一样的,即有差异性!有时候细想这种方式,不得不说真是一种不错的 用于预防考生现场交头接耳作弊 的法子(真TN是个天才!),

接下来,我们将基于缓存中间件Redis的集合Set实战实现“在线考试系统”中这一典型的业务场景,即获取随机、唯一且乱序的试卷题目列表,其核心业务流程如下图所示:

从该业务流程图中,我们将主要做两件事情:

A、项目启动后从数据库DB中拉出所有的试卷题目列表,并将其塞入缓存Set集合中

B、开发一请求方法,用于从缓存中获取随机、无序且唯一的 N 道试题,并将其返回给当前成功登录考试系统的考生!

(1)工欲善其事,同样也是必先利其器,首先我们需要建立一数据库表“试卷题目表”,用于存储管理员新增的题目信息,其DDL如下所示:

CREATE TABLE `problem` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `title` varchar(150) CHARACTER SET utf8mb4 DEFAULT NULL COMMENT '问题标题',
  `choice_a` varchar(100) CHARACTER SET utf8mb4 DEFAULT NULL COMMENT '选项A',
  `choice_b` varchar(100) CHARACTER SET utf8mb4 DEFAULT NULL COMMENT '选项B',
  `is_active` tinyint(4) DEFAULT '1' COMMENT '是否有效(1=是;0=否)',
  `order_by` tinyint(4) DEFAULT '0' COMMENT '排序',
  PRIMARY KEY (`id`),
  UNIQUE KEY `idx_title` (`title`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='试卷题目表';

采用Mybatis逆向工程或者代码生成器可生成该数据库表的Entity实体类、Mapper操作接口以及用于操作动态SQL的Mapper.xml。除此之外,我们还在该数据库表中新增了一系列的题目(算是测试用例),如下图所示:

(2)紧接着,我们开发一个 ProblemService服务类,实现“项目启动后前往数据库DB拉取试卷题目列表,并将其塞入缓存集合Set中去”!同时,也在该服务类ProblemService中实现“从缓存集合Set中获取随机、乱序且唯一的N道题目列表”,其完整源代码如下所示:

/**试卷题目服务
 * @Author:debug (SteadyJack)
 * @Link: weixin-> debug0868 qq-> 1948831260
**/
@Service
public class ProblemService {
    private static final Logger log= LoggerFactory.getLogger(ProblemService.class);

    @Autowired
    private ProblemMapper problemMapper;

    @Autowired
    private RedisTemplate redisTemplate;

    //TODO:项目启动拉取出数据库中的题库,并塞入缓存Set集合中
    @PostConstruct
    public void init(){
        initDBToCache();
    }

    //TODO:拉取出数据库中的所有题目列表,并塞入缓存Set集合中
    private void initDBToCache(){
        try {
            //redisTemplate.delete(Arrays.asList(Constant.RedisSetProblemKey,Constant.RedisSetProblemsKey));
            SetOperations<String,Problem> setOperations=redisTemplate.opsForSet();
            Set<Problem> set=problemMapper.getAll();
            if (set!=null && !set.isEmpty()){
                set.forEach(problem -> setOperations.add(Constant.RedisSetProblemKey,problem));
                set.forEach(problem -> setOperations.add(Constant.RedisSetProblemsKey,problem));
            }
        }catch (Exception e){
            log.error("项目启动拉取出数据库中的题库,并塞入缓存Set集合中-发生异常:",e.fillInStackTrace());
        }
    }

    //TODO:从缓存中获取随机的、乱序的试题列表
    public Set<Problem> getRandomEntitys(Integer total){
        Set<Problem> problems=Sets.newHashSet();
        try {
            SetOperations<String,Problem> setOperations=redisTemplate.opsForSet();
            problems=setOperations.distinctRandomMembers(Constant.RedisSetProblemsKey,total);
        }catch (Exception e){
            log.error("从缓存中获取随机的、乱序的试题列表-发生异常:",e.fillInStackTrace());
        }
        return problems;
    }
}

更多请见:http://www.mark-to-win.com/tutorial/51070.html 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值