• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

日本雅虎程序员跳槽程序测试问题

php 搞代码 3年前 (2022-01-25) 18次浏览 已收录 0个评论
文章目录[隐藏]

一救援机器人有三种跳跃模式,可分别跳跃1m,2m,3m的距离,请用程序实现该机器人行进n米路程时可用的跳跃方式。
程序语言不限,当距离n值足够大时,程序执行时间尽量小。

例:当距离为4米时,输出结果有7种:

<code>1m,1m,1m,1m1m,1m,2m1m,2m,1m1m,3m2m,1m,1m2m,2m3m,1m</code>

回复内容:

一救援机器人有三种跳跃模式,可分别跳跃1m,2m,3m的距离,请用程序实现该机器人行进n米路程时可用的跳跃方式。
程序语言不限,当距离n值足够大时,程序执行时间尽量小。

例:当距离为4米时,输出结果有7种:

<code>1m,1m,1m,1m1m,1m,2m1m,2m,1m1m,3m2m,1m,1m2m,2m3m,1m</code>

谢谢邀请。

出这道题,明显是为了考察你 DP。我也不多说,直接放码过来了:

<code>cpp</code><code>void step(int n, std::string str) {    if (n == 0) { std::cout << str << "\b " <= 1) step(n-1, str+"1m,");    if (n >= 2) step(n-2, str+"2m,");    if (n >= 3) step(n-3, str+"3m,");}</code>

n == 4 的时候,调用:step(4, ""); 原样输出你想要的。


这里只是用最短的代码表述我的思路,怕爆栈的,自行修改。

It is quite similar to my Facebook interview question: a game has 3 levels of scores: 1 point, 2 points, 5 points. please print all the ways you can get 10 points.

简单想了一下,这道题目优化思路和斐波那契数列的优化思路相同:记录f(n)。

根据题目可知f(n)=f(n-1)+f(1)=f(n-2)+f(2)=f(n-3)+f(3)=f(1)+f(n-1)=..
手机码字后面两个等式就不写了。
f(n)的可能也就这6种(当然肯定包含重复)

那么一点点推导f(4)因为f123已知,f4可以在O(1)内得到,记录。
f5,此时f1234已知,f5也能在O(1)得到,记录。
那么f(n),根据上述的公式,可以在O(n)内得到。

这是大致思路,接下来解决重复问题就行了。

根据 @AUV_503516 的思路, 写以下代码, 存储结构和空间还可以优化

<code>#!/usr/bin/env python3# -*- coding: utf-8 -*-## @file     robot_steps_calculator.py # @author   kaka_ace# @date     Mar 27 2015# @breif     #import copyclass RobotBrain(object):    CACHE_INIT = {        1: [['1']],        2: [['1', '1'], ['2']],        3: [['1', '1' ,'1'], ['1', '2'], ['2', '1'], ['3']],    }    CACHE_KEY_MAP_INIT = dict() # 缓存如 '11', '21' etc. 去重作用    for k, v in CACHE_INIT.items():        for l in v:            s = "".join(l)            CACHE_KEY_MAP_INIT[s] = True    def __init__(self):        self.cache = copy.deepcopy(self.CACHE_INIT)        self.cache_key_map = copy.deepcopy(self.CACHE_KEY_MAP_INIT)    def output_solution(self, n: "total length, positive number") -> None:        if n <= 3:            print(RobotBrain._ouput_result(self.cache[n]))            return        i = 4        while i  None:        for sl in self.cache[i-delta]:            s = ''.join(sl)            delta_str = str(delta)            if reverse is True:                # delta + f(i - delta)                new_key = delta_str + s            else:                # f(i - delta) + delta                new_key = s + delta_str            if new_key not in self.cache_key_map:                self.cache_key_map[new_key] = True                self.cache[i].append([delta_str] + sl)    @staticmethod    def _ouput_result(cache_list) -> None:        for cache_result in cache_list:            print(",".join([i+"m" for i in cache_result]))if __name__ == "__main__":    r = RobotBrain()    r.output_solution(5)</code>

