YAOI Round #17 (Div.2) C. Sum 题解

出题人题解。

考虑差分,记 ci=aiai1(i>1)

ai 进行操作后原数组和差分数组的变化如下:

ai=ai1+ai+1ai

ci=(ai1+ai+1ai)ai1=ai+1ai

ci+1=ai+1(ai1+ai+1ai)=aiai1

我们发现,对 ai 进行操作等价于交换 cici+1

所以题目可以转换为:对数列的差分数组进行任意排列,求最后和的最大值。

c1=a1,则有 ai=c1+c2+...+ci

i=1nai=a1+a2+...+an=(c1)+(c1+c2)+...+(c1+c2+...+cn)=n×c1+(n1)×c2+...+1×cn

那么我们只需要将 c 数组从大到小排个序就好了。

需要注意的是,a1 是不会改变的,所以 c1 也不会改变,排序的时候不能把 c1 放进去。

std:

#include<cstdio>
#include<algorithm>
#include<cmath>
using namespace std;
int t,n,a[100003],c[100003];
long long ans; //答案可能爆 int
inline bool cmp(int x,int y){ //排序的辅助函数,作用是改变排序方式 
    return x>y; //x>y 是从大到小,改成 x<y 就是从小到大 
}
int main(){
    scanf("%d",&t);
    while(~--t){
        scanf("%d",&n);
        for(int i=1;i<=n;++i){
            scanf("%d",&a[i]);
        }
        for(int i=2;i<=n;++i){
            c[i]=a[i]-a[i-1]; //差分数组 
        }
        sort(c+2,c+n+1,cmp); //从大到小排 
        ans=1ll*n*a[1]; //第一个数不会变,所以一定被加 n 次 
        for(int i=2;i<=n;++i){
            ans+=1ll*(n-i+1)*c[i]; //第 i 大的数被加 n-i+1 次 
        }
        printf("%lld\n",ans);
    }
    return 0;
}

上面那个是比较方便理解的,事实上,代码可以写的更简短一些:

#include<cstdio>
#include<algorithm>
#include<cmath>
using namespace std;
int t,n,a,b,c[100003];
long long ans; 
int main(){
    scanf("%d",&t);
    while(~--t){
        scanf("%d%d",&n,&a),ans=1ll*n*a;
        for(int i=1;i<n;++i){
            scanf("%d",&b),c[i]=b-a,a=b;
        }
        sort(c+1,c+n);
        for(int i=1;i<n;++i){
            ans+=1ll*i*c[i];
        }
        printf("%lld\n",ans);
    }
    return 0;
}



posted @   Yikuwa  阅读(123)  评论(0编辑  收藏  举报
编辑推荐:
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
阅读排行:
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· Manus爆火,是硬核还是营销?
· 一文读懂知识蒸馏
· 终于写完轮子一部分:tcp代理 了,记录一下
点击右上角即可分享
微信分享提示