目录
- 1. 班级活动
- 1. 问题描述
- 2. 输入格式
- 3. 输出格式
- 4. 样例输入
- 5. 样例输出
- 6. 样例说明
- 7. 评测用例规模与约定
- 8. 原题链接
- 2. 解题思路
- 3. AC_Code
1. 班级活动
前置知识点:思维,分类讨论
1. 问题描述
小明的老师准备组织一次班级活动。班上一共有 名 ( 为偶数) 同学,老师想把所有的同学进行分组,每两名同学一组。为了公平,老师给每名同学随机分配了一个 以内的正整数作为 ,第 名同学的 为 。
老师希望通过更改若干名同学的 使得对于任意一名同学 ,有且仅有另一名同学 的 与其相同 ()。请问老师最少需要更改多少名同学的 ?
2. 输入格式
输入共
第一行为一个正整数 。
第二行为 个由空格隔开的整数 。
3. 输出格式
输出共
4. 样例输入
4
1 2 2 3
5. 样例输出
1
6. 样例说明
仅需要把 改为 或者把 改为
7. 评测用例规模与约定
对于 的数据,保证 。
对于 的数据,保证 。
8. 原题链接
2. 解题思路
首先明确一点,假设某个 的同学数量为 ,因为题目要求任意 只能有两名同学,所以一定会有 名同学修改自己的 。我们可以计算出每个 需要修改自身的同学数量之和,并将这个数量设为 ,即满足:
还有一个特殊群体我们不能忽略,就是编号 唯一的同学,我们设这群同学的数量为 。他们特殊在有可能需要修改自身 ,也有可能不需要,我们需要进行分类讨论。
- 当
在这种情况下, 唯一的 名同学是不需要修改自身 的。我们可以从 名同学中选出 名同学修改自身 去与 唯一的同学对应,剩下的 名同学仍然是需要修改自身 的,所以答案即是 。
假设有一个 集合 ,此时 唯一的集合为 ,必须修改的 集合为 。我们只需要让后一个集合的 分别修改为
- 当
在这种情况下,部分 唯一的 名同学是需要修改自身 的。同样假设有一个 集合 ,此时 唯一的集合为 ,必须修改的 集合为 。按照同样策略,我们让必须修改的 集合与 唯一的集合对应上,即将必须修改的 集合变为 。
但此时仍然发现 唯一的集合剩余的两个 为 ,我们需要让他们它们一致,所以需要修改其中一个。
假设剩余 个呢?那我们需要修改
假设剩余 个呢?那我们需要修改
显然结论就是需要修改剩余 个数的一半,即这种情况下答案是:
小疑问:如果 a-b 为奇数怎么办?
结论: 一定为偶数。我们可以假设数组已经存在 对匹配好的 ,根据我们对 的定义,显然符合式子 。题目告知我们 一定为偶数,且 也一定为偶数,那么 也一定为偶数,即说明 奇偶性一定相同,得证
时间复杂度:。
3. AC_Code
- C++
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
int n;
int main()
{
cin >> n;
map<int, int> cnt;
for (int i = 0; i < n; ++i)
{
int x;
cin >> x;
cnt[x]++;
}
int a = 0, b = 0;
for (auto [x, y] : cnt)
{
if (y == 1)
{
a++;
}
else if (y > 2)
{
b += y - 2;
}
}
if (b >= a)
{
cout << b << '\n';
}
else
{
cout << (a - b) / 2 + b << '\n';
}
return 0;
}
- Java
import java.util.*;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int n = sc.nextInt();
Map<Integer, Integer> cnt = new HashMap<>();
for (int i = 0; i < n; ++i) {
int x = sc.nextInt();
cnt.put(x, cnt.getOrDefault(x, 0) + 1);
}
int a = 0, b = 0;
for (Map.Entry<Integer, Integer> entry : cnt.entrySet()) {
int y = entry.getValue();
if (y == 1) {
a++;
} else if (y > 2) {
b += y - 2;
}
}
if (b >= a) {
System.out.println(b);
} else {
System.out.println((a - b) / 2 + b);
}
}
}
- Python
n = int(input())
line = list(map(int, input().split()))
cnt = {}
for i in range(n):
x = line[i]
if x in cnt:
cnt[x] += 1
else:
cnt[x] = 1
a = 0
b = 0
for y in cnt.values():
if y == 1:
a += 1
elif y > 2:
b += y - 2
if b >= a:
print(b)
else:
print((a - b) // 2 + b)