【题解】CF82D Two out of Three

给定一个序列\(a_i\),每次可以删除前\(3\)个中的\(2\)个,代价为所删两数的\(a_i\)最大值;若数字个数小于\(3\),就一次删完,代价同样为\(a_i\)最大值。求删掉所有数的最小代价以及方案。

\(n\le 1000,1\le a_i\le 10^6\)

Solution

这样的删数方式很特别,我们需要寻找一些性质。

从序列的形状上看,没有被删的数应该是 一个单点+一个后缀,也可能只有一个后缀。

啥,你说咋看出来的?后缀显然,单点多删几次就可以发现。

那不就很好dp了吗。

\(f_{i,j}\)表示单点位置为\(i\),后缀左端点为\(j\)时,最小代价是多少。

刷表,讨论删掉前\(3\)个中的哪\(2\)个。

\(f_{i,j}\)可以贡献到:

  • \(f_{i,j+2}\)
  • \(f_{j+1,j+1}\)
  • \(f_{j,j+2}\)

还要考虑只有后缀的情况,不妨设当\(i=j\)时,后缀为从\(i\)开始的区间。
\(f_{i,i}\)可以贡献到

  • \(f_{i+2,i+2}\)
  • \(f_{i+1,i+3}\)
  • \(f_{i,i+3}\)

然后就是记录方案。

代码:

#include<bits/stdc++.h>
#define fi first
#define se second
#define mp make_pair
#define pb push_back
using namespace std;
inline int read(){
	int x=0,f=0;
	char ch=getchar();
	while(ch<'0'||ch>'9') f|=(ch=='-'),ch=getchar();
	while('0'<=ch&&ch<='9') x=x*10+ch-'0',ch=getchar();
	return f?-x:x;
}
typedef pair<int,int> pii;
const int N=1005;
int n,a[N],f[N][N],pre[N][N][2];
void upd(int x,int y,int i,int j,int val){
	//用f[i,j]更新f[x,y],val为转移时的代价 
	if(f[x][y]>f[i][j]+val){
		f[x][y]=f[i][j]+val;
		pre[x][y][0]=i;
		pre[x][y][1]=j;
	}
}
int main(){
	n=read();
	for(int i=1;i<=n;++i) a[i]=read();
	memset(f,0x3f,sizeof(f));
	f[1][1]=0;
	for(int i=1;i<=n+2;++i){
		upd(i+2,i+2,i,i,max(a[i],a[i+1]));
		upd(i+1,i+3,i,i,max(a[i],a[i+2]));
		upd(i,i+3,i,i,max(a[i+1],a[i+2]));
		for(int j=i+1;j<=n+2;++j){
			upd(i,j+2,i,j,max(a[j],a[j+1]));
			upd(j+1,j+1,i,j,max(a[i],a[j]));
			upd(j,j+2,i,j,max(a[i],a[j+1])); 
		}
	}
	vector<pii>ans;
	int x,y;
	if(n&1) printf("%d\n",f[n+2][n+2]),x=y=n+2;
	else printf("%d\n",f[n+1][n+1]),x=y=n+1;
	while(x>1||y>1){
		int px=pre[x][y][0],py=pre[x][y][1];
		if(px==py){
			if(x==px+2&&y==py+2) ans.pb(mp(px,px+1));
			if(x==px+1&&y==py+3) ans.pb(mp(px,px+2));
			if(x==px&&y==py+3) ans.pb(mp(px+1,px+2));
		}else{
			if(x==py+1&&y==py+1) ans.pb(mp(px,py));
			if(x==py&&y==py+2) ans.pb(mp(px,py+1));
			if(x==px&&y==py+2) ans.pb(mp(py,py+1));
		}
		x=px;
		y=py;
	}
	reverse(ans.begin(),ans.end());
	for(int i=0;i<ans.size();++i){
//		printf("%d %d\n",ans[i].fi,ans[i].se);
		if(ans[i].fi<=n) printf("%d ",ans[i].fi);
		if(ans[i].se<=n) printf("%d ",ans[i].se);
		printf("\n");
	}
	return 0;
}
/*
给定一个序列,每次可以删除前3个中的2个,代价为max(ai),剩余一个时代价为a,求最小代价 

剩下的元素是一个单点 & 一个后缀

f[i,j]:单点在i,后缀左端点为j,最小时间为多少

f[i,i] -> f[i+2,i+2],f[i+1,i+3],f[i,i+3]
f[i,j] -> f[i,j+2],f[j+1,j+1],f[j,j+2]
*/
posted @ 2021-12-18 12:07  hzy1  阅读(36)  评论(0编辑  收藏  举报