目录
列表项排序设计:分数索引思想与实践

列表项排序设计:分数索引思想与实践

以「课程章节排序」为例,说明一种无需大规模重写序号的列表排序方案:通过「前驱 ord + 后继 ord」在数轴上插空,用**分数索引(Fractional Indexing)**的思想实现 O(1) 插入与拖拽排序,并给出创建/更新时的业务规则、边界处理以及可复用的业务场景。

前端效果:


一、排序的核心思想

1.1 问题:为什么不用 1、2、3、4…

若列表顺序用连续整数 1、2、3、4… 表示:

  • 插入:在 2 和 3 之间插入一条,需要把 3、4、5… 全部 +1,更新量大,且高并发下易冲突。
  • 拖拽:把第 5 条拖到第 2 条前面,同样要批量更新后面所有项的序号。

当列表很长、或排序操作频繁时,这种「连续整数序号」方案会带来大量写操作和锁竞争。

1.2 思路:在数轴上「插空」

不追求序号连续,而是把每一项对应到实数轴上的一个点,顺序即数轴上的先后关系:

  • A 与 B 之间插入:新项的 ord 取 A 与 B 的 ord 的中间值,例如 ord = (ordA + ordB) / 2
  • 某一条之后插入:新项的 ord = 前一条的 ord + 一个固定步长,例如 ord = preOrd + 1000
  • 放到最后:新项的 ord = 当前最大 ord + 步长
  • 放到最前:新项的 ord = 0(或约定一个最小基准)。

这样:

  • 插入/移动通常只改当前这一条的 ord,不需要批量更新其他项。
  • 顺序由「数轴上的大小关系」决定,与是否连续整数无关。

这就是 Fractional Indexing(分数索引) 的简化实现:用稀疏的、可再分的数值表示顺序,通过「取中间值」在任意两点之间插入,避免全局重排。

1.3 本方案中的具体约定

  • 使用整数 ord,顺序按 ORDER BY ord ASC
  • 中间插入ord = (preOrd + nextOrd) / 2(整数除法)。
  • 前驱后插入ord = preOrd + ORD_STEP(如 1000)。
  • 追加到末尾ord = max(ord) + ORD_STEP;无任何项时 ord = ORD_STEP
  • 置顶ord = 0

步长 ORD_STEP 抽成配置常量(如 1000),既保证新项与已有项不重叠,又为将来在两点之间多次插入留出空间(多次取半后可能需要在某次触发「重算序号」或扩大数值域,这是后续优化点)。


二、创建时的业务规则(新增章节)

创建时不直接传 ord,而是通过前驱/后继的 ord 表达「插在哪里」,由后端统一计算新 ord。

入参 含义 计算方式
preOrd + nextOrd 都有 插在两者之间 ord = (preOrd + nextOrd) / 2
preOrd 插在前一条后面 ord = preOrd + ORD_STEP
都不传 追加到末尾 ord = max(当前课程下 ord) + ORD_STEP,空列表时为 ORD_STEP

业务要点:

  • 新增接口只接受 courseIdtitle、可选的 preOrd/nextOrd 等,不暴露「直接写 ord」,避免前端传错导致顺序错乱。
  • 同一课程下的 ord 只在「当前课程」内取 max,不同课程互不影响。
  • 使用分布式锁(如按管理员 + 课程维度)保证同一课程下的创建/排序并发安全。

三、更新时的业务规则(拖拽/改顺序)

更新时既支持「按位置语义」的排序,也支持直接改 ord(如后台批处理)。

入参 含义 计算方式
preOrd + nextOrd 都有 移到两者之间 ord = (preOrd + nextOrd) / 2
preOrd 移到前一条后面 ord = preOrd + ORD_STEP
nextOrd(preOrd 为空) 移到第一位(置顶) ord = 0
ord 直接指定序号 使用传入的ord
都不传 不改顺序 不更新 ord 字段

与创建的差异:

  • 仅传 nextOrd:表示「把当前项放到 nextOrd 对应的那条前面」,即置顶,因此 ord = 0。这样前端把列表最后一条拖到最前时,只需传 nextOrd = 当前第一条的 ord,无需 preOrd。
  • 支持直接传 ord,便于管理端或脚本批量调整顺序。

四、边界与实现细节

