2024-11-13:求出所有子序列的能量和。用go语言,给定一个整数数组nums和一个正整数k, 定义一个子序列的能量为子序列中任意两个元素之间的差值绝对值的最小值。 找出nums中长度为k的所有子
2024-11-13:求出所有子序列的能量和。用go语言,给定一个整数数组nums和一个正整数k,
定义一个子序列的能量为子序列中任意两个元素之间的差值绝对值的最小值。
找出nums中长度为k的所有子序列的能量和,
对结果取模10^9 + 7后返回。
输入:nums = [1,2,3,4], k = 3。
输出:4。
解释:
nums 中总共有 4 个长度为 3 的子序列:[1,2,3] ,[1,3,4] ,[1,2,4] 和 [2,3,4] 。能量和为 |2 - 3| + |3 - 4| + |2 - 1| + |3 - 4| = 4 。
答案2024-11-13:
题目来自leetcode3098。
大体步骤如下:
1.输入解析:
输入一个整数数组
nums和一个正整数k。例如:
nums = [1, 2, 3, 4],k = 3。
2.预处理:
对
nums进行排序,以便更容易处理差值。计算所有可能的差值
vals,即对于每一对(nums[i], nums[j])(i > j),计算nums[i] - nums[j],并将这些差值存入vals。将一个无穷大值
inf添加到vals中,确保后续处理边界情况。对
vals进行排序并去重,得到唯一的差值数组。
3.动态规划数组初始化:
初始化三维数组
d,其中d[i][p][v]表示考虑到第i个元素,长度为p的子序列中,最小差值为vals[v]的子序列个数。初始化二维数组
border,其中border[i][p]表示考虑到第i个元素,长度为p的子序列中,当前处理到的vals数组的索引边界。初始化二维数组
sum和suf,用于计算前缀和和后缀和,以便快速更新d数组。
4.动态规划填充:
遍历
nums中的每个元素nums[i],并对于每个j < i,计算nums[i] - nums[j]在vals中的位置pos。对于每个可能的子序列长度
p(从1到k),更新d,sum,suf, 和border数组。这一步的核心是利用前缀和和后缀和快速更新
d数组,以及利用border数组来避免重复计算。
5.结果计算:
遍历每个
d[i][k][v],其中i是nums的索引,k是子序列长度,v是vals的索引。计算每个
d[i][k][v]对结果的贡献,即vals[v] * d[i][k][v],并累加到res中。对
res取模10^9 + 7后返回。
6.输出:
- 输出最终计算得到的
res。
时间复杂度:
排序
nums:O(n log n)生成
vals并排序去重:O(n^2 log n^2)(因为最多有n(n-1)/2个差值,但去重和排序的复杂度较高)动态规划填充:
O(n^2 k m),其中n是nums的长度,k是子序列长度,m是vals的长度(去重后的差值个数)。结果计算:
O(n m)总时间复杂度:由于
m最多为n^2,因此总时间复杂度为O(n^4 k),但在实际情况下,由于vals的去重,m通常远小于n^2。
空间复杂度:
三维数组
d:O(n k m)二维数组
border,sum,suf:O(n k)和O(k m)其他辅助数组和变量:
O(n^2)(用于存储差值vals)总空间复杂度:
O(n k m + n^2),同样地,由于m通常远小于n^2,实际空间使用会更少。
综上所述,尽管理论上的时间复杂度和空间复杂度较高,但由于 vals 的去重和排序效率,以及动态规划过程中的前缀和、后缀和优化,实际运行时的性能可能会更好。
Go完整代码如下:
package main
import (
"fmt"
"sort"
)
const mod = 1e9 + 7
const inf = 0x3f3f3f3f
func sumOfPowers(nums []int, k int) int {
n := len(nums)
sort.Ints(nums)
var vals []int
for i := 0; i < n; i++ {
for j := 0; j < i; j++ {
vals = append(vals, nums[i]-nums[j])
}
}
vals = append(vals, inf)
sort.Ints(vals)
vals = unique(vals)
d := make([][][]int, n)
for i := range d {
d[i] = make([][]int, k+1)
for j := range d[i] {
d[i][j] = make([]int, len(vals))
}
}
border := make([][]int, n)
for i := range border {
border[i] = make([]int, k+1)
}
sum := make([][]int, k+1)
for i := range sum {
sum[i] = make([]int, len(vals))
}
suf := make([][]int, n)
for i := range suf {
suf[i] = make([]int, k+1)
}
for i := 0; i < n; i++ {
for j := 0; j < i; j++ {
pos := sort.SearchInts(vals, nums[i]-nums[j])
for p := 1; p <= k; p++ {
for border[j][p] < pos {
sum[p][border[j][p]] = (sum[p][border[j][p]] - suf[j][p] + mod) % mod
sum[p][border[j][p]] = (sum[p][border[j][p]] + d[j][p][border[j][p]]) % mod
suf[j][p] = (suf[j][p] - d[j][p][border[j][p]] + mod) % mod
border[j][p]++
sum[p][border[j][p]] = (sum[p][border[j][p]] + suf[j][p]) % mod
}
}
}
d[i][1][len(vals)-1] = 1
for p := 2; p <= k; p++ {
for v := 0; v < len(vals); v++ {
d[i][p][v] = sum[p-1][v]
}
}
for p := 1; p <= k; p++ {
for v := 0; v < len(vals); v++ {
suf[i][p] = (suf[i][p] + d[i][p][v]) % mod
}
sum[p][0] = (sum[p][0] + suf[i][p]) % mod
}
}
res := 0
for i := 0; i < n; i++ {
for v := 0; v < len(vals); v++ {
res = (res + int(int64(vals[v])*int64(d[i][k][v])%mod)) % mod
}
}
return res
}
func unique(arr []int) []int {
if len(arr) == 0 {
return arr
}
result := []int{arr[0]}
for _, v := range arr {
if v != result[len(result)-1] {
result = append(result, v)
}
}
return result
}
func main() {
nums := []int{1, 2, 3, 4}
k := 3
fmt.Println(sumOfPowers(nums, k))
}