用C写了一段:

<code>unsigned int count(unsigned int n){    switch(n) {        case 0: return 0;        case 1: return 1;        case 2: return 2;        case 3: return 4;        default: return (count(n - 1) + count(n - 2) + count(n - 3));    }    return 0;}</code>

运行结果:

<code>n = 0; t = 0n = 1; t = 1n = 2; t = 2n = 3; t = 4n = 4; t = 7n = 5; t = 13n = 6; t = 24n = 7; t = 44n = 8; t = 81n = 9; t = 149...</code>

好吧,原来是要详细列出每种方法的方案,我没认真看…… =_,=b

直接采用

本文来*源gaodai^.ma#com搞#代!码网
搞gaodaima代码

递归的话会有很多重复的计算,比如在n=7的时候,会有1+1+1+steps(4),1+2+steps(4),3+steps(4),所以step(4)会被重复计算多次。因此在需要做记忆之前的结果

<code>int steps[LENGTH] = {0, 1, 2, 4};int n, i;for ( i = 4; i <= n; i++) {        steps[i] = steps[i-1]+steps[i-2]+steps[i-3];}printf("%d\n", steps[n]);</code>

写个For循环算了,

这也就OI初中组的水平……典型的动态规划一点儿弯儿都没绕

这也就是考递归的问题.

如果是要求一共有多少种方法,就可以简单用一位数组做动态规划。如果要求输出所有解,就老老实实的求吧。代码用递归写的。

<code>JAVA</code><code>    /**     * return number of all distinct paths for a given distance..     * O(n) time and O(n) space using dp.     */    public int allPath(int n) {        int[] dp = new int[n+1];        dp[0] = 1; // useless        dp[1] = 1;        dp[2] = 2;        dp[3] = 4;        for (int i = 4; i <= n; i++) {            dp[i] = dp[i-1] + dp[i-2] + dp[i-3];        }        return dp[n];    }    /**     *     * return all distinct paths for a given distance.     *     */    public List<List> allPaths(int n) {        List<List> res = new ArrayList<List>();        helper(n, 0, new ArrayList(), res);        return res;    }    private void helper(int n, int cur, List list, List<List> res) {        if (cur > n) {            return;        }        if (cur == n) {            res.add(list);            return;        }        for (int i = 1; i < 4 && cur + i <= n; i++) {            List newList = new ArrayList(list);            newList.add(i);            helper(n, cur + i, newList, res);        }    }</code>

分别n/3,n/2,看余数是多少。如果被3整除就是3333…; n/3有余数y2,y2/2没有余数,就是33333…2; n/3有余数y2,y2/2也有余数,最终的余数就用1来跳,3333…1。上面是n很大的情况。如果n小于等于3,一步就够。

f(n)=1,n=1
f(n)=2,n=2
f(n)=4,n=3
f(n)=f(n-1)+f(n-2)+f(n-3),n>3

<code>int fuck(int length){    if(length <0)    {        return 0;    }    if(length == 1 || length == 0)        return 1;    return fuck(length-1)+fuck(length-2)+fuck(length-3);}</code>

指数的时间复杂度,因为产生了重复计算,按照斐波那契数列的方法,改成迭代或者在递归函数里多传点参数,时间复杂度能变成线性的。

艹,看错题目了,将错就错

<code>void p(const list &state){    for(auto it=state.begin();it!=state.end();it++)        if(it != --state.end())            cout<<*it<<"m,";        else            cout<<*it<<'m';    cout<<endl;}list append(list m,int k){    m.push_back(k);    return m;}int cfuck(int length,list state){    if(length < 0){        return 0;    }    if(length == 0) {        p(state);        return 1;    }    return cfuck(length-1,append(state,1))+cfuck(length-2,append(state,2))+cfuck(length-3,append(state,3));}</code>

哎,C++知识全忘光了(反正我也进不了雅虎)。不过这道题目无论怎么样时间复杂度都是指数级的,所以就无所谓在最短的时间内了

大神你们都超屌的


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:日本雅虎程序员跳槽程序测试问题

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址