[NOIP2021] 方差

一、题目

点此看题

二、解法

这个题我去年做不起也只能认栽了,刷题还是不够多,刷不到相同的套路。

题目转化:定义长度为 \(n-1\) 的差分数组 \(b_i=a_{i+1}-a_i\),那么操作位置 \(1<i<n\) 就等价于交换 \(b_{i-1}\)\(b_i\),这是因为我们可以把所有数都减去 \(a_1\) 所以差分数组长度只有 \(n-1\),剩下的可以自己手算验证。

那么直接暴力重排退火可以获得 \(88\) 分(参数是 T=1e3,D=0.94,eps=1e-4

有一个结论是如果我们从小到大,用插入法来规划差分数组的重排,那么新插入的差分一定只会在数组首位或者数组末位。我们知道方差乘上 \(n^2\) 等于 \(n\cdot \sum a_i^2-(\sum a_i)^2\),所以只需要记录 \(\sum a_i\),维护最小的 \(\sum a_i^2\) 就可以方便地计算代价。

具体来说就是设 \(dp[i][j]\) 表示只考虑现在插入的差分值 \(\sum a=j\) 的最小 \(\sum a^2\),转移:

  • 插入到首位:\(dp[i][j+b_i\cdot i]\leftarrow dp[i-1][j]+2\cdot j\cdot b_i+b_i\cdot b_i\cdot i\),至于系数展开二次项就可以获得。
  • 插入到末位:\(dp[i][j+s_i]\leftarrow dp[i-1][j]+s_i\cdot s_i\),其中 \(s_i=\sum_{j=1}^i b_j\)

通过最后一档数据需要跳过大段的 \(0\),时间复杂度 \(O(n\cdot a^2)\)

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm> 
using namespace std;
const int M = 10005;
#define int long long
const int inf = 0x3f3f3f3f3f3f3f3f;
int read()
{
	int x=0,f=1;char c;
	while((c=getchar())<'0' || c>'9') {if(c=='-') f=-1;}
	while(c>='0' && c<='9') {x=(x<<3)+(x<<1)+(c^48);c=getchar();}
	return x*f;
}
int n,z,a[M],b[M],s[M],dp[2][M*55];
void upd(int &x,int y) {x=min(x,y);}
signed main()
{
	n=read();
	for(int i=1;i<=n;i++) a[i]=read();
	for(int i=1;i<n;i++) b[i]=a[i+1]-a[i];
	sort(b+1,b+n);
	for(int i=1;i<n;i++) s[i]=s[i-1]+b[i];
	for(z=1;z<n && b[z]==0;z++);
	memset(dp,0x3f,sizeof dp);
	dp[(z-1)&1][0]=0;
	for(int i=z;i<n;i++)
	{
		int w=i&1;
		for(int j=0;j<=a[n]*n;j++) dp[w][j]=inf;
		for(int j=0;j<=a[n]*n;j++) if(dp[w^1][j]<inf)
		{
			//head
			upd(dp[w][j+b[i]*i],dp[w^1][j]
				+2*j*b[i]+b[i]*b[i]*i);
			//tail
			upd(dp[w][j+s[i]],dp[w^1][j]+s[i]*s[i]);
		}
	}
	int ans=inf;
	for(int j=0;j<=a[n]*n;j++) if(dp[(n-1)&1][j]<inf)
		upd(ans,n*dp[(n-1)&1][j]-j*j);
	printf("%lld\n",ans);
}
posted @ 2022-04-24 22:25  C202044zxy  阅读(130)  评论(0编辑  收藏  举报