Rust完整代码如下:
use std::collections::HashSet;
const MOD: i32 = 1_000_000_007;
const INF: i32 = 0x3f3f3f3f;
fn sum_of_powers(nums: Vec<i32>, k: usize) -> i32 {
let n = nums.len();
let mut nums = nums;
nums.sort();
let mut vals = Vec::new();
for i in 0..n {
for j in 0..i {
vals.push(nums[i] - nums[j]);
}
}
vals.push(INF);
vals.sort();
vals = unique(vals);
let mut d = vec![vec![vec![0; vals.len()]; k + 1]; n];
let mut border = vec![vec![0; k + 1]; n];
let mut sum = vec![vec![0; vals.len()]; k + 1];
let mut suf = vec![vec![0; k + 1]; n];
for i in 0..n {
for j in 0..i {
let pos = match vals.binary_search(&(nums[i] - nums[j])) {
Ok(p) => p,
Err(p) => p,
};
for p in 1..=k {
while border[j][p] < pos {
sum[p][border[j][p]] =
(sum[p][border[j][p]] - suf[j][p] + MOD) % MOD;
sum[p][border[j][p]] =
(sum[p][border[j][p]] + d[j][p][border[j][p]]) % MOD;
suf[j][p] = (suf[j][p] - d[j][p][border[j][p]] + MOD) % MOD;
border[j][p] += 1;
sum[p][border[j][p]] =
(sum[p][border[j][p]] + suf[j][p]) % MOD;
}
}
}
d[i][1][vals.len() - 1] = 1;
for p in 2..=k {
for v in 0..vals.len() {
d[i][p][v] = sum[p - 1][v];
}
}
for p in 1..=k {
for v in 0..vals.len() {
suf[i][p] = (suf[i][p] + d[i][p][v]) % MOD;
}
sum[p][0] = (sum[p][0] + suf[i][p]) % MOD;
}
}
let mut res = 0;
for i in 0..n {
for v in 0..vals.len() {
res = (res + vals[v] as i64 * d[i][k][v] as i64 % MOD as i64) % MOD as i64;
}
}
res as i32
}
fn unique(mut arr: Vec<i32>) -> Vec<i32> {
if arr.is_empty() {
return arr;
}
arr.sort();
let mut result = Vec::new();
let mut prev = arr[0];
result.push(prev);
for &v in &arr[1..] {
if v != prev {
result.push(v);
prev = v;
}
}
result
}
fn main() {
let nums = vec![1, 2, 3, 4];
let k = 3;
println!("{}", sum_of_powers(nums, k));
}

