题目描述
小涵很喜欢电脑游戏,这些天他正在玩一个叫做《三国》的游戏。
在游戏中,小涵和计算机各执一方,组建各自的军队进行对战。游戏中共有 N N N 位武将( N N N为偶数且不小于 4 4 4),任意两个武将之间有一个“默契值”,表示若此两位武将作为一对组合作战时,该组合的威力有多大。游戏开始前,所有武将都是自由的(称为自由武将,一旦某个自由武将被选中作为某方军队的一员,那么他就不再是自由武将了),换句话说,所谓的自由武将不属于任何一方。
游戏开始,小涵和计算机要从自由武将中挑选武将组成自己的军队,规则如下:小涵先从自由武将中选出一个加入自己的军队,然后计算机也从自由武将中选出一个加入计算机方的军队。接下来一直按照“小涵→计算机→小涵→……”的顺序选择武将,直到所有的武将被双方均分完。然后,程序自动从双方军队中各挑出一对默契值最高的武将组合代表自己的军队进行二对二比武,拥有更高默契值的一对武将组合获胜,表示两军交战,拥有获胜武将组合的一方获胜。
已知计算机一方选择武将的原则是尽量破坏对手下一步将形成的最强组合,它采取的具体策略如下:任何时刻,轮到计算机挑选时,它会尝试将对手军队中的每个武将与当前每个自由武将进行一一配对,找出所有配对中默契值最高的那对武将组合,并将该组合中的自由武将选入自己的军队。 下面举例说明计算机的选将策略,例如,游戏中一共有 6 6 6个武将,他们相互之间的默契值如下表所示:
双方选将过程如下所示:
小涵想知道,如果计算机在一局游戏中始终坚持上面这个策略,那么自己有没有可能必胜?如果有,在所有可能的胜利结局中,自己那对用于比武的武将组合的默契值最大是多少?
假设整个游戏过程中,对战双方任何时候均能看到自由武将队中的武将和对方军队的武将。为了简化问题,保证对于不同的武将组合,其默契值均不相同。
输入格式
共 N 行。
第一行为一个偶数 N N N,表示武将的个数。
第 $2 $行到第 $N 行 里 , 第 行里,第 行里,第i+1 行 有 行有 行有N_i 个 非 负 整 数 , 每 两 个 数 之 间 用 一 个 空 格 隔 开 , 表 示 个非负整数,每两个数之间用一个空格隔开,表示 个非负整数,每两个数之间用一个空格隔开,表示 i 号 武 将 和 号武将和 号武将和 i+1,i+2,…,N $号武将之间的默契值( 0 ≤ 0≤ 0≤默契值 ≤ 1 , 000 , 000 , 000 ≤1,000,000,000 ≤1,000,000,000)。
输出格式
共 1 1 1 或 $2 $行。
若对于给定的游戏输入,存在可以让小涵获胜的选将顺序,则输出$ 1$,并另起一行输出所有获胜的情况中,小涵最终选出的武将组合的最大默契值。如果不存在可以让小涵获胜的选将顺序,则输出 0 0 0。
样例 #1
样例输入 #1
6
5 28 16 29 27
23 3 20 1
8 32 26
33 11
12
样例输出 #1
1
32
样例 #2
样例输入 #2
8
42 24 10 29 27 12 58
31 8 16 26 80 6
25 3 36 11 5
33 20 17 13
15 77 9
4 50
19
样例输出 #2
1
77
提示
【数据范围】
对于$ 40%$的数据有 N ≤ 10 N≤10 N≤10。
对于$ 70% 的 数 据 有 的数据有 的数据有 N≤18$。
对于 100 % 100\% 100%的数据有 N ≤ 500 N≤500 N≤500。
解题思路:
博弈论题目的典型特征:只有两个玩家,游戏规则固定,游戏里面有可变的数据,询问你是否必胜
由此可以看出这是一道博弈论类型的题目
博弈论类型的题目,不要尝试去用代码模拟过程,而是要去分析获胜的条件、先手的影响,接下来对本题进行分析
我们先说出结论,然后对其进行证明
小涵是必胜的,证明如下:
小涵每拿到一名武将,与其最默契的那位就会被计算机取走,所以小涵不可能拿到最优组合
那么小涵能拿到的最大值一定是次优解中的最大值,拿到该值的过程如下
小涵先取走次优解中的一个,计算机会取走与之匹配的最默契的武将,然后小涵就能拿到次优解了
简单的分析一下就能知道,如果存在比次优解更大的默契值,那么一定是最优组合中的一种
但之后计算机无论选择哪一种最优组合,我们都可以去抢夺另一半
至此,证明完毕,以下是代码的实现
#include <iostream>
#include <algorithm>
using namespace std;
int generals[501][501] = { 0 };
int main() {
int N, value, max_value;
cin >> N;
for (int i = 1; i < N; i++)
for (int j = i + 1; j <= N; j++) {//读入武将数据
cin >> value;
generals[i][j] = value;
generals[j][i] = value;
}
for (int i = 1; i <= N; i++)//排序,便于找出每一个武将对应的最优解和次优解
sort(generals[i] + 1, generals[i] + N + 1);
max_value = generals[1][N - 1];
for (int i = 2; i <= N; i++) {
max_value < generals[i][N - 1] ? max_value = generals[i][N - 1] : max_value = max_value;
}
cout << 1 << '\n' << max_value << endl;
return 0;
}