《算法设计手册》面试题解答 第二章:算法分析

简介:

简介:

  《算法设计手册》(The Algorithm Design Manual)课后面试题和解答。包括:未知大小的集合选取k个元素、数据备份方案、寻找数组最小数时赋值语句执行次数的期望、100层大楼抛大理石(抛灯泡)、电子秤找不足量金币、天平找重球、公司合并方案总数、海盗分赃等。

 

2-43.

  从n元集合中取一个k元子集,并要求每个元素概率相等。若n未知又该如何解决?

解答:

  原问题在http://www.cnblogs.com/wuyuegb2312/p/3141292.html#title3已经解答清楚了。

  此文也解决了从未知行数中等概率选择一行的方法,那么如果需要选择k个呢?为了简化起见,限定未知的n>=k。那么先选择前k个元素;对第k+1个元素,以概率k/(k+1)选择之,并以等概率1/k代替已选择的k个元素中的一个。

  这样,这一轮里前k个元素中某一个能留下的概率为1/(k+1) + [k/(k+1)] * [(k-1)/k] = k/(k+1)。那么对于第k+2个元素,类似地,以概率k/(k+2)选择之,并以等概率1/k代替已选择的k个元素中的一个,循环直至所有元素遍历结束。

  (解答启发自http://blog.csdn.net/fall221/article/details/8805733

 

2-44.

  有1000份各不相同的数据项和1000个存储数据的结点,每个结点可以存放3份不同的数据项。制定一个备份方案,使其在某些结点失效时损失最小。当任意3个结点失效时,丢失的数据的期望数值是多少?

解答:

  让所有结点存储内容不完全相同,是基本的。如果令结点n保存数据项n、n+1、n+2(取1000的模),最好情况是3个结点失效时不损失数据(各不连续或者2个连续),最坏情况是3个连续结点失效(n、n+1、n+2),损失n+2这一份数据。这个概率很容易计算,是1/10^(-6)。

  相关延伸阅读:

  RAID思想,保存异或结果,不过我算的概率相同:http://stackoverflow.com/questions/10291690/1000-items-1000-nodes-3-items-per-node-best-replication-scheme-to-minimize-da

  类似的处理:http://blog.csdn.net/quietsound/article/details/9285375

 

2-45.

  从n+1个数中找出最小数时,min = A[i]语句执行次数的期望。其中数组下标A[0...n]。

  (首次赋值min=A[0]计算在内)

解答:

  首先想到的是逆序对。但是通过序列[1,3,2]和[1,2,3]可知,两者赋值次数是一样的,显然与逆序对无关。

  直接从期望入手,记E(n)为n+1个数时的该赋值语句期望执行次数。

  最小的数在末尾A[n]的概率为1/n,这时需要一次赋值。

  最小的数不在末尾A[n],则末尾处不需赋值,增加的赋值数为0。

  这样,有E(n) = E(n-1) +1/n且E(0)=0。其中E(n-1)已经包括最小的数在非A[n]处的所有情况。因此E(n)为

          

  n比较大时,约为lnn。

  (解答来自于:http://nbl.cewit.stonybrook.edu:60128/mediawiki/index.php/TADM2E_2.45

 

2-46.

  你有一栋100层高的大厦和2块大理石。你想找出一个最低楼层,从这个楼层开始,在它和比它更高的楼层上抛下大理石,落地会碎。如果有无限多块大理石,怎么最快找出这一层(抛的最少次数)?如果只有两块呢?(另一种说法:100层楼扔灯泡)

解答:

  无限多块时,很容易想到用折半查找来进行。但是如果使用最少的大理石块又该如何找呢?注意到如果某次抛下时大理石未碎,那么下次仍能够使用。为了只用两块大理石来判断楼层数,由于碎裂的最低楼层是一定的,先做分析:

  假设n是最多需要的测试次数(确定但未知,最坏情况),在第n层扔,如果碎了,由于只剩下一个灯泡,为了确定1至n-1层中哪一层,最坏情况是从1层一直扔到n-1层,共n-1次,加上之前的1次是n次,不超过n;如果未碎,那么上到2n-1层再扔,碎时同样有最坏情况从n+1层扔到2n-2层共n-2次,加上之前的是n次。依次类推,总结出每次测试都是n次,且确定的楼层范围是:n->2n-1->3n-2->...,即n + (n-1) + ... +1>100。求解使不等式成立的最小正整数是n=14。

  (解答来自于:http://stackoverflow.com/questions/6547/two-marbles

 

2-47.

  有10袋金币,其中9袋中每个金币都是10克,另1袋中每个金币都是9克。给你一个能够数字显示重量的电子秤,找出全是9克的金币的那一袋。

解答:

  看到称重量可能会联想到用天平找几堆商品中不足量的一份。可是这里用的是能显示具体数值的电子秤哎!如果还用天平的思想,可太死板了。

  想想权重的概念,解法马上有了:从1号至10号袋中分别取1至10个金币,放到电子秤上。如果10袋全是10克的,那么应该显示550克;而缺几克,那就说明那个袋子里是9克的金币。一次解决,是不是很简单?

  当然,权重也可以用别的方案分配,不过上面是最简单的。

  (参考自:http://nbl.cewit.stonybrook.edu:60128/mediawiki/index.php/TADM2E_2.47

 

2-48.

  8个外表一样的球,其中7个等重,另1个比它们重。用天平在2次的比较后找出这个重球。

解答:

  由于有“2次比较”的提示,这个就简单了:把球分为(1,2,3),(4,5,6),(7,8)三组,第一次称(1,2,3)和(4,5,6),若一组重,那么将3个中取2个可得重的是这2个之一还是其余一个;否则将(7,8)称重。

  如果按这个思路,9个球也能够解决。

 

2-49.

  将n个公司合并成1个公司,一共有几种不同的方式合并?

解答:

  记合并方式f(n)种,f(n)=f(n-1)*g(n),其中g(n)是将n个公司通过一次合并变为n-1个公司的方法。显然g(n)为组合数C(n,2)。递推公式解得

  

  (参考自http://nbl.cewit.stonybrook.edu:60128/mediawiki/index.php/TADM2E_2.49

 

2-50.

  Rramanujan数是可以写作a^3+b^3=c^3+d^3的数,其中a!=c且b!=d,或者a!=d且b!=c。找出满足所有给定小于n的Rramanujan数。

解答:

  编程实现如下

复制代码
#include<stdio.h>
int main() {
    int n, i, count=0, j, k, int_count;
    printf("Enter the number of values needed: ");
    scanf("%d", &n);
    i = 1;
    while(count < n) {
       int_count = 0;
       for (j=1; j<=((int(pow(i, 1.0/3)); j++) {
          for(k=j+1; k<=((int(pow(i,1.0/3)); k++) {
              if(j*j*j+k*k*k == i)
              int_count++;
          }
       }
       if(int_count == 2) {
          count++;
          printf("\nGot %d Hardy-Ramanujan numbers %d", count, i);  
       }
       i++;
    }
}
复制代码

  注意几点以加速搜索:

  确保j<k,这样另一对j、k与原来的必然互不相等;

  两个for循环终止条件是j^3<i。

  (参考自:http://stackoverflow.com/questions/11410798/finding-hardy-ramanujan-numbers

 

2-51.

  6个海盗分300美刀,分赃方法如下:有由资历最深的提出分配方案,进行投票。如果获得半数及以上支持,那么按照他的分配方案进行;否则,其他海盗杀掉他,资历第二深的海盗来提出分配方案,其后仍要投票表决,依次类推。请分析分配方法。(所有海盗都以自己存活为最优考虑,保证存活的条件下才尽可能希望多获得美刀。)

解答:

  将海盗按资历有深到浅编号为1,2,3,4,5,6。假设只剩下5,6两个海盗时,5无论提出怎样的分配方案,都能使支持过半(自己的一票),那么分配方法必然是:

  (5,6) ---->(300,0)

  时间倒流,加入海盗4。海盗4为了防止投票不过半而被杀死,需要拉拢海盗6。由于6在上一种情况里什么也分不到,这时只要4分给6,无论多少,6都会同意。这样,4给他最少的数目,即1。而5希望进入下一轮投票,因此只有给他全部300刀才能拉拢。由于投票已过半(2/3),不必拉拢5。此时分配方案为:

  (4,5,6) ---->(299,0,1)

  加入海盗3时,出于类似的考虑,提出这样的方案:

  (3,4,5,6) ---->(299,0,1,0),这样能拉拢5,不至于3死后,5什么也分不到。

  (299,1,0,0)必然不能拉拢4,因为将3干掉后4能获得299。至于为什么不用(299,0,0,1)这样的方案呢?此时6无论是否同意,他都有机会在本轮获得1美刀或者把3干掉后的下一轮必然获得1美刀。因此3的幸存与否是未知的,他不能做出这种使自己生命可能受到威胁的分配方式。后面的分析类似。有:

  (2,3,4,5,6) --->(298,0,1,0,1)

  (1,2,3,4,5,6) ----> (298,0,1,0,1,0)

  (参考自:http://nbl.cewit.stonybrook.edu:60128/mediawiki/index.php/TADM2E_2.51,这里我补充了不少分析。)

 

2.52.

  如果2.51中只有1美刀来分赃,谁会得到它,在这个分配过程中会死多少人?

解答:

  和上一题分析类似,当只有2个海盗时,年长那个一定能得到那1美刀;当海盗数大于等于3时,任何分配给自己的分配方案都不能保证自己一定能存活,不如直接分给倒数第二个海盗。(但即使如此,也可能被杀死。投票将演化成杀死一部分海盗,然后让倒数第二个海盗获得的过程。)

  (更一般的情况请参考http://en.wikipedia.org/wiki/Pirate_game。注意,中文wiki上没有详细的扩展情况的介绍。)

 








本文转自五岳博客园博客,原文链接:www.cnblogs.com/wuyuegb2312/p/3258670.html,如需转载请自行联系原作者

目录
相关文章
|
3月前
|
机器学习/深度学习 算法 搜索推荐
从理论到实践,Python算法复杂度分析一站式教程,助你轻松驾驭大数据挑战!
【10月更文挑战第4天】在大数据时代,算法效率至关重要。本文从理论入手,介绍时间复杂度和空间复杂度两个核心概念,并通过冒泡排序和快速排序的Python实现详细分析其复杂度。冒泡排序的时间复杂度为O(n^2),空间复杂度为O(1);快速排序平均时间复杂度为O(n log n),空间复杂度为O(log n)。文章还介绍了算法选择、分而治之及空间换时间等优化策略,帮助你在大数据挑战中游刃有余。
95 3
|
12天前
|
Java 数据库连接 Maven
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
自动装配是现在面试中常考的一道面试题。本文基于最新的 SpringBoot 3.3.3 版本的源码来分析自动装配的原理,并在文未说明了SpringBoot2和SpringBoot3的自动装配源码中区别,以及面试回答的拿分核心话术。
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
|
3天前
|
存储 算法 安全
基于哈希表的文件共享平台 C++ 算法实现与分析
在数字化时代,文件共享平台不可或缺。本文探讨哈希表在文件共享中的应用,包括原理、优势及C++实现。哈希表通过键值对快速访问文件元数据(如文件名、大小、位置等),查找时间复杂度为O(1),显著提升查找速度和用户体验。代码示例展示了文件上传和搜索功能,实际应用中需解决哈希冲突、动态扩容和线程安全等问题,以优化性能。
|
12天前
|
缓存 算法 搜索推荐
Java中的算法优化与复杂度分析
在Java开发中,理解和优化算法的时间复杂度和空间复杂度是提升程序性能的关键。通过合理选择数据结构、避免重复计算、应用分治法等策略,可以显著提高算法效率。在实际开发中,应该根据具体需求和场景,选择合适的优化方法,从而编写出高效、可靠的代码。
25 6
|
2月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
65 1
|
3月前
|
算法 Java 数据库
美团面试:百亿级分片,如何设计基因算法?
40岁老架构师尼恩分享分库分表的基因算法设计,涵盖分片键选择、水平拆分策略及基因法优化查询效率等内容,助力面试者应对大厂技术面试,提高架构设计能力。
美团面试:百亿级分片,如何设计基因算法?
|
3月前
|
算法 前端开发 Java
数据结构与算法学习四:单链表面试题,新浪、腾讯【有难度】、百度面试题
这篇文章总结了单链表的常见面试题,并提供了详细的问题分析、思路分析以及Java代码实现,包括求单链表中有效节点的个数、查找单链表中的倒数第k个节点、单链表的反转以及从尾到头打印单链表等题目。
39 1
数据结构与算法学习四:单链表面试题,新浪、腾讯【有难度】、百度面试题
|
3月前
|
并行计算 算法 IDE
【灵码助力Cuda算法分析】分析共享内存的矩阵乘法优化
本文介绍了如何利用通义灵码在Visual Studio 2022中对基于CUDA的共享内存矩阵乘法优化代码进行深入分析。文章从整体程序结构入手,逐步深入到线程调度、矩阵分块、循环展开等关键细节,最后通过带入具体值的方式进一步解析复杂循环逻辑,展示了通义灵码在辅助理解和优化CUDA编程中的强大功能。
|
3月前
|
机器学习/深度学习 算法 Java
机器学习、基础算法、python常见面试题必知必答系列大全:(面试问题持续更新)
机器学习、基础算法、python常见面试题必知必答系列大全:(面试问题持续更新)
|
3月前
|
算法 Java 数据库
美团面试:百亿级分片,如何设计基因算法?
40岁老架构师尼恩在读者群中分享了关于分库分表的基因算法设计,旨在帮助大家应对一线互联网企业的面试题。文章详细介绍了分库分表的背景、分片键的设计目标和建议,以及基因法的具体应用和优缺点。通过系统化的梳理,帮助读者提升架构、设计和开发水平,顺利通过面试。
美团面试:百亿级分片,如何设计基因算法?

热门文章

最新文章