我有一个BigInteger值,假设它是282并且在变量x中。我现在想写一个while循环,声明:
while b2 isn't a perfect square:
a ← a + 1
b2 ← a*a - N
endwhile
我如何使用BigInteger做这样的事情?
编辑:我的目的是写this method。正如文章所述,必须检查b2是否不是正方形。
答案 0 :(得分:11)
计算整数平方根,然后检查其正方形是否为数字。这是我使用Heron's method计算平方根的方法:
private static final BigInteger TWO = BigInteger.valueOf(2);
/**
* Computes the integer square root of a number.
*
* @param n The number.
*
* @return The integer square root, i.e. the largest number whose square
* doesn't exceed n.
*/
public static BigInteger sqrt(BigInteger n)
{
if (n.signum() >= 0)
{
final int bitLength = n.bitLength();
BigInteger root = BigInteger.ONE.shiftLeft(bitLength / 2);
while (!isSqrt(n, root))
{
root = root.add(n.divide(root)).divide(TWO);
}
return root;
}
else
{
throw new ArithmeticException("square root of negative number");
}
}
private static boolean isSqrt(BigInteger n, BigInteger root)
{
final BigInteger lowerBound = root.pow(2);
final BigInteger upperBound = root.add(BigInteger.ONE).pow(2);
return lowerBound.compareTo(n) <= 0
&& n.compareTo(upperBound) < 0;
}
答案 1 :(得分:2)
我找到了一个使用here的sqrt方法,并简化了平方测试。
private static final BigInteger b100 = new BigInteger("100");
private static final boolean[] isSquareResidue;
static{
isSquareResidue = new boolean[100];
for(int i =0;i<100;i++){
isSquareResidue[(i*i)%100]=true;
}
}
public static boolean isSquare(final BigInteger r) {
final int y = (int) r.mod(b100).longValue();
boolean check = false;
if (isSquareResidue[y]) {
final BigInteger temp = sqrt(r);
if (r.compareTo(temp.pow(2)) == 0) {
check = true;
}
}
return check;
}
public static BigInteger sqrt(final BigInteger val) {
final BigInteger two = BigInteger.valueOf(2);
BigInteger a = BigInteger.ONE.shiftLeft(val.bitLength() / 2);
BigInteger b;
do {
b = val.divide(a);
a = (a.add(b)).divide(two);
} while (a.subtract(b).abs().compareTo(two) >= 0);
return a;
}
答案 2 :(得分:1)
我用这个:
SQRPerfect是我要测试的数字: 这也有一个很好的Square Root,所以如果你需要它也可以使用它。如果你将Square Root带入Perfect Square测试,你可以加快一点,但我喜欢这两个功能。
if(PerfectSQR(SQRPerfect)){
Do Something
}
public static Boolean PerfectSQR(BigInteger A) {
Boolean p=false;
BigInteger B=SQRT(A);
BigInteger C=B.multiply(B);
if (C.equals(A)){p=true;}
return p;
}
public static BigInteger SQRT(BigInteger A) {
BigInteger a=BigInteger.ONE,b=A.shiftRight(5).add(BigInteger.valueOf(8));
while ((b.compareTo(a))>=0){
BigInteger mid = a.add(b).shiftRight(1);
if (mid.multiply(mid).compareTo(A)>0){b=mid.subtract(BigInteger.ONE);}
else{a=mid.add(BigInteger.ONE);}
}
return a.subtract(BigInteger.ONE);
}
答案 3 :(得分:0)
using System.Numerics; // needed for BigInteger
/* Variables */
BigInteger a, b, b2, n, p, q;
int flag;
/* Assign Data */
n = 10147;
a = iSqrt(n);
/* Algorithm */
do
{ a = a + 1;
b2 = (a * a) – n;
b = iSqrt(b2);
flag = BigInteger.Compare(b * b, b2);
} while(flag != 0);
/* Output Data */
p = a + b;
q = a – b;
/* Method */
private static BigInteger iSqrt(BigInteger num)
{ // Finds the integer square root of a positive number
if (0 == num) { return 0; } // Avoid zero divide
BigInteger n = (num / 2) + 1; // Initial estimate, never low
BigInteger n1 = (n + (num / n)) / 2;
while (n1 < n)
{ n = n1;
n1 = (n + (num / n)) / 2;
}
return n;
} // end iSqrt()
答案 4 :(得分:0)
private static boolean isSqrt(BigInteger n, BigInteger root)
{
final BigInteger lowerBound = root.pow(2);
final BigInteger upperBound = root.add(BigInteger.ONE).pow(2);
return lowerBound.compareTo(n) <= 0
&& n.compareTo(upperBound) < 0;
}
我使用JavaScript BigInt尝试了上述方法:
function isPerfectSqrt(n, root) {
const lowerBound = root**2n;
const upperBound = (root+1n)**2n
return lowerBound <= n && n < upperBound;
}
并发现它(在Node V8中)仅比单层快60%:
function isPerfectSqrt(n, root) {
return (n/root === root && n%root === 0n)
}
答案 5 :(得分:0)
您要对其进行完全平方测试的数字是 A。B 是 A 的整数平方根,.sqrt() 函数返回平方根的下层整数。返回 B*B=A 的布尔值。如果它是一个完美的正方形,布尔返回为“真”,如果它不是一个完美的正方形,则返回“假”。
public static Boolean PerfectSQR(BigInteger A) {
BigInteger B = A.sqrt();
return B.multiply(B).equals(A);
}
答案 6 :(得分:-1)
请勿使用此...
BigInteger n = ...;
double n_as_double = n.doubleValue();
double n_sqrt = Math.sqrt(n_as_double);
BigInteger n_sqrt_as_int = new BigDecimal(n_sqrt).toBigInteger();
if (n_sqrt_as_int.pow(2).equals(n)) {
// number is perfect square
}
正如Christian Semrau在下面评论的那样 - 这不起作用。我很抱歉发布了错误的答案。