A. Short Substrings
题解:按题意模拟即可
/*Keep on going Never give up*/ #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc++.h> const int maxn = 3100; const int MaxN = 0x3f3f3f3f; const int MinN = 0xc0c0c00c; typedef long long ll; const int mod = 1000007; using namespace std; string s; int main() { int t; cin>>t; while(t--){ cin>>s; if(s.size()==2){ cout<<s<<endl; continue ; } cout<<s[0]; for(int i=1;i<s.size()-1;i+=2){ cout<<s[i]; } cout<<s[s.size()-1]<<endl; } return 0; }
B - Even Array
判断一下不符合条件的情况,看看两种不符合条件的情况是否相等,如果相等即可调换,不相等输出-1.
/*Keep on going Never give up*/ #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc++.h> const int maxn = 3100; const int MaxN = 0x3f3f3f3f; const int MinN = 0xc0c0c00c; typedef long long ll; const int mod = 1000007; using namespace std; int a[maxn]; int main() { int t; cin>>t; while(t--){ int n; cin>>n; int cnt=0,cnt1=0; for(int i=0;i<n;i++){ scanf("%d",&a[i]); if(a[i]%2==0&&i%2!=0) cnt++; if(a[i]%2!=0&&i%2==0) cnt1++; } if(cnt!=cnt1) cout<<-1<<endl; else cout<<cnt<<endl; } return 0; }
C - Social Distance
题解:我们把有1的地方扩散一下,也就是假设第i个地方是1,那么i到i+k和i+k是都不可以做人的,那么我们把不能做人的地方用#打个标记即可,然后寻找有0的地方,并且没次在0的地方坐下一个客人,位置要向前移动k个。
/*Keep on going Never give up*/ #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc++.h> const int maxn = 3100; const int MaxN = 0x3f3f3f3f; const int MinN = 0xc0c0c00c; typedef long long ll; const int mod = 1000007; using namespace std; string s; int main() { int t,k,n; cin>>t; while(t--){ cin>>n>>k; cin>>s; for(int i=0;i<n;i++){ if(s[i]=='1'){ for(int j=max(0,i-k);j<=min(n-1,i+k);j++){ s[j]='#'; } } } int ans=0; for(int i=0;i<n;i++){ if(s[i]=='0'){ ans++; i+=k; } } cout<<ans<<endl; } return 0; }
D - Task On The Board
感觉这个题目涉及了一点点思维,但是模拟起来细节有点多。
思路:对于0的位置,字母一定是最大的,对于字母第二大的元素,他只会受第一大字母的影响,同理,第三大字母只受比他大的字母的影响。
所以我们遍历每个位置看看,某个字母是否符合这个位置的b数组的大小
还有一个小细节,就是我们需要看看这个字母最多能匹配多少个位置,如果位置的个数大于当前字母的个数,说明这个字母不可以用。
/*Keep on going Never give up*/ #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc++.h> const int maxn = 3100; const int MaxN = 0x3f3f3f3f; const int MinN = 0xc0c0c00c; typedef long long ll; const int mod = 1000007; using namespace std; char ans[maxn]; string s; int visited[maxn],cnt[maxn]; int a[maxn]; int main() { int t; cin>>t; while(t--){ cin>>s; int n,z=0; cin>>n; memset(cnt,0,sizeof(cnt)); memset(visited,false,sizeof(visited)); char imax='a'; for(int i=0;i<s.size();i++){ cnt[s[i]-'a']++; if(imax<s[i]) imax=s[i]; } for(int i=0;i<n;i++) scanf("%d",&a[i]); for(int i=0;i<n;i++){ if(!a[i]) visited[i]=true,z++; } for(int i=25;i>=0;i--){ if(cnt[i]>=z){ for(int j=0;j<n;j++) if(!a[j]) ans[j]=i+'a'; cnt[i]=0; break; } cnt[i]=0; } for(int i=25;i>=0;i--){ if(cnt[i]!=0){ int x=0; for(int j=0;j<n;j++){ if(!visited[j]){ int sum=0; for(int k=0;k<n;k++){ if(visited[k]&&ans[k]>i+'a'){ sum+=abs(j-k); } } if(sum==a[j]){ visited[j]=true; x++; ans[j]=i+'a'; } } } if(x>cnt[i]){ for(int j=0;j<n;j++){ if(ans[j]==i+'a') visited[j]=false; } } } } for(int i=0;i<n;i++) printf("%c",ans[i]); printf("\n"); } return 0; }
E - Necklace Assembly
做的时候也想到过跟因子有关,可惜太菜了,没想出来。
参考了这个大佬的思路:https://www.bilibili.com/video/BV1xT4y1J7yd?p=2
/*Keep on going Never give up*/ #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc++.h> const int maxn = 3100; const int MaxN = 0x3f3f3f3f; const int MinN = 0xc0c0c00c; typedef long long ll; const int mod = 1000007; using namespace std; string s; int cnt[maxn]; int main() { int t; cin>>t; while(t--){ int n,k; cin>>n>>k; cin>>s; memset(cnt,0,sizeof(cnt)); for(int i=0;i<s.size();i++){ cnt[s[i]-'a']++; } vector<int> v; for(int i=1;i<=k;i++){ if(k%i==0) v.push_back(i); } int ans=0; for(int i=1;i<=n;i++){ for(auto x : v){ if(x%i==0){ ans=max(ans,i); } if(i%x==0){ int dx=i/x; int cnt1=0; for(int j=0;j<26;j++){ cnt1+=cnt[j]/dx; } if(cnt1>=x){ ans=max(ans,i); } } } } cout<<ans<<endl; } return 0; }
F1 - Flying Sort (Easy Version)
本题数据较大,所以要事先离散化一下,为什么可以这样离散化,是因为条件给出了每个数字最多出现一次,所以我们把他按照由小到大的顺序分别代表1-n的数字。
在用dp求出最长上升子序列即可。
/*Keep on going Never give up*/ #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc++.h> const int maxn = 3100; const int MaxN = 0x3f3f3f3f; const int MinN = 0xc0c0c00c; typedef long long ll; const int mod = 1000007; using namespace std; int dp[maxn]; int a[maxn],b[maxn]; map<int,int> mp; int main() { int t; cin>>t; while(t--){ int n; cin>>n; memset(dp,0,sizeof(dp)); mp.clear(); for(int i=0;i<n;i++){ scanf("%d",&a[i]); b[i]=a[i]; mp[b[i]]=i; } sort(b,b+n); for(int i=0;i<n;i++){ a[mp[b[i]]]=i; } int imax=1; for(int i=0;i<n;i++){ dp[a[i]]=dp[a[i]-1]+1; imax=max(imax,dp[a[i]]); } cout<<n-imax<<endl; } return 0; }