区间价值 --- 题解--动态规划

目录

区间价值

题目描述

输入描述:

输出描述:

输入

输出

备注:

思路:

代码:


 

区间价值

J-区间价值_牛客竞赛动态规划专题班习题课 (nowcoder.com)
 

时间限制:C/C++ 2秒,其他语言4秒
空间限制:C/C++ 262144K,其他语言524288K
64bit IO Format: %lld

题目描述

对于一个数组a,定义其价值是其中不同的数的个数,比如对于数组[3,2,2,3,1],价值就是3。对于一个给定的长度len,求出所有长度为lenlenlen的子区间的价值之和是对于吉吉国王来说很重要,现在吉吉国王会告诉你他想知道的长度lenlenlen,你需要告诉吉吉国王答案。

比如数组[3,2,2,3,1],长度为2的子区间有[3,2],[2,2],[2,3],[3,1],那么价值分别是2,1,2,2,因此这个数组长度为2的价值和就是7。

输入描述:

第一行一个n表示数组的长度。

第二行n个数,第iii个数表示ai。

第三行一个q表示询问的次数。

接下来q行,每行一个整数表示查询的长度。

输出描述:

输出q行,第i行表示第i个询问的答案。

示例1

输入

5
3 2 4 3 1
4
1
2
3
4

输出

5
8
9
7

备注:

1≤n≤1e6 

思路:

这道题容易想到的是暴力解法(区间dp)但这肯定是会爆时间的。

现在设dp[i] 表示区间为i时的价值和。

那怎么从dp[i-1] 转移到 dp[i]

假如 当前区间为3, 数组为 32441

324 -> 3244 贡献不变

244 -> 2441 贡献加1

441 -> null 贡献-2

这里可以看出从dp[i-1] 到 dp[i] 会损伤掉后面 i-1个数的贡献值,并且前几个区间有s[i]的贡献增加。

前几个区间中那些区间是会提供贡献,或者说那些数在区间变大时可以提供贡献,这是可以预处理出来的,因为可以观察发现只有两个相同数的相隔距离大于等于i时,他们才会在长度为i的区间中提供一个贡献。(这里需要用一个后缀和统计)

代码:

import java.util.Scanner;

/**
 * @ProjectName: study3
 * @FileName: Ex7
 * @author:HWJ
 * @Data: 2023/12/5 19:53
 */
public class Main {
     static int maxN = (int) 1e6 + 5;
    public static void main(String[] args) {
        Scanner input = new Scanner(System.in);
        int n = input.nextInt();
        int[] arr = new int[maxN];
        int[] last = new int[maxN];
        long[] s = new long[maxN];
        int[] diff = new int[maxN];
        int[] cnt = new int[maxN];
        for (int i = 1; i <= n; i++) {
            arr[i] = input.nextInt();
            s[i - last[arr[i]]]++;
            last[arr[i]] = i;
        }
        for(int i = n; i > 0; i--){
            s[i] = s[i] + s[i + 1];
        }
        int tot = 0;
        for(int i = n; i > 0; i--){
            if (++cnt[arr[i]] == 1) tot++;
            diff[n - i + 1] = tot;
        }
        long[] ans = new long[n + 1];
        ans[1] = n;
        for(int i = 2; i <= n; i++){
            ans[i] = ans[i - 1] + s[i] - diff[i - 1];
        }
        int q = input.nextInt();
        for (int i = 0; i < q; i++) {
            int a = input.nextInt();
            System.out.println(ans[a]);
        }
    }
}

你可能感兴趣的:(动态规划,算法)