2024-11-13:求出所有子序列的能量和。用go语言,给定一个整数数组nums和一个正整数k, 定义一个子序列的能量为子序列中任意两个元素之间的差值绝对值的最小值。 找出nums中长度为k的所有子的更多相关文章
- JAVA中计算两个日期时间的差值竟然也有这么多门道
上半年春招的时候,作为面试官,对于面试表现的不错的同学会要求其写一小段代码看看.题目很简单: 给定一个日期,然后计算下距离今天相差的天数. 本以为这么个问题就是用来活跃面试氛围的,但是结果却让人大跌眼 ...
- 在excel中如何计算两个时间之间的差[转]
因为时间是由序列号所代表的,用户可以用较晚的时间减去较早的时间以得到间隔.例如,单元格A3含有5:30,单元格B3含有14:00,下面的公式返回8:30(间隔8小时30分). =B3-A3 然而,如果 ...
- [Xcode 实际操作]九、实用进阶-(4)计算两个日期间的差值
目录:[Swift]Xcode实际操作 本文将演示如何计算两个日期之间的差值. 在项目导航区,打开视图控制器的代码文件[ViewController.swift] import UIKit class ...
- 用Scala实现集合中相邻元素间的差值
欢迎转载,转载请注明出处,徽沪一郎. 概要 代码这东西,不写肯定不行,新学Scala不久,将实际遇到的一些问题记录下来,日后也好查找. 今天讲的是如何计算同一集合中元素两两之间的差值,即求开始集合(a ...
- 给定一个数组,求如果排序后,相邻两个元素的最大差值,要求时间复杂度为O(N)
第一种方法: 计数排序后,然后找出两两之间的最大差值 计数排序的时间复杂度是O(N) public class CountSort { public static void main(String[] ...
- Jquery计算时间戳之间的差值,可返回年,月,日,小时等
/** * 计算时间戳之间的差值 * @param startTime 开始时间戳 * @param endTime 结束时间戳 * @param type 返回指定类型差值(year, month, ...
- 218。重复元素II(重复元素的下标差值<=K)(哈希)
给定一个整数数组和一个整数 k,判断数组中是否存在两个不同的索引 i 和 j,使得 nums [i] = nums [j],并且 i 和 j 的差的 绝对值 至多为 k. 示例 1: 输入: nums ...
- js计算两个日期的天数差值
js计算两个日期的天数差值 通过两个日期计算这两个日期之间的天数差值 /** * 计算天数差的函数,通用 * @param sDate1 * @param sDate2 * @returns {Num ...
- [转]ASP.NET MVC中的两个Action之间值的传递--TempData
本文转自:ASP.NET MVC中的两个Action之间值的传递--TempData 一. ASP.NET MVC中的TempData 在ASP.NET MVC框架的ControllerBase中存在 ...
- 【编程题目】一个整数数组,长度为 n,将其分为 m 份,使各份的和相等,求 m 的最大值★★ (自己没有做出来!!)
45.雅虎(运算.矩阵): 2.一个整数数组,长度为 n,将其分为 m 份,使各份的和相等,求 m 的最大值 比如{3,2,4,3,6} 可以分成 {3,2,4,3,6} m=1; {3,6}{2,4 ...
随机推荐
- k8s实践——命名空间隔离+request-key机制解决CSI内核态域名解析
0x01 背景 Pod需要使用远程存储的PV,由同k8s集群内的服务提供的存储服务.一开始的做法是: CSI中解析Service的clusterIP. 然后使用clusterIP挂载PV卷. 但因为走 ...
- JavaScript设计模式样例十七 —— 迭代器模式
迭代器模式(Itrator Pattern) 定义:用于顺序访问集合对象的元素,不需要知道集合对象的底层表示.目的:提供一种方法顺序访问一个聚合对象中各个元素, 而又无须暴露该对象的内部表示.场景:$ ...
- 关于捣鼓Gentoo的一些见解
现在很少有人使用gentoo,大家对它的印象都是一个很难用的系统,我想给大家讲讲折腾Gentoo一年的心得,仅供参考 使用archlinux安装盘,genfstab生成fstab 使用gentoo-k ...
- Windows PE 安装
Microsoft 官方提供的 Windows PE 默认只有命令行界面.如果想要使用带有桌面环境的 Windows PE,推荐使用微 PE . 下载并安装 Windows ADK 和 WinPE 加 ...
- 一个操作系统的设计与实现——第19章 IA32-e模式
19.1 64位段描述符与GDT 在32位操作系统中,我们使用的是平坦模型而非分段模型,从而,段描述符的段基址和段限长均成了摆设.在64位模式下,就连CPU也淘汰了分段模型,转而固定使用平坦模型. 6 ...
- 知识点考古:php5的面向对象学习笔记
闲来无事翻看以前收藏的资料,考古到保存的这篇文章对php的OOP的整理还很系统.原链接已经打不开(http://www.cublog.cn/u/17686/showart.php?id=146562) ...
- Asp.net Core 学习笔记 Azure Storage
更新: 2021-07-22 使用 Azure storage 以后, 还要解决一个 url 的问题. 文件自然是通过我们的 domain 来访问才合理丫. 这个是 azure 的 url : htt ...
- 阿里面试官常问的TCP和UDP,你真的弄懂了吗?
前 言 作为软件测试,大家都知道一些常用的网络协议是我们必须要了解和掌握的,面试的时候面试官也非常喜欢问一些协议相关的问题,其中有两个协议因为非常基础,出现的频率非常之高,分别是 "T ...
- Python— 函数
函数定义: 示例: # 求 1 - n 的和 def sum(date): sum = 0 for i in range(1,date + 1): sum += i return sum # 函数调用 ...
- win10安装linux的gcc
mysy2下载gcc 过程比较艰苦,2024年秋冬讲课,被linux毒打了3天 pacman -S mingw-w64-ucrt-x86_64-gcc 这个一次成功,不行继续接大招 实在不行安装 ...