首页 > 编程语言 >田忌赛马的算法逻辑

田忌赛马的算法逻辑

时间:2024-02-06 16:55:39浏览次数:28  
标签:逻辑 田忌赛马 int 算法 田忌 齐王 sizeof nums1 nums2

田忌赛马的故事大家应该都听说过:

田忌和齐王赛马,两人的马分上中下三等,如果同等级的马对应着比赛,田忌赢不了齐王。但是田忌遇到了孙膑,孙膑就教他用自己的下等马对齐王的上等马,再用自己的上等马对齐王的中等马,最后用自己的中等马对齐王的下等马,结果三局两胜,田忌赢了。

当然,这段历史也挺有意思的,那个讽齐王纳谏,自恋的不行的邹忌和田忌是同一时期的人,他俩后来就杠上了。不过这是题外话,我们这里就打住。

以前学到田忌赛马课文的时,我就在想,如果不是三匹马比赛,而是一百匹马比赛,孙膑还能不能合理地安排比赛的顺序,赢得齐王呢?

当时没想出什么好的点子,只觉得这里面最核心问题是要尽可能让自己占便宜,让对方吃亏。总结来说就是,打得过就打,打不过就拿自己的垃圾和对方的精锐互换。

不过,我一直没具体把这个思路实现出来,直到最近刷到力扣第 870 题「优势洗牌」,一眼就发现这是田忌赛马问题的加强版:

给你输入两个长度相等的数组 nums1 和 nums2,请你重新组织 nums1 中元素的位置,使得 nums1 的「优势」最大化。

如果 nums1[i] > nums2[i],就是说 nums1 在索引 i 上对 nums2[i] 有「优势」。优势最大化也就是说让你重新组织 nums1,尽可能多的让 nums1[i] > nums2[i]。

比如输入:

nums1 = [12,24,8,32]
nums2 = [13,25,32,11]

你的算法应该返回 [24,32,8,12],因为这样排列 nums1 的话有三个元素都有「优势」。

这就像田忌赛马的情景,nums1 就是田忌的马,nums2 就是齐王的马,数组中的元素就是马的战斗力,你就是孙膑,展示你真正的技术吧。

仔细想想,这个题的解法还是有点扑朔迷离的。什么时候应该放弃抵抗去送人头,什么时候应该硬刚?这里面应该有一种算法策略来最大化「优势」。

送人头一定是迫不得已而为之的权宜之计,否则隔壁田忌就会以为你是齐王买来的演员。只有田忌的上等马比不过齐王的上等马时,才会用下等马去和齐王的上等马互换。

对于比较复杂的问题,可以尝试从特殊情况考虑。

你想,谁应该去应对齐王最快的马?肯定是田忌最快的那匹马,我们简称一号选手。

如果田忌的一号选手比不过齐王的一号选手,那其他马肯定是白给了,显然这种情况应该用田忌垫底的马去送人头,降低己方损失,保存实力,增加接下来比赛的胜率。

但如果田忌的一号选手能比得过齐王的一号选手,那就和齐王硬刚好了,反正这把田忌可以赢。

你也许说,这种情况下说不定田忌的二号选手也能干得过齐王的一号选手?如果可以的话,让二号选手去对决齐王的一号选手,不是更节约?

就好比,如果考 60 分就能过的话,何必考 90 分?每多考一分就亏一分,刚刚好卡在 60 分是最划算的。

这种节约的策略是没问题的,但是没有必要。这也是本题有趣的地方,需要绕个脑筋急转弯:

我们暂且把田忌的一号选手称为 T1,二号选手称为 T2,齐王的一号选手称为 Q1。

如果 T2 能赢 Q1,你试图保存己方实力,让 T2 去战 Q1,把 T1 留着是为了对付谁?

显然,你担心齐王还有战力大于 T2 的马,可以让 T1 去对付。

但是你仔细想想,现在 T2 已经是可以战胜 Q1 的,Q1 可是齐王的最快的马耶,齐王剩下的那些马里,怎么可能还有比 T2 更强的马?

所以,没必要节约,最后我们得出的策略就是:

