使用递归实现买汽水(华为面试题)

今天老范问了我一个问题问题:一个人买汽水,一块钱一瓶汽水,三个瓶盖可以换一瓶汽水,两个空瓶可以换一瓶汽水问20块钱可以买多少汽水?注意:使用递归这一题乍一看,哎哟,这么简单,能买几瓶?恩。。五瓶!为啥啊?多了我喝不完啊!老范说,喝不完关你屁事,又不是给你喝哦哦哦,那没事儿了,我想想。在知道自己的人生安全得到了保障之后,我冷静下来仔细思考了如何用递归实现这个问题

大家好,又见面了,我是你们的朋友全栈君。

今天老范问了我一个问题
问题:
一个人买汽水,一块钱一瓶汽水,三个瓶盖可以换一瓶汽水,两个空瓶可以换一瓶汽水
问20块钱可以买多少汽水?
注意:使用递归

这一题乍一看,哎哟,这么简单,能买几瓶?恩。。
五瓶!
为啥啊?

多了我喝不完啊!

老范说,喝不完关你屁事,又不是给你喝

哦哦哦,那没事儿了,我想想。
在知道自己的人生安全得到了保障之后,我冷静下来仔细思考了如何用递归实现这个问题

首先想了想什么是递归

  1. 方法内调用自己的方法的现象称为递归调用
  2. 递归现象允许程序执行到某个阶段时整体调用重新来过

以及递归的注意事项:

  1. 方法内部调用自己的方法不能100%成立,否则就是死循环
  2. 递归层数尽量少,因为会消耗内存,运行效率低

不能100%成立也就是说,要有终止条件,达成某个条件,就要跳出递归调用

瓶盖和空瓶子换饮料,这是一个怎么样的过程?了解了这个过程,才能分析出这中间哪些变量在发生变化,转换,以及变化到哪个程度,就不满足继续递归的条件了,也就是满足终止条件了。

