Sqrt(x) -- LeetCode
来源:互联网 发布:开源oa系统 java 编辑:程序博客网 时间:2024/06/06 17:13
原题链接: http://oj.leetcode.com/problems/sqrtx/
这是一道数值处理的题目,和Divide Two Integers不同,这道题一般采用数值中经常用的另一种方法:二分法。基本思路是跟二分查找类似,要求是知道结果的范围,取定左界和右界,然后每次砍掉不满足条件的一半,知道左界和右界相遇。算法的时间复杂度是O(logx),空间复杂度是O(1)。代码如下:
public int sqrt(int x) { if(x<0) return -1; if(x==0) return 0; int l=1; int r=x/2+1; while(l<=r) { int m = (l+r)/2; if(m<=x/m && x/(m+1)<m+1) return m; if(x/m<m) { r = m-1; } else { l = m+1; } } return 0;}
二分法在数值计算中非常常见,还是得熟练掌握。其实这个题目还有另一种方法,称为牛顿法。不过他更多的是一种数学方法,算法在这里没有太多体现,不过牛顿法是数值计算中非常重要的方法,所以我还是介绍一下。不太了解牛顿法基本思想的朋友,可以参考一下牛顿法-维基百科。一般牛顿法是用数值方法来解一个f(y)=0的方程(为什么变量在这里用y是因为我们要求的开方是x,避免歧义)。对于这个问题我们构造f(y)=y^2-x,其中x是我们要求平方根的数,那么当f(y)=0时,即y^2-x=0,所以y=sqrt(x),即是我们要求的平方根。f(y)的导数f'(y)=2*y,根据牛顿法的迭代公式我们可以得到y_(n+1)=y_n-f(n)/f'(n)=(y_n+x/y_n)/2。最后根据以上公式来迭代解以上方程。
public int sqrt(int x) { if (x == 0) return 0; double lastY = 0; double y = 1; while (y != lastY) { lastY = y; y = (y + x / y) / 2; } return (int)(y);}实际面试遇到的题目可能不是对一个整数开方,而是对一个实数。方法和整数其实是一致的,只是结束条件换成左界和右界的差的绝对值小于某一个epsilon(极小值)即可。这里注意一个小问题,就是在java中我们可以用==来判断两个double是否相等,而在C++中我们则需要通过两个数的绝对值差小于某个极小值来判断两个double的相等性。实际上两个double因为精度问题往往是不可能每一位完全相等的,java中只是帮我们做了这种判定。比较典型的数值处理的题目还有Divide Two Integers,Pow(x,n)等,其实方法差不多,一般就是用二分法或者以2为基进行位处理的方法。
58 0
- LeetCode: Sqrt(x)
- LeetCode sqrt(x)
- [Leetcode] Sqrt(x)
- LeetCode :Sqrt(x)
- leetcode 53: Sqrt(x)
- [LeetCode] Sqrt(x)
- Leetcode 69 sqrt(x)
- [Leetcode] Sqrt(x)
- leetcode:Sqrt(x)
- 【leetcode】sqrt(int x)
- [LeetCode]Sqrt(x)
- [leetcode]Sqrt(x)
- LeetCode-Sqrt(x)
- [leetcode] Sqrt(x)
- [leetcode] Sqrt(x)
- LeetCode - Sqrt(x)
- leetcode之sqrt(x)
- LeetCode | Sqrt(x)
- 《UNIX网络编程》源码的使用
- linux mysql 外网访问
- 再议gluPerspective和gluLookAt的关系
- 动态规划-斐波那契数列
- (总结)Nginx配置文件nginx.conf中文详解
- Sqrt(x) -- LeetCode
- 【2014-02-28】收集最新jquery特效与素材资源
- java从文件中读取内容方法
- windows下配置git默认编辑器
- 如何做域名解析(A记录、CNAME记录、URL转发、二级域名)
- sort
- (七)类
- PAT 1058. A+B in Hogwarts (20)
- android手机分享