将齐王和田忌的马按照战斗力排序,然后按照排名一一对比。如果田忌的马能赢,那就比赛,如果赢不了,那就换个垫底的来送人头,保存实力。

上述思路的代码逻辑如下:

#include <stdio.h>
#include <stdlib.h>

// 比较函数,用于qsort排序
int compare(const void* a, const void* b) {
return (*(int*)a - *(int*)b);
}

int main() {
// 定义田忌和齐王各自的马的速度数组
int nums1[] = { 4, 6, 2, 8, 5 }; // 田忌的马
int nums2[] = { 5, 7, 1, 9, 3 }; // 齐王的马
int n = sizeof(nums1) / sizeof(nums1[0]); // 计算数组长度

// 对nums1和nums2进行排序
qsort(nums1, n, sizeof(int), compare); // 使用qsort进行快速排序
qsort(nums2, n, sizeof(int), compare); // 使用qsort进行快速排序

// 从最快的马开始比
for (int i = n - 1; i >= 0; i--) {
if (nums1[i] > nums2[i]) {
//比得过,跟他比
   
}
else {
//比不过,换个垫底的来送人头

}
}

return 0;
}

根据这个思路,我们需要对两个数组排序,但是 nums2 中元素的顺序不能改变,因为计算结果的顺序依赖 nums2 的顺序,所以不能直接对 nums2 进行排序,而是利用其他数据结构来辅助。

同时,最终的解法还用到前文 双指针技巧汇总 总结的双指针算法模板,用以处理「送人头」的情况:

#include <stdio.h>
#include <stdlib.h>

typedef struct {
int index;
int value;
} Pair;

int compare(const void* a, const void* b) {
return ((Pair*)a)->value - ((Pair*)b)->value;
}

int* advantageCount(int* nums1, int nums1Size, int* nums2, int nums2Size, int* returnSize) {
// 给 nums2 降序排序
Pair* maxpq = (Pair*)malloc(nums2Size * sizeof(Pair));
for (int i = 0; i < nums2Size; i++) {
maxpq[i].index = i;
maxpq[i].value = nums2[i];
}
qsort(maxpq, nums2Size, sizeof(Pair), compare);

// 给 nums1 升序排序
qsort(nums1, nums1Size, sizeof(int), compare);

// nums1[left] 是最小值,nums1[right] 是最大值
int left = 0, right = nums1Size - 1;
int* res = (int*)malloc(nums2Size * sizeof(int));

for (int i = 0; i < nums2Size; i++) {
int index = maxpq[i].index;
int maxval = maxpq[i].value;
if (maxval < nums1[right]) {
// 如果 nums1[right] 能胜过 maxval,那就自己上
res[index] = nums1[right];
right--;
}
else {
// 否则用最小值混一下,养精蓄锐
res[index] = nums1[left];
left++;
}
}

*returnSize = nums2Size;
return res;
}

int main() {
int nums1[] = { 4, 6, 2, 8, 5 }; // 田忌的马
int nums2[] = { 5, 7, 1, 9, 3 }; // 齐王的马
int nums1Size = sizeof(nums1) / sizeof(nums1[0]);
int nums2Size = sizeof(nums2) / sizeof(nums2[0]);
int returnSize;

int* res = advantageCount(nums1, nums1Size, nums2, nums2Size, &returnSize);

printf("结果数组:");
for (int i = 0; i < returnSize; i++) {
printf("%d ", res[i]);
}
printf("\n");

free(res);

return 0;
}

算法的时间复杂度很好分析,也就是二叉堆和排序的复杂度 O(nlogn)。

至此,这道田忌赛马的题就解决了,其代码实现上用到了双指针技巧,从最快的马开始,比得过就比,比不过就送,这样就能对任意数量的马求取一个最优的比赛策略了。

标签:逻辑,田忌赛马,int,算法,田忌,齐王,sizeof,nums1,nums2
From: https://www.cnblogs.com/lulixiu1999/p/18009989

