传说是小米家的一道面试题难倒了某Java程序员。扑克牌排序问题。

网上说的是有位网友在面试小米Java岗三次后,终于挺进了第三轮面试,结果还是败在了两道算法题上面。

1、写个读方法和写方法,实现读写锁

2、一副从1到n的牌,每次从牌堆顶取一张放桌子上,再取一张放牌堆底,直到手机没牌,最后桌子上的牌是从1到n有序,设计程序,输入n,输出牌堆的顺序数

(来源:https://zhuanlan.zhihu.com/p/38850888)

第一个问题不重要,这里讨论第二个扑克牌排序的问题。

此题也没有写清楚,说“最后桌子上的牌是从1到n有序”,是从上到下还是从下到上数是1到n呢?我这里暂定为从上到下,因为两个方向的解题思路是一致的,不管哪个方向无所谓。且暂定牌面数值为1到无穷大,而不是A到K。

从上到下就是,初始,牌在手上,成某顺序排列。第一步,从手里牌顶取一张放到桌子上;第二步,从手里牌顶取一张放到手里牌底。就这两个动作一直重复,其中除了第一张,后续放到桌上的牌要叠在桌上牌的上面,也就是不能在桌上乱丢。直到手里牌全部放到了桌子上。其中当手里只剩一张牌,而上一步的动作是把倒数第二张牌从手牌顶移到了桌上,按规则此时应该把手牌顶的牌移到手牌底,而此时手牌顶和手牌底是同一张,所以此动作可以当作最后一张手牌不动,而下一步,又得将手牌顶(也即是手牌底,即是手中最后一张)移到桌上,至此,移动结束。最后要达到的状态就是桌上的牌,从上往下数,第一张是1,第二张是2,……直到n。

  我一开始看到这题,突然感觉好简单,因为牌的数量无所谓,5张排序和500张排序的算法是一样的。想了想,5张而已,多简单啊,这张先在上面,那张又在上面,这张又到下面,感觉逻辑过程很简单。

  但实际在代码实现的过程中就懵了,要对牌量无上限的牌堆做出这样的排序,使得无论牌多牌少,只需要一个算法,几行代码,统统不在话下,在不考虑时间复杂度的情况下。

  所以说计算机的思维过程跟人不一样啊……用计算机来做就是不如想象中那么简单。

  我的实现思路很傻瓜化,跟我以前解数学问题如出一辙,就是先举例子,比如取n等于2的情况,初始顺序(如无特殊说明,均指从上往下的顺序)很容易得到,[2,1],再看n=3,脑算一下就知道是[3,1,2],n=4则对应[4,2,3,1],具体如下:

以下为心算结果:
    #n=1,  [1]
    #n=2,  [2,1]
    #n=3,  [3,1,2]
    #n=4,  [4,2,3,1]
    #n=5,  [5,1,4,2,3]
    #n=6,  [6,3,5,1,4,2]
    #n=7,  [7,2,6,3,5,1,4]

  万变不离其宗啊,我忽然就看到了里面的规律。每个列表的第一个值肯定是n,不用多说;并且每个列表除去最后一个元素,剩下的跟下一个列表的尾部一模一样,比如n=4时候,除去最后一个1,为[4,2,3],而[4,2,3]恰恰是当n=5时候的尾巴;并且暂时除去的1跑到了这个尾巴的前一个位置,即第二位。第一位是n,第二位是前一个的最后一个元素1,剩下的是n=4时列表[4,2,3,1]除去1的[4,2,3]。然后我就笑了,心想这个算法也太sb了吧,像是照葫芦画瓢,没任何技术含量,并且我隐约感觉到此递归算法的时间复杂度是最高的。这很明显,每一个列表都可以通过前面一个推出来,跟斐波那契数列算法是一样的,可以用递归。

  试了一下,Python代码如下:

def cards_on_hand(n):
    if n<=0 or isinstance(n,int)==False:
        print(‘请输入正整数‘)
    if n==1:
        return [1]
    if n>1:
        previous=cards_on_hand(n-1)
        return [n, previous.pop(-1)]+previous

  此函数需引入参数n,即牌的数量。返回值为列表,即手上牌从上往下的顺序成列。

  用jupyter notebook测试结果显示,当n小于等于2965,可秒算,等于2966或更大时,程序报错,“RecursionError: maximum recursion depth exceeded in comparison”。意思大概是递归超出最大深度。当然实际的复杂度测算也是必须的,等下面跟网上其它算法比较的时候再用。

  其它算法,有一个,思想来源于自称“微软工程师”的大哥,有个网友用python初略实现了“巨硬哥”的算法,叫做“逆向法”。我只是感觉这个逆向的原理跟递归是一样的算法。具体如下:

#网友对逆向法的Python代码实现:(这位网友是把最终桌上的牌从下往上数的,也就是最下面是1,最上面是n。但这不影响)
def MiTest2(n):
    result=[]
    for i in range(n):
        result = result[-1:] + result[:-1]
        result = [(n-i)] + result
    return result

  其中的for循环不断改变result的值,直到循环到n,看起来这个逻辑过程跟递归一样一样的。

  也是用jupyter运行一下,看是不是到2965也算封顶了。

  结果令人很满意,我试到了50000,没错,就是五万,差不多n=50000时,花了20秒的时间。但不会报递归错误。

  由此我发现这个逆向法跟递归是有渊源的,其中逆向法的数学原理绝对是妙不可言的,我等一下再分析。

  先把这段网友代码,按从上往下数的顺序再改一下:

def hand_cards(n):
    if n<=0 or isinstance(n,int)==False:
        print(‘请输入正整数‘)
    result=[]
    for i in range(n):
        result = result[-1:] + result[:-1]
        result = [i+1] + result
    return result

改成这个顺序,那最终桌上的牌就是从上往下数,1到n,网友原代码的结果是,从上往下数,n到1。代码中关键就改了for循环中第二行,result = [(n-i)] + result改成了result = [i+1] + result。

很明显,这个“逆向法”(从上往下数)的步骤是,对于任意给定正整数n:

首先生成一个列表[1];

将其最后一位元素移到首位,变成[1],再将下一个2挤入首位,生成第二个列表[2,1];

将其最后一位元素移到首位,变成[1,2],再将下一个3挤入首位,生成[3,1,2];

将其最后一位元素移到首位,变成[2,3,1],再将下一个4挤入首位,生成[4,2,3,1];

重复这两个步骤,直到n。

  

原文地址:https://www.cnblogs.com/oler/p/9648890.html

时间: 2024-11-02 21:46:15

传说是小米家的一道面试题难倒了某Java程序员。扑克牌排序问题。的相关文章

Java程序员面试题集(1-50)

下面的内容是对网上原有的Java面试题集及答案进行了全面修订之后给出的负责任的题目和答案,原来的题目中有很多重复题目和无价值的题目,还有不少的参考答案也是错误的,修改后的Java面试题集参照了JDK最新版本,去掉了EJB 2.x等无用内容,补充了数据结构和算法相关的题目.经典面试编程题.大型网站技术架构.操作系统.数据库.软件测试.设计模式.UML等内容,同时还对很多知识点进行了深入的剖析,例如hashCode方法的设计.垃圾收集的堆和代.Java新的并发编程.NIO.2等,相信对准备入职的Ja

Java程序员面试题收集(6)

<!————————————————————————————基础题122道,代码题19道————————————————————————————> JAVA相关基础知识1.面向对象的特征有哪些方面1.抽象:抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问题,而只是选择其中的一部分,暂时不用部分细节.抽象包括两个方面,一是过程抽象,二是数据抽象.2.继承:继承是一种联结类的层次模型,并且允许和鼓励类的重用,它提供了一种明确表述共性的方法

Java程序员面试题集(86-115)

摘 要:下面的内容包括Struts 2和Hibernate的常见面试题,虽然Struts 2在2013年6月曝出高危漏洞后已经显得江河日下,而Spring MVC的异军突起更加加速了Struts 2的陨落,但面试中仍然有可能被问及和此框架相关的内容,毕竟Struts 2曾经被阿里巴巴.京东以及政府企业门户网站广泛采用.另一方面,Hibernate目前仍然是ORM框架中的中坚力量,MyBatis在此领域也有不容 忽视的一席之地,因此了解这两个ORM框架对Java程序员是很有必要的.第一期发布的Ja

Java程序员面试题收集(3)

面试中被问到过的题目: 1.<%@ include=""/>和<jsp:include page="" flush="true"/> 2.知道的GOF 模式,说明 3.自己的长期规划是什么,近期规划是什么? 4.override和overload 5.final关键字 6.系统如何分层,前台,逻辑,数据访问.你的职责,代码如何保证质量的 7.getAttribute 和getParameter 区别 8.forward和r

Java程序员的10道XML面试题

如今,面对web开发人员的Java各种面试中,XML面试题在各种编程工作的面试中很常见.XML是一种成熟的技术,经常作为从一个平台到其他平台传输数据的标准.XML面试问题包括用于转换XML文件的XSLT技术,以及XPATH,XQuery等各种XML技术和XML基础知识. 笔者从论坛收集看到常见的XML面试问答题.这些问题大部分在Java面试中会问到,同时在C,C++,Scala或其他语言的编程面试中同样很有用处.作为编程人员所需要的技能之一,在任何技术工作面试之前准备一些XML问题是很有意义的.

小米Java程序员第二轮面试10个问题,你是否会被刷掉?

近日,开发者头条上分享了一篇"小米java第二轮面经",有很多的java程序员表示非常有兴趣. 下面l就和各位分享小米java第二轮面经(华为java工程师笔试面试题可以看文章某尾): 0.谈谈对Spring的认识 1.java集合的继承关系 2.你知道CLOSE_WAIT这个状态吗? 3.谈谈你对jvm.java内存模型的认识? 4.Java里的红黑树是怎么实现的? 5.谈一下对strust2的理解 6.IOC最大的好处是什么呢? 7.长连接与短连接有什么区别? 8.HashMap怎

java程序员面试题大全含答案(2018--2019)

java程序员面试题大全含答案(2018--2019) 1.10道经典java面试题_实习生必问! 2.15个Java线程并发面试题和答案 3.15个高级Java多线程面试题及回答 4.2018年java分布式相关最新面试题 5.2018最新java技术面试题与答案 6.4个Spring常见面试题及答案解析 7.css面试题及答案 8.HR常问面试题总结(上) 9.HR常问面试题总结(下) 10.html面试题及答案 11.java中String类的面试题大全含答案 12.java二叉树算法面试

90%的Java程序员不会的10到Java面试题

先来看看这10道Java面试题你会几道?1.为什么等待和通知是在 Object 类而不是 Thread 中声明的?2.为什么Java中不支持多重继承?3.为什么Java不支持运算符重载?4.为什么 String 在 Java 中是不可变的?5.为什么 char 数组比 Java 中的 String 更适合存储密码?6.如何使用双重检查锁定在 Java 中创建线程安全的单例? 编写 Java 程序时, 如何在 Java 中创建死锁并修复它? 如果你的Serializable类包含一个不可序列化的成

Java程序员面试题集(71-85)

Java程序员面试题集(71-85) 摘要:这一部分主要包括了UML(统一建模语言).面向对象的设计原则(六原则一法则).GoF设计模式.企业级设计模式.JDBC(Java数据库连接).XML(可扩展标记语言)等知识. 71.UML是什么?UML中有哪些图? 答:UML是统一建模语言(Unified Modeling Language)的缩写,它发表于1997年,综合了当时已经存在的面向对象的建模语言.方法和过程,是一个支持模型化和软件系统开发的图形化语言,为软件开发的所有阶段提供模型化和可视化