跳到主要内容

题解:P9421 [蓝桥杯 2023 国 B] 班级活动

· 阅读需 2 分钟
lailai
Blogger

原题链接

题意简述

每次修改 11 个元素,求序列中刚好每 22 个元素相同的最小操作次数。

解题思路

  1. 对于每个元素 ii 的出现次数 aia_i,分类讨论:
  • ai=2a_i=2,不用修改。
  • ai=1a_i=1,可以直接修改其他元素,也可以把其他元素修改为 ii
  • ai>2a_i>2,修改多余的 ai2a_i-2 个元素,且优先修改成其他 aj=1a_j=1 的元素。
  1. 代码实现:
  • 读入 tt,记录每个数出现的次数 aia_i
  • 枚举所有的 aia_i
{s1s1+1ai=1s2s2+(ai2)ai>2\begin{cases} s_1\gets s_1+1 & a_i=1 \\ s_2\gets s_2+(a_i-2) & a_i>2 \end{cases}
  • 如果 s2s1s_2\ge s_1,说明有足够 ai>2a_i>2 的元素可以给 ai=1a_i=1 的元素;
  • 否则,还要修改 s1s2s1-s2ai=1a_i=1 的元素,每 22 个一组,要修改 s1s22\frac{s1-s2}{2} 个元素。
  1. 因此,最终答案为 s2+max(s1s2,0)2s2+\frac{\max(s1-s2,0)}{2}

参考代码

#include <bits/stdc++.h>
using namespace std;

const int N=100005;
int a[N];
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n;
cin>>n;
for(int i=1;i<=n;i++)
{
int t;
cin>>t;
a[t]++;
}
int s1=0,s2=0;
for(int i=1;i<=n;i++)
{
if(a[i]==1)
{
s1+=1;
}
else if(a[i]>2)
{
s2+=a[i]-2;
}
}
cout<<s2+max(s1-s2,0)/2<<'\n';
return 0;
}