4.1 空列表与首项

  • 创建且不传 preOrd/nextOrdmax(ord) 无值,则 ord = 0 + ORD_STEP,即第一条为 1000。
  • 更新且仅传 nextOrd:置顶,ord = 0。若原来就有 ord=0 的项,列表里会出现两个 0,顺序按 id 或更新时间等次要字段区分;也可在「仅 nextOrd」时改为 min(ord)-1 或做一次轻量重排,视产品需求而定。

4.2 整数取半与精度

  • (preOrd + nextOrd) / 2 为整数除法。若 preOrd=100, nextOrd=102,得到 101;若 100 与 101 之间再插,得到 100,与 100 重复。此时需要:
    • 要么接受「相等时按 id 等次要字段排序」;
    • 要么在检测到 ord 冲突或间距过小时触发局部或全局重算(例如按当前顺序重新赋 1000、2000、3000…),避免长期取半导致精度用尽。

4.3 步长常量

  • ORD_STEP(如 1000)抽成配置常量,便于:
    • 统一「前驱后插入」与「追加到末尾」的步长;
    • 未来若列表极长或插入极频繁,可调大步长或配合重算策略。

4.4 并发与锁

  • 创建/更新顺序时对「同一课程」或「同一管理员+课程」加锁,避免并发生成相同 ord 或 max(ord) 读不准。

4.5 日志与可观测性

  • 在计算 ord 的分支打 debug 日志(preOrd、nextOrd、计算得到的 ord),便于排查顺序错乱和边界问题。

五、适用场景与复用建议

同一套「前驱 ord + 后继 ord + 步长」的思想,可用于所有用户可拖拽排序、且希望避免大规模更新的列表场景,例如:

场景 说明
课程章节/目录 本文实现:按 preOrd/nextOrd 插入与移动,置顶用 nextOrd 且 ord=0。
导航/菜单排序 菜单项、分类的展示顺序,支持拖拽调整。
看板列顺序 看板(Kanban)各列的顺序,或同一列内卡片的顺序。
待办/任务列表 任务顺序、子任务顺序。
相册/媒体顺序 相册内照片、播放列表内条目。
表单题目顺序 问卷、表单的题目顺序。
商品/规格顺序 商品列表、SKU 或规格的展示顺序。

复用要点:

  • 表结构:列表项表有 ord(或 sort_order)字段,同一「父」维度内排序(如 courseId、menuId、boardId)。
  • 创建:支持可选的 preOrd/nextOrd,不传则追加到末尾(max(ord)+STEP)。
  • 更新:支持 preOrd、nextOrd、以及「仅 nextOrd 表示置顶」的语义;可选保留直接传 ord 的能力。
  • 步长:统一常量,便于调优与重算。
  • 边界:明确空列表、置顶、取半冲突时的策略(接受相等或触发重算)。

六、小结与启发

  • 排序的本质是「在有序数轴上找一个位置」;用稀疏的、可插空的 ord,而不是连续整数,能把插入/移动的代价从 O(n) 降为 O(1),并减少锁竞争。
  • 创建时用「前驱/后继」表达意图,更新时同样用 preOrd/nextOrd 表达「移到哪两个之间」或「放到前一个后面/放到最前」,接口语义清晰,前端拖拽只需传前后项的 ord。
  • 边界要显式约定:空列表、置顶、取半导致相等时的处理,以及步长与必要时重算序号。
  • 步长与规则抽成常量和独立方法,便于在不同业务(章节、菜单、看板等)中复用,形成可积累的「列表排序」技术方案。
