CSP2021题解

T1:[CSP-S 2021] 廊桥分配

考虑到你新增一个廊桥并不能影响当前可以停靠在廊桥的飞机,考虑无限廊桥时,一架飞机最少需要多少廊桥才可以停靠,如果当前有多个廊桥,肯定选择编号小的(廊桥用的越少,另一边就越多),直接优先队列维护即可

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e5+5;
#define pii pair<int,int>
int readint(){
	int x=0,f=1;char s=getchar();
	#define sc (s=getchar())
	while(s<'0'||s>'9'){
		if(s=='-')
			f=-1;
		sc;
	}
	while(s>='0'&&s<='9'){
		x=(x<<3)+(x<<1)+(s^48);
		sc;
	}
	#undef sc
	return x*f;
}
struct zz{
	int a,b;
}s[maxn],t[maxn];
bool cmp(const zz &x,const zz &y){
	return x.a<y.a;
}
int ind1[maxn],ind2[maxn];
int ans1[maxn],ans2[maxn];
void getans(int n,zz s[],int ans[],int ind[]){
	priority_queue<pii,vector<pii>,greater<pii> >q;
	priority_queue<int,vector<int>,greater<int> >vec;
	while(!q.empty()) q.pop();
	while(!vec.empty()) vec.pop();
	int cnt=1;
	q.push(make_pair(s[1].b,1));
	ans[1]++;
	for(int i=2;i<=n;i++){
		while(!q.empty()&&s[i].a>q.top().first){
			vec.push(q.top().second);
			q.pop();
		}
		if(!vec.empty()){
			ind[i]=vec.top();
			vec.pop();
			q.push(make_pair(s[i].b,ind[i]));
		}
		else{
			ind[i]=++cnt;
			q.push(make_pair(s[i].b,ind[i]));
		}
		ans[ind[i]]++;
	}
}
int main (){
	int n=readint(),m1=readint(),m2=readint();
	for(int i=1;i<=m1;i++){
		s[i].a=readint(),s[i].b=readint();
	}
	for(int i=1;i<=m2;i++){
		t[i].a=readint(),t[i].b=readint();
	}
	sort(s+1,s+m1+1,cmp);
	sort(t+1,t+m2+1,cmp);
	getans(m1,s,ans1,ind1);
	getans(m2,t,ans2,ind2);
	for(int i=1;i<=m1;i++)
		ans1[i]+=ans1[i-1];
	for(int i=1;i<=m2;i++)
		ans2[i]+=ans2[i-1];
	int maxx=0;
	for(int i=0;i<=n;i++)
		maxx=max(maxx,ans1[i]+ans2[n-i]);
	cout<<maxx<<endl;
	return 0;
} 

T2:[CSP-S 2021] 括号序列

开头没有注意到不能 ( S A S ) (SAS) (SAS)
如果可以,那可以不用区间 D P DP DP
但是他不行
考虑 i , l e n i,len ilen是以 i i i为开头的,长度为 l e n len len的序列
d p dp dp是统计 ( A ) , ( S A ) , ( A S ) (A),(SA),(AS) (A),(SA),(AS)的情况
g g g是统计所有合法序列的情况
但是,手推发现这样 g g g的状态是 n 2 n^2 n2,转移是 n 2 n^2 n2
所以改变定义
定义 g [ 0 ] g[0] g[0]是这个区间的所有合法方案, g [ 1 ] g[1] g[1] A S AS AS的方案数
转移看代码
(如果定义 g [ i ] [ j ] g[i][j] g[i][j]这样的区间的话,预处理会比较麻烦)

#include <bits/stdc++.h>
using namespace std;
const int mo = 1e9+7;
const int maxn = 505;
int readint(){
	int x=0,f=1;char s=getchar();
	#define sc (s=getchar())
	while(s<'0'||s>'9'){
		if(s=='-')
			f=-1;
		sc;
	}
	while(s>='0'&&s<='9'){
		x=(x<<3)+(x<<1)+(s^48);
		sc;
	}
	#undef sc
	return x*f;
}
char s[maxn];
int dp[maxn][maxn],g[maxn][maxn][2];//前者是单独考虑(AS) (A) (SA)的情况
//后者的零号位是题目要求的合法序列的数量 1号位是后方全是星号的数量 
int main (){
	int n=readint(),m=readint();
	scanf("%s",s+1);
	for(int i=1;i<=n;i++)
		g[i][0][0]=dp[i][0]=1;
	for(int len=2;len<=n;len++){
		for(int i=1;i<=n-len+1;i++){
			int j=i+len-1;
			for(int k=1;k<=m&&k<len-1;k++){
				if(s[j-k+1]=='('||s[j-k+1]==')')
					break;
				g[i][len][1]=(g[i][len][1]+g[i][len-k][0])%mo;
			}
			if(s[i]=='*'||s[i]==')'||s[j]=='*'||s[j]=='(')
				continue;
			dp[i][len]=(g[i+1][len-2][0]+g[i+1][len-2][1])%mo;
			for(int k=2;k<len&&k<=m+1;k++){
				if(s[i+k-1]=='('||s[i+k-1]==')')
					break;
				dp[i][len]=(dp[i][len]+g[i+k][len-1-k][0])%mo;
			}
			g[i][len][0]=dp[i][len];
			for(int k=2;k<len-1;k++){
				g[i][len][0]=(g[i][len][0]+1ll*g[i][k][1]*dp[i+k][len-k])%mo;
	
			}
			for(int k=2;k<len-1;k++){
				g[i][len][0]=(g[i][len][0]+1ll*g[i][k][0]*dp[i+k][len-k])%mo;
			}
		}
	}
	printf("%d\n",g[1][n][0]);
	return 0;
} 

