当前位置: 首页 > article >正文

【LC插件开发】基于Java实现FSRS(自由间隔重复调度算法)

在这里插入图片描述

😊你好,我是小航,一个正在变秃、变强的文艺倾年。
🔔本文讲解【LC插件开发】基于Java实现FSRS(自由间隔重复调度算法),期待与你一同探索、学习、进步,一起卷起来叭!

目录

  • 一、FSRS
  • 二、算法实现
    • 卡片状态
      • 状态更新
    • 评分等级
    • 算法结果返回
    • 算法逻辑
      • 核心参数配置
      • New状态
      • REVIEW状态
      • LEARNING状态
      • 公式讲解
        • 下次复习间隔计算
        • 难度更新公式
    • 示例场景
      • 用户首次学习新卡片
      • 用户回答"GOOD"后

一、FSRS

FSRS(Frequent Spaced Repetition System)是一种基于动态间隔重复的记忆算法,本算法基于 SuperMemo 作者 Piotr Wozniak 提出的 DSR 模型开发。该模型考虑了影响记忆的三个变量:难度(difficulty),稳定性(stability)和可提取性(retrievability)。

  • 稳定性指的是记忆的存储强度,越高,记忆遗忘得越慢。
  • 可提取性指的是记忆的检索强度,越低,记忆遗忘的概率越高。

本文重点从下面三个角度进行复现:

  1. ​使用稳定性(stability)难度(difficulty)两个参数动态调整复习策略
  2. ​定义4种评分等级(AGAIN/GOOD/HARD/EASY)触发不同的参数更新规则
    3.基于指数衰减模型和机器学习参数优化复习间隔

在本模型中,考虑了以下记忆规律:

  • 记忆材料越难,记忆稳定性增长越慢
  • 记忆稳定性越高,记忆稳定性增长越慢(又称为记忆稳定化衰减)
  • 记忆可提取性越低,记忆稳定性增长越快(又称为记忆稳定化曲线)

二、算法实现

卡片状态

public enum FSRSState {/*** 从未学习过*/NEW(0),/*** 刚刚第一次学习*/LEARNING(1),/*** 完成LEARNING状态*/REVIEW(2),/*** 在REVIEW状态时忘记*/RELEARNING(3);
}

状态更新

在这里插入图片描述

