LeetCode 2615.等值距离和:分组(哈希表+前缀和)

【LetMeFly】2615.等值距离和:分组(哈希表+前缀和)

力扣题目链接:https://leetcode.cn/problems/sum-of-distances/

给你一个下标从 0 开始的整数数组 nums 。现有一个长度等于 nums.length 的数组 arr 。对于满足 nums[j] == nums[i]j != i 的所有 jarr[i] 等于所有 |i - j| 之和。如果不存在这样的 j ,则令 arr[i] 等于 0

返回数组arr

示例 1:

复制代码
输入:nums = [1,3,1,1,2]
输出:[5,0,3,4,0]
解释:
i = 0 ,nums[0] == nums[2] 且 nums[0] == nums[3] 。因此,arr[0] = |0 - 2| + |0 - 3| = 5 。 
i = 1 ,arr[1] = 0 因为不存在值等于 3 的其他下标。
i = 2 ,nums[2] == nums[0] 且 nums[2] == nums[3] 。因此,arr[2] = |2 - 0| + |2 - 3| = 3 。
i = 3 ,nums[3] == nums[0] 且 nums[3] == nums[2] 。因此,arr[3] = |3 - 0| + |3 - 2| = 4 。 
i = 4 ,arr[4] = 0 因为不存在值等于 2 的其他下标。

示例 2:

复制代码
输入:nums = [0,5,3]
输出:[0,0,0]
解释:因为 nums 中的元素互不相同,对于所有 i ,都有 arr[i] = 0 。

提示:

  • 1 <= nums.length <= 105
  • 0 <= nums[i] <= 109

解题方法:哈希表 + 前缀和

读起来题意理解稍难,在此转述下:

对于数组中下标为 i i i的元素 n u m s [ i ] nums[i] nums[i],它对应的要返回的结果 a n s [ i ] ans[i] ans[i]怎么算呢?

对于 n u m s nums nums中所有和 n u m s [ i ] nums[i] nums[i]相等的 n u m s [ j ] nums[j] nums[j]们,计算 a b s ( j − i ) abs(j-i) abs(j−i)之和即为所求。

显而易见,对于元素 n u m s [ i ] nums[i] nums[i],其最终结果仅与和 n u m s [ i ] nums[i] nums[i]相等的元素有关。因此我们可以使用一个哈希表,键为 n u m s [ i ] nums[i] nums[i]值为所有等于 n u m s [ i ] nums[i] nums[i]的下标们,遍历一遍即可实现原始数组按照值的分组。

好了,分好组了,现在我们得到了很多的 i d x s idxs idxs数组,每个 i d x s idxs idxs数组存放的都是在 n u m s nums nums中值相等的下标们。问题就变成了给你一个升序的 i d x s idxs idxs数组,对于其中的元素 a a a,求数组中其他所有元素和 a a a的的差的绝对值之和。

怎么算?前缀和+后缀和就好:

创建一个后缀和数组 s u f f i x suffix suffix,令 s u f f i x [ i ] suffix[i] suffix[i]为 ∑ j = i j < l e n ( i d x s ) i d x s [ j ] \sum_{j=i}^{j\lt len(idxs)} idxs[j] ∑j=ij<len(idxs)idxs[j](即 i d x s idxs idxs数组中从 i d x s [ i ] idxs[i] idxs[i]到最后一个元素的元素和)。

从前到后遍历 i d x s idxs idxs数组,使用一个变量 p r e f i x prefix prefix, p r e f i x prefix prefix代表这个元素之前所有元素的和(不包含当前遍历到的元素)。

那么, i d x s [ i ] idxs[i] idxs[i]要算出的最终结果就是: i i i后面所有元素之和( s u f f i x [ i + 1 ] suffix[i+1] suffix[i+1]) − - − 后面元素个数 × i d x s [ i ] + i \times idxs[i] + i ×idxs[i]+i前面元素个数 × i d x s [ i ] \times idxs[i] ×idxs[i] − - − 前面元素之和 p r e f i x prefix prefix。

另外,也可以不使用 s u f f i x suffix suffix数组,改为 s u f f i x [ i + 1 ] suffix[i+1] suffix[i+1]由 t o t a l − p r e f i x − i d x s [ i ] total-prefix-idxs[i] total−prefix−idxs[i]得出。

  • 时间复杂度 O ( l e n ( n u m s ) ) O(len(nums)) O(len(nums))
  • 空间复杂度 O ( l e n ( n u m s ) ) O(len(nums)) O(len(nums))

AC代码

C++
cpp 复制代码
/*
 * @LastEditTime: 2026-04-24 13:53:47
 */
typedef long long ll;
class Solution {
private:
    void cal(vector<int>& idxs, vector<ll>& ans) {
        int n = idxs.size();
        ll total = 0;
        for (int i = n - 1; i >= 0; i--) {
            total += idxs[i];
        }
        ll prefix = 0;
        for (int i = 0; i < n; i++) {
            ans[idxs[i]] += (total - prefix - idxs[i]) - (ll)(n - i - 1) * idxs[i];
            ans[idxs[i]] += (ll)i * idxs[i] - prefix;
            prefix += idxs[i];
        }
    }
public:
    vector<ll> distance(vector<int>& nums) {
        unordered_map<int, vector<int>> ma;
        for (int i = 0; i < nums.size(); i++) {
            ma[nums[i]].push_back(i);
        }

        vector<ll> ans(nums.size());
        for (auto&& [_, idxs] : ma) {
            cal(idxs, ans);
        }
        return ans;
    }
};

同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~

千篇源码题解已开源

相关推荐
啦啦啦_99992 小时前
2. KNN算法之 分类&回归API实现
算法
X journey2 小时前
机器学习进阶(23):K-means聚类
人工智能·算法·机器学习
mjhcsp2 小时前
根号快速计算牛顿迭代法
开发语言·c++·算法·迭代法
菜鸟丁小真2 小时前
LeetCode hot100-79.单词搜索
数据结构·算法·leetcode·深度优先·知识总结
WL_Aurora2 小时前
排序算法(二)
java·算法·排序算法
Tisfy2 小时前
LeetCode 2833.距离原点最远的点:计数
算法·leetcode·字符串·题解·模拟·计数
浅念-2 小时前
LeetCode 模拟算法:用「还原过程」搞定编程题的入门钥匙
开发语言·c++·学习·算法·leetcode·职场和发展·模拟
圣保罗的大教堂2 小时前
leetcode 1722. 执行交换操作后的最小汉明距离 中等
leetcode
t-think2 小时前
操作符详解-C语言(下)
c语言·算法