相关文章

  • 今天回顾-回溯算法-131. 分割回文串
    注意点&感悟:对这个范围还是没太梳理清楚,我的感觉是当前位置是start_index,每轮的结束是i所以范围是[start_index,i+1]题目链接:131.分割回文串自己独立写的代码:classSolution:defpartition(self,s:str)->List[List[str]]:res=[]self.back......
  • Kruskal算法
    问题描述有一张\(n\)个顶点、\(m\)条边的无向图,且是连通图,求最小生成树。算法简析\(Kruskal\)是一种求最小生成树的算法。设该图为\(G=(V,E)\)。最小生成树即所求为\(G_T=(V_T,E_T)\),因为图是连通的,所以最小生成树会覆盖所有的顶点,即\(V==V_T\)。\(G_T\)的真......
  • 代码随想录算法训练营第十三天 | 59.螺旋矩阵II 209.长度最小的子数组 977.有序数
    977.有序数组的平方 已解答简单 相关标签相关企业 给你一个按 非递减顺序 排序的整数数组 nums,返回 每个数字的平方 组成的新数组,要求也按 非递减顺序 排序。 示例1:输入:nums=[-4,-1,0,3,10]输出:[0,1,9,16,100]解释:平方后,数组变为[16......
  • 2024牛客寒假算法基础集训营1
    2024牛客寒假算法基础集训营1A.DFS搜索思路:看dfs其实就是一道字符题目#include<bits/stdc++.h>usingnamespacestd;stringx="dfs";stringy="DFS";voidsolve(){intn;cin>>n;stringst;cin>>st;intxx=0,yy=......
  • 文心一言 VS 讯飞星火 VS chatgpt (195)-- 算法导论14.3 3题
    三、用go语言,请给出一个有效的算法,对一个给定的区间i,返回一个与i重叠且具有最小低端点的区间;或者当这样的区间不存在时返回T.nil。文心一言,去掉math,代码正常运行:在Go语言中,你可以使用结构体来表示区间,并使用切片来存储区间集合。然后,你可以遍历区间集合,找到与给定区间重叠且具......
  • R语言逻辑回归、决策树、随机森林、神经网络预测患者心脏病数据混淆矩阵可视化
    全文链接:https://tecdat.cn/?p=33760原文出处:拓端数据部落公众号概述:众所周知,心脏疾病是目前全球最主要的死因。开发一个能够预测患者心脏疾病存在的计算系统将显著降低死亡率并大幅降低医疗保健成本。机器学习在全球许多领域中被广泛应用,尤其在医疗行业中越来越受欢迎。机器......
  • 【算法】位运算
    常见的位运算操作(操作对象为正整数时)1.按位与(&):当两个数全为1才为1,否则为0。两个数字做与运算,结果不会变大。2.按位或(|):当两个数中有一个1为1,否则为0。两个数字做或运算,结果不会变小。3.按位异或(^):两个数不同为1,相同为0.两个数字做异或运算,结果可能变大,也可能变小。异或的......
  • Prim算法
    问题描述有一张\(n\)个顶点、\(m\)条边的无向图,且是连通图,求最小生成树。Prim算法简析\(Prim\)算法是一种求最小生成树的算法。设该图为\(G=(V,E)\)。最小生成树即所求为\(G_T=(V_T,E_T)\),因为图是连通的,所以最小生成树会覆盖所有的顶点,即\(V==V_T\)。\(G_T\)......
  • C++编程练习||实现分数类Fraction1、实现分数的+,-,*,/ 2、逻辑运算==、!=、<、<=、>、>
    题目:实现分数类Fraction  classFraction{   intnumerator,denominator;   public:   ....  };  要求:1、实现分数的+,-,*,/2、逻辑运算==、!=、<、<=、>、>=6种运输符号。3、实现输出<<,输入 >>操作符重载。  样例1输入:   12 ......
  • 【算法专题】约数个数
    约数个数约数个数的普通求法首先我们根据数的唯一分解定理,对\(N\)进行分解得:\[N=p_1^{c_1}\timesp_2^{c_2}\timesp_3^{c_3}\times...\timesp_k^{c_k}\]由约数的定义:\(p_1^{c_1}=p_1^{0}\timesp_1^{1}\timesp_1^{2}\times...\timesp_1^{c_1}\)共有\(c_1......