CSP-S2021 题解

T1.廊桥分配

\(n^2\)暴力:直接枚举分配过程按照题意模拟。
对于每架飞机,设当前区域内廊桥总数为\(x\),那么只有当\(x\)到达一定值时,它才有贡献
考虑对于每个可能的\(x\),求出来它可以使得多少飞机有贡献,那么只要前缀和一下再\(O(n)\)扫一遍就可以求得答案
思考怎么求每个\(x\)的贡献值,注意到体重说有一个先到先得的规定,也就是说当廊桥数量固定后,情况是唯一的,不存在方案优不优的问题
从简单入手,每个飞机抽象称一条线段,如果只有应该廊桥,那么肯定是一开始不交的线段有贡献
可以用set维护,每次相当于从前往后扫,不交的线段数量就是当前的\(x\)的贡献数,如果到头了从头开始,\(x\)要加一
由于每次一定会从集合里删去元素,所以算上lower_bound的复杂度上界是\(n\log n\)

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N=100050;
int a1[N],a2[N],b1[N],b2[N];
int lsh[4*N],n,m1,m2,tot,cnt;
int ans1[N],ans2[N];
set <pair<int,int> >s;
signed main()
{
	//freopen("airport.in","r",stdin);
	//freopen("airport.out","w",stdout);
	cin>>n>>m1>>m2;
	for(int i=1;i<=m1;i++)scanf("%lld%lld",&a1[i],&a2[i]);
	for(int i=1;i<=m2;i++)scanf("%lld%lld",&b1[i],&b2[i]);
	for(int i=1;i<=m1;i++)lsh[i]=a1[i],lsh[m1+i]=a2[i];
	tot=m1*2;
	for(int i=1;i<=m2;i++)lsh[tot+i]=b1[i],lsh[tot+m2+i]=b2[i];
	tot+=m2*2;sort(lsh+1,lsh+1+tot);
	cnt=unique(lsh+1,lsh+1+tot)-lsh-1;
	for(int i=1;i<=m1;i++)
	{
		a1[i]=lower_bound(lsh+1,lsh+1+cnt,a1[i])-lsh;
		a2[i]=lower_bound(lsh+1,lsh+1+cnt,a2[i])-lsh;
	}
	for(int i=1;i<=m2;i++)
	{
		b1[i]=lower_bound(lsh+1,lsh+1+cnt,b1[i])-lsh;
		b2[i]=lower_bound(lsh+1,lsh+1+cnt,b2[i])-lsh;
	}
	for(int i=1;i<=m1;i++)s.insert(make_pair(a1[i],a2[i]));
	int ma=1e9,num=0;
	while(s.size())
	{
		auto ga=s.lower_bound(make_pair(ma,0));
		if(ga==s.end())
		{
			num++;ga=s.begin();
			ma=(*ga).second;s.erase(ga);
			ans1[num]++;continue;
		}
		ans1[num]++;ma=(*ga).second;
		s.erase(ga);
	}
	for(int i=1;i<=m2;i++)s.insert(make_pair(b1[i],b2[i]));
	ma=1e9;num=0;
	while(s.size())
	{
		auto ga=s.lower_bound(make_pair(ma,0));
		if(ga==s.end())
		{
			num++;ga=s.begin();
			ma=(*ga).second;s.erase(ga);
			ans2[num]++;continue;
		}
		ans2[num]++;ma=(*ga).second;
		s.erase(ga);
	}
	for(int i=1;i<=n;i++)
	{
		ans1[i]=ans1[i-1]+ans1[i];
		ans2[i]=ans2[i-1]+ans2[i];
	}
	int an=0;
	for(int i=0;i<=n;i++)an=max(an,ans1[i]+ans2[n-i]);
	cout<<an<<endl;
	return 0;
}

因为并不需要在值域上开数组,所以离散化是不必要的

T2.括号序列

