使用redisson客户端java位移运算实现多条件排名

1、使用条件说明,由于一个分区有大概1万学生按照答题分数进行实时排名。所以采用redisson的RScoredSortedSet分数排序
2、排序的条件:
第一步、按照分数排序
第二步、如果分数相等按照附加题的分数排序
第三步、如果附加题的分数还相等则按照提交答案的时间戳正序排序
3、分析步骤:
RScoredSortedSet分数是Double类型支持64位 与 Long的位数一致
0000000000000000000000000000000000000000000000000000000000000000

按照排序的先后顺序依次从高位到低位
分数 + 附加题分数 + 时间戳

3.1、由于时间戳按照正序排序所以先提交的要拍前面,说明分数要大一些。提前设置一个很大的时间戳 - 当前更新题目的系统时间 = xxxx
这里设置的是 4085218235000L;//2099年时间戳 到2099年的
转换成二进制最大 42 位
3.2、附加题由于数量有限制分数最大不操过50分,给其分配7位
附加题分数最大支持 1111111 = 2 + 4 + 8 + 16 + 32 + 64 + 128

3.3、最后剩下 64 - 42 - 7 = 15 位 可供答题分数使用

3.4、注意,如果分数存在小数先转换为非小数

4、如下图所示
在这里插入图片描述
5、代码:

public class ScoreUtils {
    private static final long PERIOD_END_TIME_STAMP = 4085218235000L;//2099年时间戳
    private static final int SCORE_LEFT = 49;//分数位移
    private static final int QUESTION_TYPE_LEFT = 42;//题目类型位移

    //排序
    public static Long toScore(Double totalStar, Long updateTime, Double challengeStar) {
        long score = 0L;
        long questionType = 0L;
        Double star = totalStar * 10;
        score = (score | star.longValue()) << SCORE_LEFT;
        questionType = (questionType | challengeStar.longValue()) << QUESTION_TYPE_LEFT;
        Long newScoreType = score | questionType;
        Long time = PERIOD_END_TIME_STAMP - updateTime;
        return newScoreType | time;
    }

    //获取分数
    public static Double getScore(Long score){
        Long scoreOrg = score >> SCORE_LEFT;
        return scoreOrg / 10d;
    }

    //获得附加题分数
    public static Double getChallengeStar(Long score){
        Long scoreOrg = score >> QUESTION_TYPE_LEFT;
        int a = 0B0000000000000001111111;
        Long star = scoreOrg & a;
        return star / 1d;
    }
}

你可能感兴趣的:(java,java,开发语言)