java实现开根号的运算

面试的时候,偶然被问到,开根号的实现,虽然给面试官讲解了思路,但是没有实际实现过,今天闲来无事,就把自己的思路写一下,做个笔记。

如果某个数字正好可以开根号为2个整数,例如1,4,9等,那就很简单了。

如果某个数字不可以正好开根号为2个整数,而且要保留几位精度,例如:2,3,5等,我们该怎么办呢?????

首先我们可以把这个数字分成整数部分和小数部分,分别计算。

例如√5≈2.236  我们可以先算出整数部分为2,然后在根据保留几位精度,去计算小数部分。依次计算十分位、百分位和千分位等,然后把整数位+十分位+百分位+千分位+。。。,结果就是我们想要的结果了。

下面我写了一个通用的方法,可以根据传的参数来保留精度。


package comc.n;

import java.math.BigDecimal;

public class Square {

	public static void main(String[] args) {
		System.out.println(Math.sqrt(5));
		System.out.println(MathSqure(5, 6));
	}
	/**
	 * 
	 * @param n  需要开根号的数据
	 * @param m  需要保留的精度,即几位小数
	 * @return
	 */
	public static double MathSqure(int n, int m){
		double[] arr = new double[m];
		if(m >0){
			arr = sc(m);
		}
		int s = sq(n);
		
		return sb(n, s, arr);
	}
	
	/**
	 * 计算整数位
	 * @param n
	 * @return
	 */
	public static int sq(int n){
		if( n == 1){
			return 1;
		}
		int tmp = 0;
		for(int i=1;i<=n/2+1;i++){
			if(i*i == n){
				tmp = i;
				break;
			}
			if(i*i > n){
				tmp = i-1;
				break;
			}
		}
		return tmp;
	}
	
	/**
	 * 计算要保留几位小数
	 * @param m
	 * @return
	 */
	public static double[] sc(int m){
		double[] arr = new double[m];
		int num = 0;
		while(num != m){
			double f = 1;
			for(int i=0;i<=num;i++){
				f = f*10;
			}
			arr[num] = 1/f;
			num++;
		}
		return arr;
	}
	
	/**
	 * 开根号
	 * @param n
	 * @param j
	 * @param arr
	 * @return
	 */
	public static double sb(int n, double j, double[] arr){
		double tmp = j;
		for(int p=0;p<arr.length;p++){
			if(p>0){
				j = tmp;//计算过后的值(整数位+小数位的和,赋值给j,下面继续运算)
			}
			for(int i=1;i<=9;i++){//小数位只有九位{0.1,0.2,0.3,0.4,0.5,0.6,0.7,0.8,0.9}
				tmp = i*arr[p]+j;//i*arr[p],相当于每次加0.1,0.2 ...
				if(tmp*tmp == n){
					return tmp;
				}
				if(tmp*tmp >n){
					//避免丢失精度
					BigDecimal c1 = new BigDecimal(Double.toString(tmp));
					BigDecimal c2 = new BigDecimal(Double.toString(arr[p]));
					tmp = c1.subtract(c2).doubleValue();
					break;
				}
			}
		}
		return tmp;
	}
}

输出结果:


  • 6
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java ,BigInteger 类型没有内置的开根方法。但是,我们可以采用牛顿迭代法来实现。具体的实现步骤如下: 1. 首先将 BigInteger 转换成 BigDecimal 类型,然后取该数的迭代初值 x0,这里可以取 1。 2. 对于一个正整数 n,它的平方根 s 可以通过以下迭代得到: s = 0.5 * (s + n/s) 在这里,我们可以用 BigDecimal 来表示数字 s 和 n/s。 3. 不断迭代直到收敛,即 abs(s-x0) < ε,其 ε 为一个比较小的正数。 4. 最后将得到的 BigDecimal 转换成 BigInteger 类型并返回即可。 以下是代码示例: ``` import java.math.BigDecimal; public class Sqrt { public static BigInteger sqrt(BigInteger n) { BigDecimal x0 = new BigDecimal("1"); BigDecimal eps = new BigDecimal("0.0001"); BigDecimal s = new BigDecimal(n); s = s.divide(x0, BigDecimal.ROUND_HALF_UP); BigDecimal x1 = new BigDecimal("0"); while (abs(x1.subtract(x0)).compareTo(eps) > 0) { x1 = x0; s = s.add(n.divide(s, BigDecimal.ROUND_HALF_UP)) .multiply(new BigDecimal("0.5")); x0 = s; } return x0.toBigInteger(); } private static BigDecimal abs(BigDecimal x) { return x.compareTo(BigDecimal.ZERO) > 0 ? x : x.negate(); } public static void main(String[] args) { BigInteger n = new BigInteger("1000000"); BigInteger ans = sqrt(n); System.out.println(ans); // 输出:31622 } } ``` 需要注意的是,在计算平方根过程,使用 BigDecimal 进行运算时,需要指定保留小数位数。具体的保留方法可以根据实际情况选择适当的常量值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值