考场上直接弃了,这也反应了dp能力确实有待提高
本来是思考的dp,但想的一直是线性dp,最终没能仔细理解题目,暴力也不大会
考虑区间dp,设计状态是\(f_{l,r}\)和\(g_{l,r}\)分别表示\(l,r\)这段区间合法,最左边和最右边的括号匹配/不匹配的方案数
为啥要设计这个状态是因为避免重复,注意这里匹配不匹配不是指是不是左右括号(否则直接不合法跳过),而是指最左边和最右边的是否能配成一对,还是他们分别和中间的什么配成了对
预处理\(p_{l,r}\)表示一个区间内是否全是\(*\)或者\(?\),一个布尔数组,转移时要用
转移的主要思路就是看拼成的合法序列是他说的哪一种,对应不同转移
边界特判\(len=2\),两端不合法直接continue,然后讨论转移方程
先是匹配的\(f\)数组
当最终\((S)\)的时候,中间一定都是星号,就有

\[f_{l,r}+=[p_{l+1,r-1}](r-l-1<=k) \]

一定注意中间长度不能超过\(k\)
最终\((A)\)也类似,中间只要合法即可

\[f_{l,r}+=f_{l+1,r-1}+g_{l+1,r-1} \]

对于\((AS)\)和\((SA)\)的转移是类似的,枚举连续一段星号的个数\(O(n)\)转移

\[(SA):f_{l,r}+=\sum_{i=1}^k (f_{l+i+1,r-1}+g_{l+i+1,r-1})[p_{l+1,l+i}] \]

\[(AS):f_{l,r}+=\sum_{i=1}^k (f_{l+1,r-i-1}+g_{l+1,r-i-1})[p_{r-i,r-1}] \]

枚举的时候注意\(i\)的上界注意不要超,应该和右端点取min
剩下的\(ASB\)和\(AB\)就是另一种情况\(g\)了,转移要枚举中间\(S\)的长度,同时避免重复要钦定后面一定是匹配的

\[g_{l,r}+=\sum_{l<i<j<r,j-i-1<=k} ((f_{l,i}+g_{l,i})\times f_{j,r})[p_{i+1,j-1}] \]

这里定义\(p_{x,x-1}\)也是1,便于转移
发现最后一个暴力转移是\(n^2\),直接写应该是65分,考虑优化
发现最后在累加就是在统计一个前面有若干个星号,后面是一个匹配的序列的方案数,那么可以直接把这个类似后缀和的数组维护出来
设\(dp_{l,r}\)表示上述数组,那么可以按照转移\(SA\)类似的方法求出来

\[dp_{l,r}=\sum_{i=1}^k f_{l+i,r}[p_{l,l+i-1}] \]

此外所有的f方案都应该计入对应的dp方案之中,所以还要加上(因为上面的式子没有算)
那么最后转移直接把对应\(j\)的枚举换成dp数组就行了,总复杂度\(n^3\)

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N=505;
const int mod=1e9+7;
int f[N][N],g[N][N],n,kk,ga[N][N];
bool p[N][N];char c[N];
signed main()
{
	cin>>n>>kk;scanf("%s",c+1);
	for(int i=1;i<=n;i++)
	{
		p[i][i-1]=1;
		for(int j=i;j<=n;j++)
		{
	 		bool fg=1;
	 		for(int k=i;k<=j;k++)if(c[k]!='?'&&c[k]!='*')
	 	  		{fg=0;break;}
	 		p[i][j]=fg;	
		}
	}
	for(int len=2;len<=n;len++)
	{
		for(int l=1;l<=n-len+1;l++)
		{
			int r=l+len-1;
			if((c[l]!='('&&c[l]!='?')||(c[r]!=')'&&c[r]!='?'))continue;
			if(len==2){f[l][r]=ga[l][r]=1;continue;}
			if(r-l-1<=kk)f[l][r]=(f[l][r]+p[l+1][r-1])%mod;
			f[l][r]=(f[l][r]+f[l+1][r-1]+g[l+1][r-1])%mod;
			for(int i=1;i<=kk&&l+1+i<=r-1;i++)if(p[l+1][l+i])f[l][r]=(f[l][r]+f[l+i+1][r-1]+g[l+i+1][r-1])%mod;
			for(int i=1;i<=kk&&r-i-1>=l+1;i++)if(p[r-i][r-1])f[l][r]=(f[l][r]+f[l+1][r-i-1]+g[l+1][r-i-1])%mod;
			for(int i=l+1;i<r;i++)g[l][r]=(g[l][r]+(f[l][i]+g[l][i])*ga[i+1][r]%mod)%mod;
		}
		for(int l=1;l<=n-len+1;l++)
		{
			int r=l+len-1;if(c[r]!=')'&&c[r]!='?')continue;
			ga[l][r]=f[l][r];
			for(int i=1;i<=kk&&l+i<=r;i++)if(p[l][l+i-1])ga[l][r]=(ga[l][r]+f[l+i][r])%mod;				
		}
	}
	cout<<(f[1][n]+g[1][n])%mod<<endl;
	return 0;
}