1块钱和2块钱的置换过程
(这是用processon画的https://www.processon.com/很好用)
(推荐一波,processon速打钱来!!!!!)
drink表示购买和置换的饮料
cup表示瓶盖
bottle表示空瓶子
首先分析每一个成员变量,

  • 1drink=1cup+1bottle;
  • 3cup=1drink;
  • 2bottle=1drink;

**所以置换的过程就是cup-3,drink+1 或者bottle-2,drink+1的过程。
所以递归调用的条件就是每一轮置换后,cup>=3||bottle>=2, &&drinks<1也就是跳出递归的条件是:每一轮置换后cup<3&&bottle<2 &&drinks<1**

因为在这个过程中,三个元素drink, cup, bottle都有连续的变化,所以递归调用时要将三个参数都传就去。
定义方法体
`public static int Soda(int drink, int cup, int bottle){

}
`
除了第一轮之外,每一轮的drink=cup/3+bottle/2, cup=cup%3, bottle=bottle%2
所以我本来是这么写的(错的):

    public static int Soda(int drinks,int caps,int bottles){

        if(caps<3&&bottles<2&&drinks<1){
            return drinks;
        }else{
            caps+=drinks;bottles+=drinks;
            drinks=caps/3+bottles/2;
            caps%=3;  bottles%=2;
            return Soda(drinks,caps,bottles)+drinks;
        }
    }

得出来是93,我惊呆了,拿给老范,老范说,不不不还有更多,应该是113。

113,那就是说,我缺少20个,正好是第一轮拿钱买的20个。我分析了一下,发现这个代码在第一次递归过程中,return Soda(drinks,caps,bottles)+drinks 这后面跟的drinks已经被第二次置换的drinks替换掉了,导致少了第一次花20块大洋买的drinks。

所以得调整顺序
将drinks=caps/3+bottles/2的过程放在return内部
return Soda(caps/3+bottles/2,caps,bottles)+drinks;
然后崩了,因为caps%=3;bottles%=2已经发生了,所以caps/3+bottles/2已经小于1了,直接满足了终止条件跳出了循环

将这两句调换顺序并放在if前,这样在每次递归时,首先将caps和bottles置换掉,之后再加上就不会影响drinks的置换数量。

  1. caps%=3; bottles%=2;
  2. caps+=drinks;bottles+=drinks;

然后我还发现,终止条件caps<3&&bottles<2&&drinks<1有点啰嗦,因为caps+=drinks;bottles+=drinks;只要drinks<1,就表示caps和drinks肯定满足终止条件。

最终代码:

public static int Soda(int drinks,int caps,int bottles){
        caps%=3;  bottles%=2;
        caps+=drinks;  bottles+=drinks;
        if(caps<3&&bottles<2){
            return drinks;
        }
        else{
            return Soda(caps/3+bottles/2,caps,bottles)+drinks;
        }
    }

得出113瓶。。。。
久久不能释怀,感觉错过了一个亿 v_v

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/127468.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • pycharm打开全部变成乱码_pycharm控制台输出中文乱码

    pycharm打开全部变成乱码_pycharm控制台输出中文乱码如上图所示:PyCharm默认打开时,pythonconsole中如果用到windows下cmd里的命令时,会输出乱码,原因:cmd默认的编码是gbk(代码页:936),而pythonconsole里面的编码是utf-8(代码页:65001),由于编码不一致,所以输出时会出现乱码解决办法:如下:1、File-Setting-Search-console-pythonconsol

    2022年8月27日
    6
  • 如何制作whl文件「建议收藏」

    如何制作whl文件「建议收藏」wheel文件Wheel和Egg都是python的打包格式,目的是支持不需要编译或制作的安装过程,实际上也是一种压缩文件,将.whl的后缀改为.zip即可可看到压缩包里面的内容。按照官网说法,wheels是发行版Python的新标准并且要取代.egg。Egg格式是由setuptools在2004年引入,而Wheel格式是由PEP427在2012年定义。Wheel现在被认为是Python的二…

    2022年5月7日
    115
  • POJ3623:Best Cow Line, Gold(后缀数组)

    POJ3623:Best Cow Line, Gold(后缀数组)

    2022年1月13日
    39
  • axisfaultexception_this alias is not available

    axisfaultexception_this alias is not available出现下面情况,可以是proxy.setEndpoint(endpoint);中endpoint没有给对导致AxisFault faultCode:{http://xml.apache.org/axis/}Server.NoService faultSubcode: faultString:TheAXISenginecouldnotfindatargetservicetoinvoke! targetServiceisGetPassword1 faultActor

    2025年11月4日
    5
  • 数仓分层简介(实时数仓架构)

    数仓1.数仓分层好处:复杂问题简单化;减少重复开发;隔离原始数据。2.数仓分层具体实现ODS(OperationDataStore)层:原始数据层,存原始数据,直接加载原始日志、数据DWD(DataWarehouseDetail)层:明细数据层也有叫DWI层,结构和粒度与原始表保持一致,对ODS层数据进行清洗(去除空值、脏数据、超过极限范围的数据、行式存储转列式存储、改压缩格式)DWS(DataWarehouseService)层:服务数据层,以DWD为基础进行轻度汇总。比如:用户当日

    2022年4月17日
    84
  • linux中pushd和popd用法,在Linux中使用pushd和popd命令操作目录的用法

    linux中pushd和popd用法,在Linux中使用pushd和popd命令操作目录的用法pushd将目录压入目录栈,进行目录切换命令用法:pushd[-n][+N|-N|dir]参数:+n切换目录,以当前目录为准,从右向左数第n个-n切换目录,以当前目录为准,从左向右数第n个例:复制代码代码如下:[root@localhostzhangy]#pushd/root~/home/zhangy#这个符号~代表根home目录复制代码代码如下:[root…

    2022年6月20日
    31

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号