【技术应用】java实现排行榜方案

news2025/1/16 15:03:42

【技术应用】java实现排行榜方案

    • 一、前言
    • 二、实现方案
      • 方案一、通过数据库实现
      • 方案二、通过集合List实现数据排序功能
      • 方案三、通过redis的zset实现
      • 方案四、通过java中的sortedSet集合实现
      • 方案五、通过java的priorityQueue队列实现

一、前言

最近在做一个项目的性能优化,涉及到一个实时数据排行榜功能的性能优化,原方案涉及实时数据排行榜数据是通过实时查询数据实现的,这样实现业务逻辑比较简单,但是在数据量比较多时,sql的order by操作是比较耗费性能;

=我们这里总结几种java实现排行榜的功能,供大家参考。=

二、实现方案

方案一、通过数据库实现

账号浏览量实时更新到数据库中,用户访问人气排行榜时,通过实时查询数据库获取排行榜数据,这也是我们原有的设计方案,性能比较低,在数据量和用户量比较少时,可以考虑;

方案二、通过集合List实现数据排序功能

    List<Integer> list = new ArrayList<>();
    list.add(3);
    list.add(5);
    list.add(1);

    list.sort((o1, o2) -> o2-o1);
    System.out.println(list);

Console

[5, 3, 1]

这种算法随着数据量越大,时间复杂度越高,同时我们也不可能每次查询一下排行榜数据都做一次排序计算,这种性能也是比较低的;如果通过定时排序实现,又会有数据延迟性能的问题;

我们常见的排序算法10种,如下:
在这里插入图片描述
但是不论是哪种算法通过查询时排序的方式实现排行榜的功能是不可取的,原因同上

方案三、通过redis的zset实现

redis有序集合redis集合类似,是不包含 相同字符串的合集。它们的差别是,每个有序集合 的成员都关联着一个评分,这个评分用于把有序集 合中的成员按最低分到最高分排列。

使用有序集合,你可以非常快地(O(log(N)))完成添加,删除和更新元素的操作。 因为元素是在插入时就排好序的,所以很快地通过评分(score)或者 位次(position)获得一个范围的元素。 访问有序集合的中间元素同样也是非常快的,因此你可以使用有序集合作为一个没用重复成员的智能列表。 在这个列表中, 你可以轻易地访问任何你需要的东西: 有序的元素,快速的存在性测试,快速访问集合中间元素!

在项目开发中,redis的zset是常用作排行榜功能的实现方式,但是依赖于redis组件实现,在没有redis的场景下如何实现呐?

方案四、通过java中的sortedSet集合实现

sortedSet集合有redis中zset数据类型一样属性,都是有序集合;
sortedSet实现类我们使用ConcurrentSkipListSet,这个类的命名我们能看出来它实现线程安全的,这很重要,我们实现的场景中涉及到多线程并发操作;

方案流程:
在这里插入图片描述
我们这里的样例方案是以抖音直播排行榜为例,各个直播间访客人数是动态变化的,人气排行榜也是动态实时变化的;
账号代表直播间浏览量代表实时访客数排行榜就是人气榜单,我们可以取Top N

方案描述:
1)账号是存在多个的,每个账号的浏览量也是实时变化的,每变化一次就生成一个浏览量消息推送到后台服务;
2)map集合存储账号已在sortedSet集合中存储数据的位置,以便在账号数据更新时,删除老数据,提高删除效率;
3)浏览量的排序发生在存入sortedSet时,所以获取榜单top N时,只需要变量sortedSet集合前N个元素即可,由于ConcurrentSkipListSet线程安全的,支持多线程新增/删除/查询sortedSet集合中的数据;

代码实现:
1)用户类

import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@NoArgsConstructor
@AllArgsConstructor
public class Account implements Comparable{

    private String username;
    private Integer visitedNumber;


    @Override
    public int compareTo(Object o) {
        Account account = (Account) o;
        return account.getVisitedNumber() - visitedNumber;
    }
}

2)sortedSet、map实现

package com.sk.common;

import com.sk.bean.Account;

import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentSkipListSet;