T3 [CSP-S 2021] 回文

刚开始没有想法,考虑第一次操作选左边还是右边(肯定优先选左边)
考虑第一次选的是左边
这个时候最后一个取出的数的位置也就确定,整个序列分成了两段,分别看成一个双端队列,能取出的情况就是两个其中一个的栈顶元素和两个其中一个的栈尾元素相同,这里就直接贪心了,也能保证正确率

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e6+5;
int readint(){
	int x=0,f=1;char s=getchar();
	#define sc (s=getchar())
	while(s<'0'||s>'9'){
		if(s=='-')
			f=-1;
		sc;
	}
	while(s>='0'&&s<='9'){
		x=(x<<3)+(x<<1)+(s^48);
		sc;
	}
	#undef sc
	return x*f;
}
int a[maxn];
char ans[maxn];
bool check(int n){
	ans[1]='L';
	ans[2*n]='L';
	deque<int> q1,q2;
	bool flag=0;
	for(int i=2;i<=2*n;i++){
		if(a[i]==a[1]){
			flag=1;
			continue;
		}
		if(!flag)
			q1.push_back(a[i]);
		else
			q2.push_front(a[i]);
	}
	int now=1;
	while(!q1.empty()||!q2.empty()){
		now++;
//		cout<<"fuck"<<"    "<<now<<endl;
//		cout<<"q1:"<<q1.front()<<" "<<q1.back()<<endl;
//		cout<<"q2:"<<q2.front()<<" "<<q2.back()<<endl;
		if(q1.size()>=2&&q1.front()==q1.back()){
			ans[now]='L',ans[2*n-now+1]='L';
//			cout<<"1"<<q1.front()<<endl;
			q1.pop_front();
			q1.pop_back();
			continue;
		}
		if(!q1.empty()&&!q2.empty()&&q1.front()==q2.back()){
			ans[now]='L',ans[2*n-now+1]='R';
			q1.pop_front();q2.pop_back();
			continue;
		}
		if(!q1.empty()&&!q2.empty()&&q2.front()==q1.back()){
			ans[now]='R',ans[2*n-now+1]='L';
//			cout<<"3"<<q1.back()<<endl;
			q2.pop_front();q1.pop_back();
			continue;
		}
		if(q2.size()>=2&&q2.front()==q2.back()){
			ans[now]='R',ans[2*n-now+1]='R';
//			cout<<"4"<<q2.back()<<endl;
			q2.pop_front();q2.pop_back();
			continue;
		}
		break;
	}
	if(now==n)
		return 1;
	return 0;	
}
bool check1(int n){
	ans[1]='R';
	ans[2*n]='L';
	bool flag=0;
	deque<int> q1,q2;
	for(int i=1;i<2*n;i++){
		if(a[i]==a[2*n]){
			flag=1;
			continue;
		}
		if(!flag)
			q1.emplace_back(a[i]);
		else
			q2.emplace_front(a[i]);
	}
	int now=1;
	while((!q1.empty())||(!q2.empty())){
		now++;
//		cout<<"fuck"<<"    "<<now<<endl;
//		cout<<"q1:"<<q1.front()<<" "<<q1.back()<<endl;
//		cout<<"q2:"<<q2.front()<<" "<<q2.back()<<endl;
		if(q1.size()>=2&&q1.front()==q1.back()){
			ans[now]='L',ans[2*n-now+1]='L';
//			cout<<"1"<<q1.front()<<endl;
			q1.pop_front();
			q1.pop_back();
			continue;
		}
		if(!q1.empty()&&!q2.empty()&&q1.front()==q2.back()){
			ans[now]='L',ans[2*n-now+1]='R';
			q1.pop_front();q2.pop_back();
			continue;
		}
		if(!q1.empty()&&!q2.empty()&&q2.front()==q1.back()){
			ans[now]='R',ans[2*n-now+1]='L';
//			cout<<"3"<<q1.back()<<endl;
			q2.pop_front();q1.pop_back();
			continue;
		}
		if(q2.size()>=2&&q2.front()==q2.back()){
			ans[now]='R',ans[2*n-now+1]='R';
//			cout<<"4"<<q2.back()<<endl;
			q2.pop_front();q2.pop_back();
			continue;
		}
		break;
	}
	if(now==n)
		return 1;
	return 0;	
}
int main (){
	int t=readint();
	while(t--){
		int n=readint();
		for(int i=1;i<=n*2;i++)
			a[i]=readint();
		if(check(n)){
			for(int i=1;i<=2*n;i++)
				printf("%c",ans[i]);
			puts("");
		}
		else if(check1(n)){
			for(int i=1;i<=2*n;i++)
				printf("%c",ans[i]);
			puts("");
		}
		else
			puts("-1");
	}
	return 0;
}
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值