zoukankan      html  css  js  c++  java
  • Codeforces 1189F. Array Beauty

    传送门

    首先可以注意到序列里面元素的顺序对答案是没有影响的,所以二话不说先排序再看看怎么搞

    考虑枚举每种子序列可能产生的贡献并算一下产生这个贡献的子序列有多少

    考虑设 $F(x)$ 表示选择的元素差值至少为 $x$ 的长度为 $k$ 的子序列的方案数

    那么最终如果直接把每个 $F(x),x in [1,max(a)]$ 加起来会发现,对于任意一种差值为 $t$ 的方案,它都被所有 $x<=t$ 的 $F(x)$ 各计算到了一次,那么总贡献即为 $t$,所以只要求出 $F$ 然后加起来就是我们要的答案

    先不考虑复杂度,那么这个显然是可以 $dp$ 的,设 $f[i][j]$ 表示长度为 $i$ ,考虑了前 $j$ 个位置,第 $j$ 个位置强制选择时的子序列方案数

    那么有转移 $f[i][j]+=f[i-1][k]$ 其中 $k<j$ 并且 $a_j-a_k>=x$ (注意这时 $a$ 已经按从小到大排序了)

    注意到随着 $j$ 的增加,合法的 $k$ 一定是越来越大的并且是一段前缀区间,那么转移显然可以维护一个指针和一个前缀和加速到 $O(1)$

    现在单次 $dp$ 的复杂度就是优秀的 $nk$ 了,那么再考虑一下枚举 $x$ 的复杂度,很好,复杂度达到了优秀的 $max(a)nk$

    然后注意到一个看似微不足道的优化:枚举 $x$ 的时候如果 $x>max(a)/k+1$ ,那么一定不存在合法的长度为 $k$ 的序列了(显然吧)

    发现复杂度就变成了 $frac {max(a)} {k+1}nk$ 即 $max(a)n$ ...

    然后就过了......

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    #include<cmath>
    using namespace std;
    typedef long long ll;
    inline int read()
    {
        int x=0,f=1; char ch=getchar();
        while(ch<'0'||ch>'9') { if(ch=='-') f=-1; ch=getchar(); }
        while(ch>='0'&&ch<='9') { x=(x<<1)+(x<<3)+(ch^48); ch=getchar(); }
        return x*f;
    }
    const int N=1007,mo=998244353;
    inline int fk(int x) { return x>=mo ? x-mo : x; }
    int n,m,A[N];
    int f[N][N],ans;
    int main()
    {
        n=read(),m=read();
        for(int i=1;i<=n;i++) A[i]=read();
        sort(A+1,A+n+1); int mx=A[n];
        for(int I=1;I<=mx/(m-1);I++)
        {
            for(int i=1;i<=n;i++) f[1][i]=1;
            for(int i=2;i<=m;i++)
            {
                int sum=0,p=1;
                for(int j=1;j<=n;j++)
                {
                    while(A[p]+I<=A[j])
                        sum=fk(sum+f[i-1][p++]);
                    f[i][j]=sum;
                }
            }
            for(int i=1;i<=n;i++) ans=fk(ans+f[m][i]);
        }
        printf("%d
    ",ans);
        return 0;
    }
  • 相关阅读:
    java中常量定义在interface中好还是定义在class中
    CharacterEncodingFilter-Spring字符编码过滤器
    Integer判断相等,到底该用==还是equals
    ThreadLocal实现session中用户信息 的线程间共享
    分布式部署引发的问题
    分布式部署
    LogBack通过MDC实现日志记录区分用户Session
    Fragment 简介 基础知识 总结 MD
    直播 相关技术文章 相关调研文章
    直播 背景 技术体系 乐视云直播Demo
  • 原文地址:https://www.cnblogs.com/LLTYYC/p/11729176.html
Copyright © 2011-2022 走看看