public class CommonTools {
    public final static SortedSet<Account> skipListSet = new ConcurrentSkipListSet<>();
    public final static Map<String,Account> setMap = new ConcurrentHashMap<>();
}

3)生产者线程

package com.sk.threads02;

import com.sk.bean.Account;
import com.sk.common.CommonTools;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.Random;
import java.util.concurrent.TimeUnit;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class ProducerSetThread implements Runnable{

    private String name;

    @Override
    public void run() {
        for (int i = 0; i < 100000; i++) {
            String username = name + i % 10;
            Random random = new Random();
            Integer visitedNumber = random.nextInt(30);
            Account account_old = CommonTools.setMap.get(username);
            Account account_new = new Account(username, visitedNumber);
            if (null != account_old) {
                if (!account_new.equals(account_old)) {
                    CommonTools.skipListSet.add(account_new);
                    CommonTools.setMap.put(username, account_new);
                    CommonTools.skipListSet.remove(account_old);
                }
            } else {
                CommonTools.skipListSet.add(account_new);
                CommonTools.setMap.put(username, account_new);
            }
            try {
                TimeUnit.MILLISECONDS.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

4)消费者线程

package com.sk.threads02;

import com.sk.bean.Account;
import com.sk.common.CommonTools;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.Iterator;
import java.util.concurrent.PriorityBlockingQueue;
import java.util.concurrent.TimeUnit;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class CustomerSetThread implements Runnable {

    private Integer topN;

    @Override
    public void run() {
        while (true) {
            int topNN = topN;
            int size = CommonTools.skipListSet.size();
            if(size < topNN){
                topNN = size;
            }
            Iterator<Account> iterator = CommonTools.skipListSet.iterator();
            for (int i = 0; i < topNN; i++) {
                System.out.println(i + 1 + "========" + iterator.next());
            }

            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            System.out.println("--------------------------------------------------------");

        }
    }
}

5)初始化类

package com.sk.test;

import com.sk.threads02.CustomerSetThread;
import com.sk.threads02.ProducerSetThread;

public class Test02 {

    public static void main(String[] args) {

        //消费者
        new Thread(new CustomerSetThread(5)).start();
        //生产者A
        new Thread(new ProducerSetThread("张三")).start();
        //生产者B
        new Thread(new ProducerSetThread("李四")).start();
    }

}

6)执行结果

--------------------------------------------------------
1========Account(username=张三1, visitedNumber=26)
2========Account(username=李四2, visitedNumber=24)
3========Account(username=李四6, visitedNumber=20)
4========Account(username=李四3, visitedNumber=17)
5========Account(username=张三2, visitedNumber=16)
--------------------------------------------------------
1========Account(username=李四6, visitedNumber=29)
2========Account(username=李四0, visitedNumber=28)
3========Account(username=张三0, visitedNumber=22)
4========Account(username=李四2, visitedNumber=21)
5========Account(username=李四4, visitedNumber=18)
--------------------------------------------------------

但是sortedSet集合实现排行榜有一个问题,那就是浏览量visitedNumber不能重复,因为集合sortedSet中数据是不可重复的,排序的属性也是不能重复的;我们知道浏览量是可能存在重复,那这种情况应该怎么办?

方案五、通过java的priorityQueue队列实现

PriorityQueue(优先队列) 采用的是堆排序,实际上是一个堆(不指定Comparator时默认为最小堆)
队列既可以根据元素的自然顺序来排序,也可以根据 Comparator来设置排序规则。队列的头是按指定排序方式的最小元素。如果多个元素都是最小值,则头是其中一个元素。新建对象的时候可以指定一个初始容量,其容量会自动增加。

同样,出于线程安全考虑,我们使用线程安全的实现类:PriorityBlockingQueue

PriorityBlockingQueue是一个无界的基于数组的优先级阻塞队列,数组的默认长度是11,也可以指定数组的长度,且可以无限的扩充,直到资源消耗尽为止,每次出队都返回优先级别最高的或者最低的元素。默认情况下元素采用自然顺序升序排序,当然我们也可以通过构造函数来指定Comparator来对元素进行排序。需要注意的是PriorityBlockingQueue不能保证同优先级元素的顺序。

方案流程:
在这里插入图片描述
方案描述:

1)方案流程与方案四项目节点方案描述同上;
2)sortedSet集合修改为了PriorityBlockingQueue优先队列,排序结合中可以存在相同浏览量的元素;
3)客户端访问排行榜时从队列queue中copy一份实时数据,取Top N,并不会影响原queue数据;
4)也可以只保留一个服务数据,定时从元queuecopy数据;
5)主queue队列,可以只存top N的数据,新数据在插入queue之前,先和队列queue中最小值比较,如果小于最小值,则不入队列,反之存入队列,删除最小值;这样能够节省内存空间;(注:流程图中没有体现,供大家参考);

