在此特别感谢牛客技术和各大高校对本次比赛的大力支持~🤩🤩🤩🥰🥰🥰
本次比赛整体难度符合预期通过率,没有很明显的锅,所以还是比较成功的比赛。
题目的难度是按照升序排列,所以也算是降低了一层难度。
祝贺各位通过选拔的同学进入寒假集训!!!
以下是出题人写的题解,不会的题可以参考一下,由于个人代码风格习惯,并未格式化~
赛题链接:(0条未读通知) 河南科技学院寒假集训选拔赛_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ (nowcoder.com)
查看密码:20240108
A.ACM?acm?⭐
由于可以重新排列字母,并且不区分大小写,所以只需要统计a和A,c和C,m和M的数量,分别计为a_cnt,c_cnt,m_cnt。然后3个的数的最小值即可,最为签到题,也是很友善。
#include<bits/stdc++.h>
using namespace std;
int main(){
string s;
cin>>s;
int a=0,c=0,m=0;
for(int i=0;i<s.size();i++){
if(s[i]=='a'||s[i]=='A') a++;
else if(s[i]=='c'||s[i]=='C') c++;
else if(s[i]=='m'||s[i]=='M') m++;
}
cout<<min({a,c,m});
return 0;
}
B:Happy or unhappy? 难度:⭐⭐
关键词:快速幂,快速幂运用或__int128_t
我们可以看到数据的范围很大,n最大到了10^18^,而且还有取模,所以让人很容易想到使用快速幂。这道题的思路其实比较简单:总共有m^n^种方案(每个格子都可以从m种颜色中任意取一种),让B神喜欢的方案有C(m,1)(m-1)^(n-1)^种(因为第一个格子选了一种后,第二个格子有m-1种选法,第三个格子有m-1种选法……第n个格子有m-1种选法,所以是(m-1)^(n-1)^,而第一个格子有m种选法,所以根据乘法原理,答案是m^n^-m(m-1)^(n-1)^(得到的答案再取模即可),由于此题取模数特意设置较大,所以可能爆longlong,所以所有相乘的地方需要使用快速幂的加法变换才可以通过.
总时间 复杂度
或者也可以直接使用__int128_t ,时间复杂度,这里由于篇幅限制,不做展示。
参考代码:
#include<bits/stdc++.h>
using namespace std;
#define i64 long long
#define endl "\n"
const i64 mod=100000000003;
i64 ksm_add(i64 a,i64 b){
i64 res=0;
while(b){
if(b&1) res=(res+a)%mod;
a=(a+a)%mod;
b=b>>1;
}
return res;
}
i64 ksm_mul(i64 a,i64 b){
i64 res=1;
while(b){
if(b&1) res=ksm_add(res,a)%mod;
a=ksm_add(a,a)%mod;
b=b>>1;}
return res;
}
void solve(){
i64 n,m;
cin>>n>>m;
i64 ans=ksm_mul(m,n)%mod-ksm_add(m,ksm_mul(m-1,n-1))%mod+mod;
ans=ans%mod;
cout<<ans<<endl;
}
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t=1;cin>>t;
while(t--){
solve();
}
return 0;
}
C:TieTie! 难度:⭐⭐
关键词:单调栈
我们可以假设每个位置都需要贴壁画,那么最多需要n张。对于任意两个位置,如果高度不一样,那么必然需要两张壁画,而对于两个相同高度的位置则只需要同一张。故需要找出第i张前面的前i-1张中是否存在a[j]==a[i](j∈[1,i-1])即可,前提是[j,i)中的山比第i座山都要高才可以,否则必然需要另外一张,因为中途高度降低,无法左右连续。对于存储前i-1个高度,我们可以使用单调栈或者队列,或者数组都可以解决。这样便可以找出重复高度的壁画cnt,最后n-cnt就是答案。
每个山的高度最多被数组和单调栈各遍历一次,所以时间复杂度为
参考代码:
#include<bits/stdc++.h>
using namespace std;
#define i64 long long
int main( )
{
int n;
cin>>n;
vector<i64>w(n+1),h(n+1);
for(int i=1;i<=n;i++){
cin>>w[i]>>h[i];
}
i64 ans=0;
stack<i64>q;
for(int i=1;i<=n;i++){
while(q.size()&&q.top()>=h[i]){
if(q.top()==h[i]) ans++;
q.pop();
}
q.push(h[i]);
}
cout<<n-ans;
return 0;
}
D:Pass the difficulty! 难度:⭐⭐⭐
关键词:BFS,状态更新,最短路
不难看出这是一个BFS的题。只需要从坐标(1,1)开始,然后把(1,1)的初始状态入队列,依次判断其上下左右的状态和当前队首的状态,相同的时候判断+2,不同的时候判断+1,然后进行更新到达(nx,ny)的最短时间,如果下一个位置的最短时间可以更新,则入下一个点的可到达状态(和当前队首相反的状态,因为相反状态才可以到达)。最后终点的最短时间就是终点状态0和1的最小值。
时间复杂度
参考代码:
#include<bits/stdc++.h>
using namespace std;
#define i64 long long
i64 d[1001][1001][2];
void solve(){
int n,m;
cin>>n>>m;
vector<vector<char>>a(n+1,vector<char>(m+1));
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++)
cin>>a[i][j];
}
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
for(int k=0;k<2;k++) d[i][j][k]=1e18;
auto bfs=[&](int x,int y)->void{
queue<array<int,3>>q;
q.push({x,y,a[x][y]-'0'});
d[x][y][a[x][y]-'0']=0;
int z;
while(!q.empty()){
auto t=q.front();
q.pop();
int nx,ny;
x=t[0],y=t[1],z=t[2];
//上
nx=x-1,ny=y;
if(nx>=1&&nx<=n&&ny>=1&&ny<=m){
int D=d[x][y][z]+1+(a[nx][ny]-'0'==z?1:0);
if(D<d[nx][ny][(a[nx][ny]-'0'==z)?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]){
q.push({nx,ny,a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')});
d[nx][ny][a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]=D;
}
}
//下
nx=x+1,ny=y;
if(nx>=1&&nx<=n&&ny>=1&&ny<=m){
int D=d[x][y][z]+1+(a[nx][ny]-'0'==z?1:0);
if(D<d[nx][ny][(a[nx][ny]-'0'==z)?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]){
q.push({nx,ny,a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')});
d[nx][ny][a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]=D;
}
}
//左
nx=x,ny=y-1;
if(nx>=1&&nx<=n&&ny>=1&&ny<=m){
int D=d[x][y][z]+1+(a[nx][ny]-'0'==z?1:0);
if(D<d[nx][ny][(a[nx][ny]-'0'==z)?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]){
q.push({nx,ny,a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')});
d[nx][ny][a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]=D;
}
}
//右
nx=x,ny=y+1;
if(nx>=1&&nx<=n&&ny>=1&&ny<=m){
int D=d[x][y][z]+1+(a[nx][ny]-'0'==z?1:0);
if(D<d[nx][ny][(a[nx][ny]-'0'==z)?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]){
q.push({nx,ny,a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')});
d[nx][ny][a[nx][ny]-'0'==z?(1^(a[nx][ny]-'0')):(a[nx][ny]-'0')]=D;
}
}
}
};
bfs(1,1);
cout<<min(d[n][m][0],d[n][m][1])<<endl;
}
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t=1;
while(t--){
solve();
}
return 0;
}
E:How much time? 难度:⭐⭐⭐
关键词:前缀和,树状数组,离散化,二分
对于区间和
可以转化为sum[r]-sum[l-1]≥x,如果要快速查询以r为结尾的区间有多少满足条件的区间,那么公式可以转化为sum[r]-x≥sum[l-1],这时只需要找出在r的左边 小于等于sum[r]-x的区间前缀和数量即可。由于有负数,所以前缀和不单调,故使用树状数组来维护r左边小于等于sum[r]-x的前缀和数量,统计r左边小于等于sum[r]-x的前缀和出现的次数,同时,前缀和的值可能很大,所以要使用离散化,把数组降为数组可容纳的范围.
总时间复杂度
参考代码:
#include<bits/stdc++.h>
using namespace std;
const int N=1e6+5;
#define i64 long long
#define endl "\n"
int n;
i64 tree[N];
vector<i64>all;
i64 lowbit(i64 x){
return x&-x;
}
void add(int x){
for(int i=x;i<all.size();i+=lowbit(i)) tree[i]++;
}
i64 sum(int x){
i64 ans=0;
for(int i=x;i;i-=lowbit(i)) ans=ans+tree[i];
return ans;
}
i64 find(i64 x){
int l=0,r=all.size()-1;
while(l<r){
int mid=l+r>>1;
if(all[mid]>=x) r=mid;
else l=mid+1;
}
return l+1;
}
void solve(){
i64 x;
cin>>n>>x;
vector<i64>a(n+1),s(n+5);
for(int i=1;i<=n;i++){
cin>>a[i];
s[i]=s[i-1]+a[i];
}
for(int i=0;i<=n;i++){
all.push_back(s[i]-x);
all.push_back(s[i]);
}
sort(all.begin(),all.end());
all.erase(unique(all.begin(),all.end()),all.end());
i64 ans=0;
add(find(0));
for(int i=1;i<=n;i++){
ans=ans+sum(find(s[i]-x));
add(find(s[i]));
}
cout<<ans;
}
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t=1;
while(t--){
solve();
}
return 0;
}
F:Attribute M 难度:⭐⭐⭐
关键词:线性DP
考虑DP。
用数组f[i] [j]表示到第i个数取模之后余数为j的情况。
首先预处理for(int i=1;i<=n;i++) f[i] [a[i]%m]=a[i];
对于每个数只有两种情况,要么不要,要么要。
对于不要的情况可直接f[i] [j]=max(f[i] [j],f[i-1] [j]);
要的情况:分类讨论1.a[i]%m≤j和2.a[i]%m>j的情况。
-
可由上一层的j-a[i]%m最大值转移而来,举例当m=5,a[i]=2,j=4,由上一层的2转移过来,因为(2+2)%5==4,
if(f[i-1] [j-a[i]%m]!=0) f[i] [j]=max({f[i] [j],f[i-1] [j],f[i-1] [j-a[i]%m]+a[i]});
如果f[i-1] [j-a[i]%m]==0,说明上一层的取模数等于j-a[i]%m并没有情况,所以不能更新,只能选择不要此数。
2.可以上一层m+j-a[i]%m转移而来,举例当m=5,a[i]=4,j=2,由上一层的3转移过来,因为(4+3)%5==2,
if(f[i-1] [m+j-a[i]%m]!=0) f[i] [j]=max({f[i] [j],f[i-1] [j],f[i-1] [m+j-a[i]%m]+a[i]});
如果f[i-1] [m+j-a[i]%m]==0,说明上一层的取模数等于m+j-a[i]%m并没有情况,所以不能更新,只能选择不要此数。
时间复杂度
参考代码:
#include<bits/stdc++.h>
using namespace std;
#define i64 long long
#define endl "\n"
void solve(){
int n,m;
cin>>n>>m;
vector<i64>a(n+1);
vector<vector<i64>>f(n+1,vector<i64>(m+1,0));
for(int i=1;i<=n;i++){
cin>>a[i];
}
f[0][0]=0;
for(int i=1;i<=n;i++) f[i][a[i]%m]=a[i];
for(int i=1;i<=n;i++){
for(int j=0;j<m;j++){
if(j>=a[i]%m){
if(f[i-1][j-a[i]%m]!=0)
f[i][j]=max({f[i][j],f[i-1][j],f[i-1][j-a[i]%m]+a[i]});
else
f[i][j]=max(f[i][j],f[i-1][j]);
}
else{
if(f[i-1][m+j-a[i]%m]!=0)
f[i][j]=max({f[i][j],f[i-1][j],f[i-1][m+j-a[i]%m]+a[i]});
else
f[i][j]=max(f[i][j],f[i-1][j]);
}
}
}
cout<<f[n][0];
}
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t=1;
while(t--){
solve();
}
return 0;
}
G:Transporting watermelons 难度:⭐⭐⭐⭐
关键词:凸多边形,外接圆,抽屉原理,FFT
由切线性质可得,一个凸四边形是圆外切四边形当且仅当该四边形的对边之和相等。故问题转化为所给数列中是否存在四个不同的数 a,b,c,d使得a+b=c+d。这是抽屉原理经典模型,暴力枚举即可。作为选拔赛的中等题,为了让大家有一个友好的参赛体验,FFT也可以通过。
时间复杂度
参考代码:
#include<bits/stdc++.h>
using namespace std;
#define i64 long long
void solve(){
int n;
cin>>n;
vector<i64>a(n+1);
for(int i=1;i<=n;i++){
cin>>a[i];
}
map<i64,i64>H;
for(int i=1;i<=n;i++){
for(int j=i+1;j<=n;j++){
if(H[a[i]+a[j]]){
cout<<"YES";
return ;
}
H[a[i]+a[j]]=1;
}
}
cout<<"NO";
}
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t=1;
while(t--){
solve();
}
return 0;
}