public void updateState() {if (this.state == FSRSState.NEW) {// NEW → Learning阶段初始化this.ratingToCard.get(FSRSRating.AGAIN).setState(FSRSState.LEARNING);this.ratingToCard.get(FSRSRating.HARD).setState(FSRSState.LEARNING);this.ratingToCard.get(FSRSRating.GOOD).setState(FSRSState.LEARNING);this.ratingToCard.get(FSRSRating.EASY).setState(FSRSState.REVIEW);} else if(this.state == FSRSState.LEARNING || this.state == FSRSState.RELEARNING) {// Learning → Review阶段转换this.ratingToCard.get(FSRSRating.AGAIN).setState(this.state);this.ratingToCard.get(FSRSRating.HARD).setState(FSRSState.REVIEW);this.ratingToCard.get(FSRSRating.GOOD).setState(FSRSState.REVIEW);this.ratingToCard.get(FSRSRating.EASY).setState(FSRSState.REVIEW);} else if(this.state == FSRSState.REVIEW) {// Review → Relearning触发条件this.ratingToCard.get(FSRSRating.AGAIN).setState(FSRSState.RELEARNING);// 其他评分保持Review状态}
}

评分等级

public enum FSRSRating {/*** 忘记;错误答案*/AGAIN(0),/*** 回忆起来;经过一定困难才答出的正确答案*/HARD(1),/*** 经过延迟答出的正确答案*/GOOD(2),/*** 完美答案*/EASY(3);
}

算法结果返回

 private int interval;private long dueTime, lastReview;private float stability, difficulty;private int elapsedDays, repetitions;private FSRSState state;
字段名称数据类型描述
intervalint计划复习间隔(天)
dueTimelong下次复习截止时间戳(毫秒)
lastReviewlong上次复习时间戳(毫秒)
stabilityfloat内容稳定性(0.1-∞)
difficultyfloat内容难度(1-10)
elapsedDaysint自上次复习经过的天数
repetitionsint已完成的复习次数
state枚举当前卡片状态(NEW/LEARNING/REVIEW等)

算法逻辑

算法整体流程如下:

在这里插入图片描述

核心参数配置

// 算法核心参数配置
private final float REQUEST_RETENTION = 0.9F;   // 目标记忆保留率(90%)
private final int MAXIMUM_INTERVAL = 36500;    // 最大复习间隔(365天)
private final float EASY_BONUS = 1.3F;        // 容易卡片间隔奖励系数
private final float HARD_FACTOR = 1.2F;         // 困难卡片难度系数
private final float[] WEIGHTS = new float[]{1F, 1F, 5F, -1F, -1F, 0.1F, 1.5F, -0.2F, 0.8F, 2, -0.2F, 0.2F, 1F}; /* 各参数权重配置 */

New状态

this.init(); // 初始化难度/稳定性
this.card.setElapsedDays(0);// 设置首次复习时间
this.card.getRatingToCard().get(FSRSRating.AGAIN).setDueTime(... + 1分钟);
this.card.getRatingToCard().get(FSRSRating.GOOD).setDueTime(... + 10分钟);
this.card.getRatingToCard().get(FSRSRating.HARD).setDueTime(... + 15分钟);
  • 难度:1.0 ~ 10.0(均值5.0)
  • 稳定性:0.1 ~ 0.5(均值0.3)

REVIEW状态

float retrievability = (float) Math.exp(Math.log(0.9) * interval / lastStability);
this.next(lastDifficulty, lastStability, retrievability); // 动态调整参数

LEARNING状态

hardInterval = nextInterval(...); 
goodInterval = Math.max(nextInterval(...), hardInterval + 1);
easyInterval = Math.max(nextInterval(...), goodInterval + 1);
this.card.schedule(hardInterval, goodInterval, easyInterval);
  • 间隔梯度:HARD < GOOD < EASY

公式讲解

下次复习间隔计算

i n t e r v a l = stability × ln ⁡ ( 0.9 ) ln ⁡ ( 1 R ) interval = \frac{\text{stability} \times \ln(0.9)}{\ln\left(\frac{1}{R}\right)} interval=ln(R1)stability×ln(0.9)

代码实现:

interval = stability * log(0.9) / log(1/retention_rate)
难度更新公式

Δ d = W 4 × ( r − 2 ) new_difficulty = min ⁡ ( max ⁡ ( μ ( W 2 , current_difficulty ) + Δ d , 1 ) , 100 ) \Delta d = W_4 \times (r - 2) \\ \text{new\_difficulty} = \min\left(\max\left(\mu(W_2, \text{current\_difficulty}) + \Delta d, \quad 1\right), \quad 100\right) Δd=W4×(r2)new_difficulty=min(max(μ(W2,current_difficulty)+Δd,1),100)
其中均值回归:
μ ( a , b ) = W 5 × a + ( 1 − W 5 ) × b \mu(a, b) = W_5 \times a + (1 - W_5) \times b μ(a,b)=W5×a+(1W5)×b

代码实现:

private float nextDifficulty(float difficulty, int retrievability) {return Math.min(Math.max(meanReversion(WEIGHTS[2], difficulty + WEIGHTS[4]*(retrievability-2)), 1), 10);
}

示例场景

用户首次学习新卡片

在这里插入图片描述

用户回答"GOOD"后

在这里插入图片描述

📌 [ 笔者 ]   文艺倾年
📃 [ 更新 ]   2025.3.23
❌ [ 勘误 ]   /* 暂无 */
📜 [ 声明 ]   由于作者水平有限,本文有错误和不准确之处在所难免,本人也很想知道这些错误,恳望读者批评指正!

在这里插入图片描述

相关文章:

【LC插件开发】基于Java实现FSRS(自由间隔重复调度算法)

&#x1f60a;你好&#xff0c;我是小航&#xff0c;一个正在变秃、变强的文艺倾年。 &#x1f514;本文讲解【LC插件开发】基于Java实现FSRS&#xff08;自由间隔重复调度算法&#xff09;&#xff0c;期待与你一同探索、学习、进步&#xff0c;一起卷起来叭&#xff01; 目录…...

java后端接收数组,数组长度超256个就会报错

1.原因 DataBinder 中默认限制了list最大只能增长到256。 2.解决方案 1.在BaseController添加InitBinder方法&#xff0c;其余继承BaseController InitBinder //类初始化是调用的方法注解public void initBinder(WebDataBinder binder) {//给这个controller配置接收list的长…...

第45章:配置更新与应用热重载策略

第45章:配置更新与应用热重载策略 作者:DogDog_Shuai 阅读时间:约25分钟 难度:中级 目录 1. 引言2. 配置更新挑战3. Kubernetes原生配置更新机制4. 应用热重载技术5. 配置更新最佳实践...

数据库MVCC详解

MVCC 1.基本介绍 数据库&#xff1a;MySQL。【很多主流数据库都使用了MVCC&#xff0c;比如MySQL的InnoDB引擎、PostgreSQL、Oracle】 MVCC&#xff0c;全称Multi-Version Concurrency Control&#xff0c;即多版本并发控制。是数据库管理系统中的一种并发控制方法。 MVCC的…...

MySQL数据库基础篇

目录 SQL的分类 数据定义语言&#xff08;DDL&#xff09;---Data Definition Language 数据操作语言(DML) ---Data Manipulation Language 数据查询语言(DQL) ---Data Query Language 数据控制语言(DCL) ---Data Control Language 事务控制语言(TCL) --- Transaction Cont…...

Rust函数、条件语句、循环

文章目录 函数**语句与表达式**条件语句循环 函数 Rust的函数基本形式是这样的 fn a_func(a: i32) -> i32 {}函数名是蛇形风格&#xff0c;rust不在意函数的声明顺序&#xff0c;只需要有声明即可 函数参数必须声明参数名称和类型 语句与表达式 这是rust非常重要的基础…...

AI比人脑更强,因为被植入思维模型【17】万物联系思维模型

万物联系,万物,并不孤立。 定义 万物联系思维模型是一种强调世界上所有事物都相互关联、相互影响的思维方式。它认为任何事物都不是孤立存在的,而是与周围的环境、其他事物以及整个宇宙构成一个有机的整体。这种联系不仅包括直接的因果关系,还涵盖了间接的、潜在的、动态的…...

Android Compose 约束布局(ConstraintLayout、Modifier.constrainAs)源码深度剖析(十二)

Android Compose 约束布局&#xff08;ConstraintLayout、Modifier.constrainAs&#xff09;源码深度剖析 一、引言 在 Android 开发中&#xff0c;布局是构建用户界面的基础。随着 Android 开发技术的不断发展&#xff0c;Jetpack Compose 作为一种全新的声明式 UI 框架应运…...

【MySQL篇】复合查询

目录 前言&#xff1a; 1&#xff0c;多表查询 2&#xff0c;自连接 3&#xff0c;子查询 3.1&#xff0c;单行子查询 3.2&#xff0c;多行子查询 3.3&#xff0c;多列子查询 3.3&#xff0c;在from子句中使用子查询 4&#xff0c;合并查询 4.1&#xff0c;union …...

点亮STM32最小系统板LED灯

对于如何点亮板载LED灯只需要掌握如何初始化GPIO引脚&#xff0c;并改变GPIO引脚的电平即可实现点亮或者熄灭LED。 Led_INFO led_info {0}; led_info 是一个结构体变量&#xff0c;类型为 Led_INFO&#xff0c;用于存储LED的状态信息。这里初始化为 {0}&#xff0c;表示所有成…...

unsloth微调QwQ32B(4bit)

unsloth微调QwQ32B(4bit) GPU: 3090 24G unsloth安装部署 pip 安装 pip install unsloth --index https://pypi.mirrors.usrc.edu.cn/simplesource /etc/network_turbopip install --force-reinstall --no-cache-dir --no-deps githttps://github.com/unslothai/unsloth.git​…...

基于腾讯云大模型知识引擎×DeepSeek的高等职业学校单独招生二级学院考前咨询系统

1、主要思路 通过大模型知识引擎DeepSeek搭建高等职业学校单独招生二级学院考前咨询专有问答&#xff0c;使得专业老师能够更好的服务考试学生&#xff0c;有利于二级学院能够更好的进行考试宣传&#xff0c;招来优秀学子&#xff01; 2、创作过程 2.1、本地部署大模型的缺陷…...

【Linux】线程库

一、线程库管理 tid其实是一个地址 void* start(void* args) {const char* name (const char *)args;while(true){printf("我是新线程 %s &#xff0c;我的地址&#xff1a;0x%lx\n",name,pthread_self());sleep(1);}return nullptr; }int main() {pthread_t tid…...

数组作为哈希表的妙用:寻找缺失的第一个正数

数组作为哈希表的妙用&#xff1a;寻找缺失的第一个正数 大家好&#xff0c;我是Echo_Wish&#xff0c;今天我们来探讨一个经典的算法问题——“缺失的第一个正数”。听起来可能有点简单&#xff0c;但它实际上是一个非常有意思且富有挑战性的题目&#xff0c;在面试中常常会碰…...

物化视图详解:数据库性能优化的利器

物化视图&#xff08;Materialized View&#xff09;作为数据库性能优化的核心手段&#xff0c;通过预计算和存储查询结果&#xff0c;显著提升了复杂查询的效率。本文将深入剖析物化视图的工作原理、应用场景及最佳实践&#xff0c;帮助企业在合适的场景中充分发挥其性能优势。…...

【C++】类和对象(匿名对象)

匿名对象 用 类型(实参) 定义出来的对象叫做匿名对象&#xff0c;相比之前我们定义的 类型 对象名(实参) 定义出来叫有名对象匿名对象生命周期只在当前一行&#xff0c;一般临时定义一个对象当前用一下即可&#xff0c;就可以定义匿名对象。 class A { public:A(int a 0):_a…...

一文读懂 GPT 与 BERT:预训练逻辑及差异剖析

在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;预训练语言模型GPT&#xff08;Generative Pretrained Transformer&#xff09;和 BERT&#xff08;Bidirectional Encoder Representations from Transformers&#xff09;作为杰出代表&#xff0c;备受关注。本文将…...

【算法】十大排序算法(含时间复杂度、核心思想)

以下是 **十大经典排序算法** 的时间复杂度、空间复杂度及稳定性总结&#xff0c;适用于面试快速回顾&#xff1a;排序算法对比表 排序算法最佳时间复杂度平均时间复杂度最差时间复杂度空间复杂度稳定性核心思想冒泡排序O(n)O(n)O(n)O(1)稳定相邻元素交换&#xff0c;大数沉底…...

渐进式滑坡多场信息演化特征与数据挖掘研究

标题:渐进式滑坡多场信息演化特征与数据挖掘研究 内容:1.摘要 摘要&#xff1a;在地质灾害频发的背景下&#xff0c;研究渐进式滑坡多场信息演化特征与数据挖掘具有重要的实际意义。本研究旨在深入探究渐进式滑坡在不同阶段的多场信息&#xff08;如应力场、位移场、渗流场等&…...

蓝桥杯备考-》单词接龙

很明显&#xff0c;这道题是可以用DFS来做的&#xff0c;我们直接暴力搜索&#xff0c;但是这里有很多点是我们需要注意的。 1.我们如何确定两个单词能接上&#xff1f; 比如touch和choose 应该合成为touchoose 就是这样两个单词&#xff0c;我们让一个指针指着第一个字符串…...

解锁C++模板参数:开启泛型编程新世界

目录 C++ 模板:编程世界的瑞士军刀 一、模板参数初相识 1.1 类型参数 1.2 非类型参数 1.3 模板模板参数 二、模板参数推导大揭秘 2.1 推导规则深度剖析 2.2 推导成功场景展示 2.3 推导失败场景解析 三、模板参数实战应用 3.1 通用算法实现 3.2 容器类设计 3.3 元…...

计算机视觉yolov8模型应用-学习笔记

计算机视觉yolov8模型应用-学习笔记 YOLOv8是由Ultralytics公司在‌2023年1月10日‌发布的一款深度学习模型。它是YOLOv5的重大更新版本&#xff0c;支持图像分类、物体检测和实例分割任务。这一版本在发布前就受到了广泛关注&#xff0c;并在发布后迅速成为目标检测领域的热门…...

【网络层协议】NAT技术内网穿透

IP地址数量限制 我们知道&#xff0c;IP地址&#xff08;IPv4&#xff09;是一个4字节32位的整数&#xff0c;那么一共只有2^32也就是接近43亿个IP地址&#xff0c;而TCP/IP协议栈规定&#xff0c;每台主机只能有一个IP地址&#xff0c;这就意味着&#xff0c;一共只有不到43亿…...

SQL中的索引是什么

在 SQL 中&#xff0c;索引&#xff08;Index&#xff09; 是一种用于加速数据检索的数据库对象&#xff0c;通过建立特定的数据结构&#xff08;如 B树、哈希表等&#xff09;&#xff0c;帮助数据库系统快速定位目标数据。以下是关于索引的详细分类、工作原理、使用场景和最佳…...

TensorFlow面试题及参考答案

目录 什么是 TensorFlow 的计算图?详细描述 TensorFlow 计算图的组成结构(节点、边、会话) 它与动态图(Eager Execution)的区别是什么?TensorFlow 静态计算图与动态图(Eager Execution)的区别及适用场景是什么? 解释张量(Tensor)的概念及其在 TensorFlow 中的作用…...

go-zero学习笔记

内容不多&#xff0c;只有部分笔记&#xff0c;剩下的没有继续学下去&#xff0c;包括路由与处理器、日志中间件、请求上下文 文章目录 1、go-zero核心库1.1 路由与处理器1.2 日志中间件1.3 请求上下文 1、go-zero核心库 1.1 路由与处理器 package mainimport ("github…...

在Ubuntu 22.04 中安装Docker的详细指南

这里写目录标题 前言一、安装 Docker1. 卸载旧版本&#xff08;如有&#xff09;2. 更新系统并安装依赖工具3. 添加 Docker 官方 GPG 密钥4. 设置 Docker 仓库5. 安装 Docker Engine6. 验证安装 二、配置 Docker 镜像加速1. 修改 Docker 配置文件2. 重启 Docker 服务3. 验证加速…...

十亿级流量削峰实战:LinkedBlockingQueue缓冲池的工程化实现

《十亿级流量削峰实战&#xff1a;LinkedBlockingQueue缓冲池的工程化实现》 本文将以电商秒杀系统为背景&#xff0c;深度解析如何通过LinkedBlockingQueue构建百万QPS级异步缓冲系统&#xff0c;包含容量计算模型、拒绝策略选择、监控埋点方案等完整实施细节&#xff0c;并提…...

深入理解 C++11 智能指针:独占、共享与弱引用的完美管理

文章目录 std::unique_ptr&#xff08;独占式智能指针&#xff09;std::shared_ptr&#xff08;共享式智能指针&#xff09;std::weak_ptr&#xff08;弱引用智能指针&#xff09;示例展示&#xff1a;智能指针的原理内存泄漏**什么是内存泄漏&#xff0c;内存泄漏的危害****如…...

AI Agent开发大全第四课-提示语工程:从简单命令到AI对话的“魔法”公式

什么是提示语工程?一个让AI“听话”的秘密 如果你曾经尝试过用ChatGPT或者其他大语言模型完成任务,那么你一定遇到过这样的情况:明明你的问题是清晰的,但答案却离题万里;或者你认为自己提供的信息足够详尽,可结果还是不理想。问题出在哪?很多时候并不是因为AI不够聪明,…...