"我视别人的钱财如粪土,但你的就不一样啦!"
本文由 程序员拉大锯 原创发布于 阳光沙滩 , 未经作者授权,禁止转载
评论
0 / 1024
推荐文章
面向 Java 程序员的 MinIO 入门教程
本文为Java程序员提供了一份详细的MinIO入门教程,涵盖MinIO的部署方法和Java SDK的集成使用。通过本文,您将学习如何在Java项目中高效管理桶和对象,快速上手MinIO这一高性能对象存储服务。
你知道:气和汽的区别吗?
了解‘气’和‘汽’的区别,掌握它们在不同语境下的含义与用法,帮助你更准确地使用中文。无论是日常交流还是写作,这对提升语言能力都大有裨益。
wsl update 下载不下来怎么办呀?
遇到Docker Desktop提示需要更新但无法解决?本文教你如何通过GitHub下载并安装WSL,轻松解决更新问题,适合使用x64芯片的用户。
今日经验:重置虚拟机的密码
本文详细记录了在KVM虚拟化环境中,如何通过virt-rescue工具重置遗忘的root密码。对于需要维护和管理虚拟机的IT人员来说,这是一份实用的排障指南,涵盖了从环境准备到密码修改的完整流程,帮助快速恢复系统访问权限。
今日工作:Android Health Connect 接入记录
本文详细讲解了如何将 Android Health Connect 接入到健康或运动类应用中,涵盖从配置、代码实现到测试验收的完整流程。适合希望统一健康数据管理、提升用户隐私合规性的开发者阅读。
Skill从入门到出家
探索AI Agent的核心能力——Skill,了解其模块化设计、渐进式披露机制和实际应用场景。从基础概念到高级实战,掌握如何构建可复用、可移植的AI技能,提升Agent处理复杂任务的能力。
Docker,Docker Compose,kubectl最近遇到的版本问题
本文分享了在使用Docker、Docker Compose和kubectl时遇到的版本问题及解决方法,适合需要更新或管理Linux系统中相关工具的开发者参考。
Google上架App退回
Google Play Console 抛出 16KB 内存页面大小合规性错误,导致应用无法上架。本文详细分析了错误原因,并提供了解决方案,帮助开发者适配 Android 15 的新要求。
国内常用的 npm 镜像源整理
在使用 npm 安装依赖时,国内开发者常常遇到速度慢的问题。本文整理了多个稳定且常用的国内 npm 镜像源,帮助提升依赖安装效率。还介绍了如何通过 nrm 工具快速切换镜像,非常适合需要优化开发环境的开发者。
列表项排序设计:分数索引思想与实践
本文介绍了分数索引思想在列表排序中的应用,通过实数轴上的插空方式实现高效插入与拖拽排序。适用于课程章节、导航菜单、看板列等多种场景,提供创建和更新时的业务规则及边界处理策略,帮助开发者优化排序性能并提升用户体验。
2026苹果电脑芯片的性能排行榜
了解2026年前后苹果电脑芯片的性能排名和关键变化,帮助你更好地选择适合自己的设备。从M1到M5,每一款芯片都有其独特优势,无论是日常办公还是专业需求都能找到合适的推荐。
在 KVM 上部署 Ubuntu 24.04 Server:企业级虚拟化完整实践指南
本文详细介绍了如何在 KVM 上部署 Ubuntu 24.04 Server,涵盖系统架构、部署步骤、核心命令解析和性能优化等内容。适合希望构建高性能、低成本企业虚拟化平台的技术人员阅读。
旧版本的kubesphere还能用的
本文介绍了如何使用KubeSphere配置和部署一个Kubernetes集群,包括修改配置文件、设置环境变量以及执行安装命令。适合需要了解Kubernetes集群搭建的开发者和系统管理员阅读。
Flutter Fragment 嵌入模式下返回键/侧滑直接退出应用-日常记录
在 Flutter 混合开发中,如何解决 Android 返回键无法正确触发 Flutter 页面返回逻辑的问题?本文详细解析了事件传递机制,并提供完整解决方案,包括 Android 和 Flutter 层的配置方式。适合开发者快速排查和修复类似问题。
记录一个问题,Post请求变Get请求了?原因很简单
本文讲述了一个关于HTTP请求方法被错误转换的问题,通过分析Nginx配置和日志,最终发现是由于301重定向导致POST请求变成GET请求。作者详细描述了问题排查过程,并给出了解决方案,对于开发人员在处理类似网络问题时具有参考价值。
Android 进阶:在非 ComponentActivity 中实现协程自动取消
本文深入探讨了在 Android 开发中如何为非 ComponentActivity 的类实现 LifecycleOwner 功能,分享了两种优雅的解决方案。通过手动注册和 Kotlin 属性委托的方式,确保协程能随 Activity 销毁而自动取消,提升代码健壮性。适合对 Android 生命周期管理和协程使用感兴趣的开发者阅读。
Uni-app 发送通知全解析(从本地通知到推送服务实战)
深入了解Uni-app中通知的四种类型及应用场景,掌握跨平台支持情况和代码实战技巧。从基础提示到高级推送架构设计,全面解析如何提升用户活跃度和业务触达能力,适合开发者系统学习与实践。
MQTT 学习指南:从入门到工程实践
深入解析MQTT在物联网中的核心作用,了解其轻量、稳定和省流量的特性。从发布/订阅模型到QoS机制,全面掌握MQTT的通信原理与应用场景,助你高效构建物联网系统。
深入理解 Java NIO:非阻塞 I/O 的原理、应用与案例实战
本文深入解析 Java NIO 的核心概念,包括 Buffer、Channel 和 Selector,对比传统 I/O 的优势,并通过实战案例展示如何构建高并发的聊天服务器。适合希望提升 Java 网络编程能力的开发者阅读。
二次函数全攻略:公式、图像、性质与应用
二次函数是数学中的核心内容,广泛应用于物理、经济和工程领域。本文系统讲解其定义、图像、性质及实际应用,通过案例帮助理解,适合学生和数学爱好者深入学习。
Java排序算法全解析:从冒泡到快速排序,案例实操
掌握Java中常见排序算法的实现方法,理解其时间复杂度和适用场景,提升编程能力。从冒泡排序到快速排序,全面解析各种排序技术,帮助你高效解决问题。
Java 面试算法题全解析:案例、讲解与面试频率分析
Java面试中,算法题是必考内容。掌握常见算法不仅能提升编程能力,还能提高通过率。本文详细讲解了两数之和、链表反转、二分查找等经典算法题的解题思路与实现代码,帮助你更有针对性地准备面试。
Java基础语法入门:跟着案例一步步学会
掌握Java基础语法,从零开始学习编程。本文通过实战案例,详细讲解程序结构、变量、运算符、条件语句、循环、数组和方法等核心内容,适合初学者系统学习。通过练习提升编程能力,为进阶开发打下坚实基础。
Java 多线程入门:从概念到实战(初学者必看)
掌握多线程编程,提升程序效率和响应速度。本文从基础概念到实战案例,一步步讲解如何在 Java 中实现多线程,帮助你快速上手并理解核心知识点。通过实际代码示例,了解线程的生命周期、常用方法及线程安全问题,适合初学者和进阶者学习参考。
Java 多线程入门:从概念到实战(初学者必看)
掌握多线程编程,提升程序效率与响应速度。本文从基础概念到实战案例,详细讲解Java多线程的实现方式、注意事项和线程安全问题,适合初学者快速上手。通过实例代码,帮助你理解如何高效编写多线程程序。
Flutter开发网路库封装示例
本教程详细介绍了如何在 Flutter 项目中通过 Dio 库封装网络请求,提供了一套完整的封装类及使用示例,适用于构建中大型项目。内容覆盖依赖引入、封装方法实现、使用场景演示以及扩展功能建议,帮助开发者快速提升代码复用性和可维护性。无论是初学者还是有经验的开发者,都能从中受益。
写完这些案例,就掌握flutter开发了。
学习如何在Flutter中实现输入框的实时显示、页面跳转与数据传递,以及动态列表的展示和网络请求数据的解析与展示。通过这些基础案例,掌握Flutter的核心功能,为构建复杂应用打下坚实基础。
Flutter学习路线
想要掌握Flutter开发?这篇详细的学习路线图将帮助你从零开始,逐步成长为Flutter开发者。覆盖基础入门、核心概念、进阶开发以及发布优化四个阶段,结合理论与实践,让你轻松掌握Flutter技能。无论是想开发手机应用还是探索跨平台开发,这份指南都能满足你的需求。立即行动,开启你的Flutter之旅吧!
什么是MCP? Monte Carlo Planning(蒙特卡洛规划)
MCP(Monte Carlo Planning,蒙特卡洛规划)是强化学习和决策系统中的重要方法,广泛应用于复杂环境下的行动策略规划。无论是博弈中的AI,还是机器人路径规划,MCP都能通过随机模拟预测未来策略的效果。其中,蒙特卡洛树搜索(MCTS)是其典型实现,具有强大的全局最优性和适应高维复杂策略的能力。文章详细解析了MCP的基本概念、与强化学习的关系、典型算法以及实际应用场景,展示了其在AlphaGo、自动驾驶、游戏AI等领域的卓越表现。
智能体相关的概念介绍一下,并且给出学习路线!
智能体是人工智能领域的重要概念,广泛应用于强化学习、多智能体系统和机器人学等方向。本文从智能体的基本概念出发,介绍了其核心组成和分类,并提供了涵盖基础知识、模型理解、实践项目及前沿研究的系统学习路线。无论是初学者还是希望深入探索的研究者,都能从中找到有价值的信息和资源。