
本文介绍一种高效算法,将连续评分范围(如1–10)按比例、非均匀地划分为若干子区间,并一一对应到长度可变的消息数组(如['bad', 'okay', 'good']),确保前序区间优先多占一个评分点,实现自然、公平的语义映射。
在构建评分型交互组件(如星级评价、情绪滑块)时,常需将数值型评分(如 currentRating = 4)映射为语义化提示(如 'Okay')。难点在于:消息数组长度可变(messagesArray.length ≤ ratingRange),且评分区间必须完整覆盖 [1, ratingRange],不可重叠、不可遗漏。简单均分(如 Math.floor((currentRating - 1) / Math.ceil(ratingRange / messagesArray.length)))会导致末尾区间被压缩甚至丢失,而本方案采用“前大后小”的自适应分段策略——优先让靠前的消息项分配更多评分点,使映射更符合用户直觉(例如低分段容错更强,“1–3分都算Bad”比“1–2分Bad,3分突变为Okay”更合理)。
核心思想是:
- 计算 ratingRange 除以 messagesArray.length 的余数 numLargeGroups,即需要「多分配1个评分点」的前 numLargeGroups 个消息;
- 剩余消息则分配基础大小 smallSize = (ratingRange - numLargeGroups) / messagesArray.length(必为整数);
- 所有「大组」总跨度为 split = numLargeGroups * (smallSize + 1);
- 对 currentRating(先转为0-based索引)判断其落在 split 前还是后,再分别用不同步长计算索引。
以下是完整、健壮的 TypeScript 实现:
function getMessageByRating(
ratingRange: number,
messagesArray: string[],
currentRating: number
): string {
// 边界防护:空数组、无效评分、超限评分
if (messagesArray.length === 0) return '';
if (currentRating < 1 || currentRating > ratingRange) {
throw new Error(`currentRating must be between 1 and ${ratingRange}`);
}
const numLargeGroups = ratingRange % messagesArray.length;
const smallSize = (ratingRange - numLargeGroups) / messagesArray.length;
const split = numLargeGroups * (smallSize + 1);
// 转为 0-based 索引便于计算
const zeroBased = currentRating - 1;
if (zeroBased < split) {
// 属于前 numLargeGroups 个“大组”,每组 size = smallSize + 1
return messagesArray[Math.floor(zeroBased / (smallSize + 1))];
} else {
// 属于后续“小组”,每组 size = smallSize
return messagesArray[
numLargeGroups + Math.floor((zeroBased - split) / smallSize)
];
}
}✅ 验证示例(全部通过原题测试用例):
console.log(getMessageByRating(5, ['Bad', 'Okay', 'Good'], 2)); // 'Bad' console.log(getMessageByRating(10, ['Bad', 'Okay', 'Good', 'Amazing'], 9)); // 'Amazing'
⚠️ 注意事项:
- 该算法假设 messagesArray 已按语义升序排列(如 ['Bad', 'Okay', 'Good']),且 ratingRange ≥ messagesArray.length;
- 若 ratingRange === messagesArray.length,则每个消息恰好对应1个评分点(完全均分);
- 时间复杂度为 O(1),无循环或递归,适合高频调用场景(如实时拖拽反馈);
- 如需支持反向映射(由消息查推荐评分区间),可扩展返回 { message: string; min: number; max: number } 对象。
此方案兼顾数学严谨性与工程实用性,是动态评分语义化映射的理想解法。










