面试题:1~ n-1 有n个数,是有序的,找出重复的那个数。
话说这道题有多种解法:
第一种:数学解法
举例
1 2 3 4 = 10
1 2 3 3 = 9
那么我们可以计算出 4 - (10-9) = 3 则3是重复的。
所以我们只要得到 1~ n-1 的和,就可以用高斯公式算出这个结果,但是前提是要知道和,那么遍历数组时间复杂度是O(n)。
如果这个数列不是有序的,这个方法依然通用。
第二种:折半查找法
由于这个数列是有序的,所以用折半查找法是再合适不过了。
array[n/2 -1] == n/2 ?不满足则往前找,满足往后找。时间复杂度是O(logn)
第三种:折X份查找法
折多少份?这个很有趣,其中有一道面试题和这个题目类似。
【题目】 话说有2个杯子,100层楼。 杯子可以在某层摔碎,用最少次测试出哪层(最底层)能摔碎杯子?
这个摔杯子可以从第一层一层层摔,最多100次,也可以2层2层摔,最多50次,也可以3层3层摔,最多33次。还能更快吗?
中午吃饭的时候某同事说可以用二分法的反向思维来做 比如 2->4->8->16->32->64->100 这个跨度越来越大,最多也要30来次。虽然这个方法不是想要的结果,但给了我们一个启示就是 我们可以找一个比较合适的度。 比如直接找某楼摔,不行再回来,行了就可以省去很多层楼,这个思想很重要。
如果我们累加呢? 1 3 6 10 15 21 28 36 45 55 66 78 91 100 这样最多是12+12=24次还能更快吗?
反向过来递减?那么次数最多的则是第一次的那层! 14 27 39 50 60 69 77 84 90 95 99 ... omg 这。。。 还能更快吗?
x+x*x >= 2n 那么,最优的次数是x 。
找重复的那个数和摔杯子不完全一样,因为我们不是2个杯子只有2次摔碎的机会。我们有N次机会。 所以 我们如果测出14摔碎,可以直接 5 9 13-> 3 -> 1 这么摔。

static void getRepeater(int s, int x, int n, int[] array)
{
if (x == 1)
{
Console.Write(array[s - 1]);
Console.Read();
return;
}
if (array[s - 1] != s)
{
n = x;
x = getX(x);
s -= x;
getRepeater(s, x, n, array);
}
else
{
x--;
getRepeater(s + x, x, n - x, array);
}
}
static int getX(int n)
{
var x = 1;
while (x + x * x < 2 * n)
{
x++;
}
return x;
}
static void Main(string[] args)
{
int[] array = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11,11, 12, 13, 14, 15, 16, 17, 18, 19, 20 };
int n = array.Length;
int x = getX(n);
int s = x;
getRepeater(s, x, n, array);
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?