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
i,len是以
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;
}