T3.回文

数据范围启发正解的时间复杂度应该就是\(O(n)\),带\(log\)的可能性不是很大
观察样例会发现,已选数字剩余位置一定是连续的,因此可以以这个进行贪心
反证如果不连续,那么两个位置一定会挡住一些数字,使得最后一定形不成回文
那么维护几个指针开始扫,优先选左边保证字典序,当选出\(n\)个数之后利用回文性质反推出剩下\(n\)个操作都是什么
一个细节是不能选择重复数字,当已经选择的和剩余位置挨上需要特判一下,例子可以看大样例98组两个一起的17

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N=500050;
int n,a[2*N],b[2*N],tot;
int ans[2*N];vector <int> p[2*N];
inline void clear()
{
	for(int i=1;i<=2*n;i++)p[i].clear(),a[i]=b[i]=ans[i]=0;
}
inline void gan(int l,int r)
{
	for(int i=1;i<=n;i++)
	{
		if(a[l]==b[n-i+1])l++,ans[++tot]=0;
		else r--,ans[++tot]=1;
	}
	for(int i=1;i<=2*n;i++)
	{	
		if(ans[i])putchar('R');
		else putchar('L');
	}
	puts("");
}
signed main()
{
	//freopen("palin.in","r",stdin);
	//freopen("palin.out","w",stdout);
	int t;cin>>t;
	while(t--)
	{
		clear();scanf("%lld",&n);tot=0;
		for(int i=1;i<=2*n;i++)scanf("%lld",&a[i]);
		for(int i=1;i<=2*n;i++)p[a[i]].push_back(i);
		int ga=p[a[1]][0];if(ga==1)ga=p[a[1]][1];
		int pl=ga-1,pr=ga+1;
		int l=2,r=2*n;b[++tot]=a[1],ans[tot]=0;
		for(int i=2;i<=n;i++)
		{
			bool ok=0;
			int p1=p[a[l]][0];if(p1==l)p1=p[a[l]][1];
			int p2=p[a[r]][0];if(p2==r)p2=p[a[r]][1];
			if(p1==pl&&l<pl)b[++tot]=a[l++],ans[tot]=0,pl--,ok=1;
			else if(p1==pr)b[++tot]=a[l++],ans[tot]=0,pr++,ok=1;
			else if(p2==pl)b[++tot]=a[r--],ans[tot]=1,pl--,ok=1;
			else if(p2==pr&&r>pr)b[++tot]=a[r--],ans[tot]=1,pr++,ok=1;
			if(!ok){tot=0;break;}
		}
		if(tot==n){gan(l,r);continue;}
		ga=p[a[2*n]][0];if(ga==2*n)ga=p[a[2*n]][1];
		pl=ga-1,pr=ga+1;
		l=1,r=2*n-1;b[++tot]=a[2*n],ans[tot]=1;
		for(int i=2;i<=n;i++)
		{
			bool ok=0;
			int p1=p[a[l]][0];if(p1==l)p1=p[a[l]][1];
			int p2=p[a[r]][0];if(p2==r)p2=p[a[r]][1];
			if(p1==pl&&l<pl)b[++tot]=a[l++],ans[tot]=0,pl--,ok=1;
			else if(p1==pr)b[++tot]=a[l++],ans[tot]=0,pr++,ok=1;
			else if(p2==pl)b[++tot]=a[r--],ans[tot]=1,pl--,ok=1;
			else if(p2==pr&&r>pr)b[++tot]=a[r--],ans[tot]=1,pr++,ok=1;
			if(!ok){tot=0;break;}
		}
		if(!tot){puts("-1");continue;}
		gan(l,r);
	}
	return 0;
}