代码实现:
1)priorityQueue、map实现

package com.sk.common;

import com.sk.bean.Account;

import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentSkipListSet;

public class CommonTools {
    public final static PriorityBlockingQueue<Account> priorityQueue = new PriorityBlockingQueue<>();
    public final static Map<String,Account> setMap = new ConcurrentHashMap<>();
}

2)生产者

package com.sk.threads;

import com.sk.bean.Account;
import com.sk.common.CommonTools;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.Random;
import java.util.concurrent.TimeUnit;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class ProducerThread implements Runnable{

    private String name;

    @Override
    public void run() {
        for (int i = 0; i < 100000; i++) {
            String username = name + i % 10;
            Random random = new Random();
            Integer visitedNumber = random.nextInt(30);
            Account account_old = CommonTools.map.get(username);
            Account account_new = new Account(username, visitedNumber);
            if (null != account_old) {
                if (!account_new.equals(account_old)) {
                    CommonTools.priorityQueue.add(account_new);
                    CommonTools.map.put(username, account_new);
                    CommonTools.priorityQueue.remove(account_old);
                }
            } else {
                CommonTools.priorityQueue.add(account_new);
                CommonTools.map.put(username, account_new);
            }
            try {
                TimeUnit.MILLISECONDS.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

3)消费者

package com.sk.threads;

import com.sk.bean.Account;
import com.sk.common.CommonTools;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.concurrent.PriorityBlockingQueue;
import java.util.concurrent.TimeUnit;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class CustomerThread implements Runnable {

    private Integer topN;

    @Override
    public void run() {
        while (true) {

            PriorityBlockingQueue<Account> queueTemp = new PriorityBlockingQueue<>();
            queueTemp.addAll(CommonTools.priorityQueue);
            int topNN = topN;
            int queueSize = queueTemp.size();
            if (queueSize < topNN) {
                topNN = queueSize;
            }
            System.out.println("+++++++++++++++++++++++++++++TOP ONE " + queueTemp.peek());
            for (int i = 0; i < topNN; i++) {
                System.out.println(i + 1 + "========" + queueTemp.remove());
            }

            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            System.out.println("--------------------------------------------------------");

        }
    }
}

4)初始化类

package com.sk.test;

import com.sk.threads.CustomerThread;
import com.sk.threads.ProducerThread;

public class Test {

    public static void main(String[] args) {

        //消费者
        new Thread(new CustomerThread(5)).start();

        //生产者A
        new Thread(new ProducerThread("张三")).start();
        //生产者B
        new Thread(new ProducerThread("李四")).start();

    }


}

5)执行结果

+++++++++++++++++++++++++++++TOP ONE Account(username=李四6, visitedNumber=24)
1========Account(username=李四6, visitedNumber=24)
2========Account(username=张三8, visitedNumber=22)
3========Account(username=张三2, visitedNumber=22)
4========Account(username=张三5, visitedNumber=22)
5========Account(username=张三0, visitedNumber=17)
--------------------------------------------------------
+++++++++++++++++++++++++++++TOP ONE Account(username=张三2, visitedNumber=28)
1========Account(username=张三2, visitedNumber=28)
2========Account(username=张三4, visitedNumber=26)
3========Account(username=李四6, visitedNumber=25)
4========Account(username=张三6, visitedNumber=24)
5========Account(username=张三7, visitedNumber=24)
--------------------------------------------------------

=注:如果大家有更好的实现方案,可以在评论区分享==========

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/198101.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

12、获取字符串信息

目录 一、获取字符串长度 二、字符串查找 &#xff08;1&#xff09;indexOf(String s) &#xff08;2&#xff09;lastIndexOf(String str) 三、获取指定索引位置的字符 一、获取字符串长度 使用String类的length()方法可获取声明的字符串对象的长度。 语法如下&#x…

Linux Centos9 Stream 安装mysql8

安装mysql8教程前言安装Mysql8.0使用Mysql yum 存储库进行安装。安装mysql8.0启动mysql 服务创建用户完成安装使用Navicat 连接刚装好的mysql如果博主的文章对您有所帮助&#xff0c;可以评论、点赞、收藏&#xff0c;支持一下博主!!!前言 操作系统&#xff1a;Linux Centos9 …

JAVA-Spring Bean作用域

目录 基本概念 Bean 作用域 spring支持的bean作用域有哪些&#xff1f; 近日研究Spring和SpringBoot的一些内容&#xff0c;给大家做一些分享&#xff0c;请大家多多提出您的宝贵意见。 学习知识要了解其涉及到的基本概念&#xff0c;才能理解这个知识&#xff0c;并且做到…

八种排序算法

文章目录1、冒泡排序1.基本思路2.代码实现3.时间复杂度和空间复杂度2、快速排序1.基本思路2.代码实现3.时间复杂度和空间复杂度3、直接插入1.基本思路2.代码实现3.时间复杂度和空间复杂度4、希尔排序1.基本思路2.代码实现3.时间复杂度和空间复杂度5、简单选择1.基本思路2.代码实…

数据库管理系统有哪些

文章目录RDBMS非RDBMSDocumentKey-valueGraphhttps://db-engines.com/en/ranking该网站根据各 DBMS的流行度&#xff0c;列出了它们的排名&#xff0c;每月更新一次。当前是2023年2月份的排名。DataBase Model这一列中显示了各 DBMS所使用的 数据模型&#xff0c;有的使用了单个…

SpringAMQP从0到1

初识MQ 同步和异步通讯 微服务间通讯有同步和异步两种方式&#xff1a; 同步通讯&#xff1a;就像打电话&#xff0c;需要实时响应。 异步通讯&#xff1a;就像发邮件&#xff0c;不需要马上回复。 两种方式各有优劣&#xff0c;打电话可以立即得到响应&#xff0c;但是你却…

Redis最佳实践 | 黑马Redis高级篇

目录 一、Redis键值设计 1、优雅的key结构 2、BigKey问题 什么是BigKey BigKey的危害 如何发现BigKey 如何删除BigKey 3、恰当的数据类型 二、批处理优化 1、Pipeline 大量数据导入的方式 MSET Pipeline 2、集群下的批处理 三、服务端优化 1、持久化配置 2、慢…

MyBatis案例 | 使用映射配置文件实现CRUD操作——多条件查询

本专栏主要是记录学习完JavaSE后学习JavaWeb部分的一些知识点总结以及遇到的一些问题等&#xff0c;如果刚开始学习Java的小伙伴可以点击下方连接查看专栏 本专栏地址&#xff1a;&#x1f525;JavaWeb Java入门篇&#xff1a; &#x1f525;Java基础学习篇 Java进阶学习篇&…

【论文阅读】Exathlon: A Benchmark for Explainable Anomaly Detection over Time Series

论文来源 标题: Exathlon: A Benchmark for Explainable Anomaly Detection over Time Series (Vincent Jacob,2021) 作者: Vincent Jacob, Fei Song, Arnaud Stiegler, Bijan Rad, Yanlei Diao, Nesime Tatbul 期刊: Proceedings of the VLDB Endowment 研究问题 Exathlon是…

尚医通(三)医院设置模块后端 | swagger | 统一日志 | 统一返回结果

目录一、医院设置模块需求二、医院设置表结构三、医院模块配置四、医院查询功能1、创建包结构&#xff0c;创建SpringBoot启动类2、编写controller代码3、创建SpringBoot配置类5、运行启动类6、统一返回的json时间格式五、医院设置逻辑删除功能1、HospitalSetController添加删除…

CDA Level Ⅱ 模拟题(二)

练习题 【单选题】1/20 一项针对全国25-35岁用户群的手机喜好调查&#xff0c;但调研项目经费大概是10万元&#xff0c;并且用户群相对集中在中国中部城市。前期预调研显示&#xff0c;用户群的数值方差和调研费用不等。以下哪种情况是比较适宜的调查方式&#xff1f; A.简单随…

【C++入门】

目录1、命名空间1.1、命名空间定义1.2、命名空间的使用2、C输入和输出3、缺省参数3.1 缺省参数概念3.2缺省参数分类4、函数重载4.1、函数重载概念4.2 C支持函数重载的原理--名字修饰5、引用5.1、引用概念5.2、引用特性5.3、常引用5.4、使用场景5.5、传值、传引用效率比较5.6、引…

【JavaEE】如何构造 HTTP请求认识HTTPS

✨哈喽&#xff0c;进来的小伙伴们&#xff0c;你们好耶&#xff01;✨ &#x1f6f0;️&#x1f6f0;️系列专栏:【JavaEE】 ✈️✈️本篇内容:如何构造 HTTP 请求同时认识HTTPS&#xff01; &#x1f680;&#x1f680;代码存放仓库gitee&#xff1a;JavaEE代码&#xff01; …

HW在即,那些被遗忘的物理安全还好吗?

近段时间&#xff0c;一个网络攻击的段子在互联网上火了起来。 “某公司被黑客勒索&#xff0c;每20分钟断一次网&#xff0c;给公司带来了极其严重的影响&#xff0c;但通过技术手段怎么也找不到问题。最后公司发现是黑客买通了保安&#xff0c;每20分钟拔一次网线。” 看完…

即时通讯系列---如何设计消息协议层方案

1. 前言 上篇即时通讯系列—如何下手做技术方案设计 最后总结出IM系统的端侧基本结构 后续文章将从下到上以此做架构设计. 本文Agenda 什么是消息同步同步协议的常见设计方案包含哪些应该采用哪种方案方案细节 2. 名词定义: 消息漫游 : 用户如何从消息服务器获取会话和消息…

MITK2021.02编译记录

编译成功效果 编译文件夹 参考教程 https://libaineu2004.blog.csdn.net/article/details/124202508?spm1001.2101.3001.6650.6&utm_mediumdistribute.pc_relevant.none-task-blog-2%7Edefault%7EOPENSEARCH%7ERate-6-124202508-blog-76460702.pc_relevant_aa2&depth_…

C语言——数据在内存中的存储

C语言——数据的存储一、C语言中常见的数据类型1.1 数据类型的基本归类二. 整型在内存中的存储2.1原码、反码、补码知识回顾2.2大小端字节序2.2.1 何为大小端字节序2.2.2为什么会产生大小端字节序2.2.3练习三、浮点型在内存中的存储3.1浮点数存储规则一、C语言中常见的数据类型…

Java两大工具库:Commons和Guava(5)

您好&#xff0c;我是湘王&#xff0c;这是我的CSDN博客。值此新春佳节&#xff0c;我给您拜年啦&#xff5e;祝您在新的一年中所求皆所愿&#xff0c;所行皆坦途&#xff0c;展宏“兔”&#xff0c;有钱“兔”&#xff0c;多喜乐&#xff0c;常安宁&#xff01;在开发中&#…

C语言最鸡肋的关键字

C语言的关键字有很多&#xff0c;要说最不常见的&#xff0c;我觉得应该是auto。 说它不常见&#xff0c;因为很多时候&#xff0c;我们都把它给省略了。 比如在函数内部定义变量a&#xff0c;正常人都会这样写&#xff1a; void function() {int a; }很少有人会在前面加一个…

1616_MIT 6.828 program header相关只是小结

全部学习汇总&#xff1a; GreyZhang/g_unix: some basic learning about unix operating system. (github.com) 前面刚刚分析了elf的header&#xff0c;看了两行代码又遇到了program的header。又是一个概念类的问题&#xff0c;还得去简单了解下。 1. 这里面的信息其实是可能有…