-2

我最近才开始学习递归,并且在特定练习方面遇到了一些麻烦;从递归状态迭代地重写函数,特别是在涉及多个基本情况的情况下:

double function(int j, int i)
{
    if(i == 0 || j == 1)
    {
        return 1;
    }

    if(i == 1 || j == 0)
    {
        return j;
    }

    if(i > 0)
    {
        return j * function(j, --i);
    }

    return 1 / (function(j, -i))
}

我在迭代地重写函数时遇到了麻烦。

4

2 回答 2

1

首先,这是您的代码压缩(我这样做是为了回答,不要在实际代码中这样做。)

double function(int j, int i) {
    if(i == 0 || j == 1) return 1;
    if(i == 1 || j == 0) return j;
    if(i > 0) return j * function(j, --i);
    return 1 / (function(j, -i)); //changed this to -i
     //might be a division by zero, you should check for that
}

由于最后一个块只能有效地发生在最外层的循环中,我们将把它拉出来:

double outer_function(int j, int i) {
    if (i<0)
        return 1 / inner_function(j, -i);
    else
        return inner_function(j, i);
}
double inner_function(int j, int i) {
    if(i == 0 || j == 1) return 1;
    if(i == 1 || j == 0) return j;
    if(i > 0) return j * inner_function(j, --i);
}

我要做的第一件事是尝试将其放入尾递归形式。这涉及重新排列方程,因此递归之后没有任何内容。(我不是 100% 确定我做对了这一步)

double inner_function(int j, int i, int times=1) {
    if(i == 0 || j == 1) return times;
    if(i == 1 || j == 0) return times*j;
    return inner_function(j, --i, times*j);
}

现在,因为在每个代码路径中,函数调用之后都没有代码,所以这是完全尾递归的。尾递归很容易变成迭代!

double inner_function(int j, int i, int times=1) {
    while(true) {
        if(i == 0 || j == 1) return times;
        if(i == 1 || j == 0) return times*j;
        //return inner_function(j, --i, times*j);
        --i;
        times *= j;
        //go again!
    }
}

如果我要从这里优化:

double function(int j, int i) {
    bool invert = false;
    if(i<0) {
         i=-i; 
         invert=true;
    }
    double result=1;
    if(i == 0) result = 0;
    else if(j == 0) result = j;
    else if (j != 1) {
        while(i--)
            result *= j;
    }
    return (invert ? 1/result : result);            
}

或者,如果我猜你的意图:

double function(int j, int i) {
    return std::pow(double(j), double(i));
}
于 2012-10-15T18:47:39.367 回答
0

在递归期间你真的只有一个基本情况。j永远不会改变,所以它只是一开始的特例。i在第一次递归调用后将始终为正并趋向于 0。所以唯一真正的基本情况是i == 1.

函数的开头可以保持不变。

double function(int j, int i)
{
    if(i == 0 || j == 1) { return 1; }
    if(i == 1 || j == 0) { return j; }

现在你只需要处理i > 0andi < 0案例(i == 0一开始就已经处理好了)。

这两种情况的不同之处在于,如果i为负,则切换符号并反转结果。

    int invert = i < 0;
    i = abs(i); // or: if (i < 0) { i = -i; }

现在看看递归部分并弄清楚发生了什么。

return j * function(j, --i);

function()将被调用直到i为 1 并且当时的结果将是j. 每次迭代都会将返回值乘以j. 这可以这样写:

    double returnValue = j; // (the i == 1 case)

    while (i-- > 1) { // loop while i is greater than 1
        returnValue = j * returnValue; // multiply j by the return value
    }

如果需要,现在反转并返回结果。

    if (invert) {
        returnValue = 1 / returnValue;
    }
    return returnValue;
}
于 2012-10-15T19:49:38.177 回答