使用PHP中的bcmath计算N次方根

6
我们正在寻找PHP中的N次根。我们需要对一个非常大的数字进行此操作,但是Windows计算器返回结果为2。使用以下代码我们得到了1。请问有人知道这是如何工作的吗?
echo bcpow(18446744073709551616, 1/64);

2
不起作用,因为bcpow()仅接受整数指数。如果您只需要64次根,则可以进行六次平方根。 - Niko
2个回答

12

看起来PHP和BC库有一些限制,我在网上搜索后发现了这篇有趣的文章/代码:

所以你应该使用这个函数:

<?php

function NRoot($num, $n) { 
    if ($n<1) return 0; // we want positive exponents 
    if ($num<=0) return 0; // we want positive numbers 
    if ($num<2) return 1; // n-th root of 1 or 2 give 1 

    // g is our guess number 
    $g=2; 

    // while (g^n < num) g=g*2 
    while (bccomp(bcpow($g,$n),$num)==-1) { 
        $g=bcmul($g,"2"); 
    } 
    // if (g^n==num) num is a power of 2, we're lucky, end of job 
    if (bccomp(bcpow($g,$n),$num)==0) { 
        return $g; 
    } 

    // if we're here num wasn't a power of 2 :( 
    $og=$g; // og means original guess and here is our upper bound 
    $g=bcdiv($g,"2"); // g is set to be our lower bound 
    $step=bcdiv(bcsub($og,$g),"2"); // step is the half of upper bound - lower bound 
    $g=bcadd($g,$step); // we start at lower bound + step , basically in the middle of our interval 

    // while step!=1 

    while (bccomp($step,"1")==1) { 
        $guess=bcpow($g,$n); 
        $step=bcdiv($step,"2"); 
        $comp=bccomp($guess,$num); // compare our guess with real number 
        if ($comp==-1) { // if guess is lower we add the new step 
            $g=bcadd($g,$step); 
        } else if ($comp==1) { // if guess is higher we sub the new step 
            $g=bcsub($g,$step); 
        } else { // if guess is exactly the num we're done, we return the value 
            return $g; 
        } 
    } 

    // whatever happened, g is the closest guess we can make so return it 
    return $g; 
}

echo NRoot("18446744073709551616","64");

?>

希望这对您有所帮助...


2
谢谢!运行得很好!我想给你+10,但我只能给+1 :P - Deep Frozen

2

我在使用HamZa的解决方案时遇到了精度任意的问题,所以我对其进行了一些修改。

<?php

function NthRoot($Base, $NthRoot, $Precision = 100) {
    if ($NthRoot <  1) return 0;
    if ($Base    <= 0) return 0;
    if ($Base    <  2) return 1;
    $retVal    = 0;
    $guess     = bcdiv($Base, 2, $Precision);
    $continue  = true;
    $step = bcdiv(bcsub($Base, $guess, $Precision), 2, $Precision);
    while ($continue) {
        $test = bccomp($Base, bcpow($guess, $NthRoot, $Precision), $Precision);
        if ($test == 0) {
            $continue = false;
            $retVal   = $guess;
        }
        else if ($test > 0) {
            $step  = bcdiv($step, 2, $Precision);
            $guess = bcadd($guess, $step, $Precision);
        }
        else if ($test < 0) {
            $guess = bcsub($guess, $step, $Precision);
        }
        if (bccomp($step, 0, $Precision) == 0) {
            $continue = false;
            $retVal   = $guess;
        }
    }
    return $retVal;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接