T4.交通规划

正解看不懂,写了个复杂对不太正确的网络流做法
黑白染色,格子里面的点直接连双向边,格子外面的白点从源点向其连边,连到最近的点,黑点由最近的点向他连边,连到汇点
发现这样答案就是最小割,即选择代价最少的边删掉把黑白割断,因此源汇点都是INF
理论是60-80,但是机房大佬都A了,人傻常数大卡不过去,洛谷数据最高95

#include <bits/stdc++.h>
using namespace std;
const int N=505;
inline int read()
{
	int x=0;char ch=getchar();
	while(ch<'0'||ch>'9')ch=getchar();
	while(ch>='0'&&ch<='9')x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
	return x;
}
struct node{
	int to,next,w;
}a[(int)2e6];
int head[N*N],mm=2;
inline void add(int x,int y,int w)
{
	a[mm].to=y;a[mm].w=w;
	a[mm].next=head[x];head[x]=mm++;
}
inline void gan(int x,int y,int z)
{
	add(x,y,z);add(y,x,0);
}
int s,t,hd[N*N],d[N*N];
inline bool bfs()
{
	memset(d,0x3f,sizeof(d));
	memcpy(head,hd,sizeof(hd));
	std::queue <int> q;
	q.push(s);d[s]=0;
	while(q.size())
	{
		int x=q.front();q.pop();
		for(int i=head[x];i;i=a[i].next)
		{
			if(!a[i].w)continue;
			int y=a[i].to;
			if(d[y]>d[x]+1)d[y]=d[x]+1,q.push(y);
		}	
		if(x==t)return 1;
	}
	return 0;
}
int dfs(int x,int flow)
{
	if(x==t)return flow;
	int rest=flow,k;
	for(int i=head[x];i;head[x]=i=a[i].next)
	{
		if(!a[i].w)continue;
		int y=a[i].to;
		if(d[y]==d[x]+1)
		{
			k=dfs(y,min(a[i].w,rest));
			if(!k)d[y]=0;
			else a[i].w-=k,a[i^1].w+=k,rest-=k;
		}
		if(!rest)break;
	}
	return flow-rest;
}
int n,m,T,w1[N][N],w2[N][N],mp[N][N],tot;
inline void clear()
{
	mm=2;memset(head,0,sizeof(head));
}
signed main()
{
	cin>>n>>m>>T;
	for(int i=1;i<=n-1;i++)for(int j=1;j<=m;j++)w1[i][j]=read();
	for(int i=1;i<=n;i++)for(int j=1;j<=m-1;j++)w2[i][j]=read();
	s=++tot,t=++tot;for(int i=1;i<=n;i++)for(int j=1;j<=m;j++)mp[i][j]=++tot;
	while(T--)
	{
		clear();int k=read(),cnt=tot;
		for(int i=1;i<=n;i++)for(int j=1;j<=m;j++)
		{
			if(i>1)gan(mp[i][j],mp[i-1][j],w1[i-1][j]);
			if(i<n)gan(mp[i][j],mp[i+1][j],w1[i][j]);
			if(j>1)gan(mp[i][j],mp[i][j-1],w2[i][j-1]);
			if(j<m)gan(mp[i][j],mp[i][j+1],w2[i][j]);
		}
		for(int i=1;i<=k;i++)
		{
			int w=read(),x=read(),op=read(),id;
			if(x>=1&&x<=m)id=mp[1][x];
			else if(x>m&&x<=m+n)id=mp[x-m][m];
			else if(x>m+n&&x<=m*2+n)id=mp[n][m-(x-(m+n))+1];
			else id=mp[n-(x-(2*m+n))+1][1];
			if(op)gan(s,++cnt,1e9),gan(cnt,id,w);
			else gan(++cnt,t,1e9),gan(id,cnt,w);
		}		
		memcpy(hd,head,sizeof(hd));
		int ans=0;
		while(bfs())ans+=dfs(s,1e9);
		printf("%d\n",ans);
	}
	return 0;
}
上一篇:CSP-S 2021 题解


下一篇:CSP 赛后总结