完全平方数的判定及整数平方根的快速求解

2024-05-04 23:18

本文主要是介绍完全平方数的判定及整数平方根的快速求解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上个星期的“有道难题”网络预赛中,某些Group的第二题涉及到了完全平方数的判定问题。相信大多数人的判定语句都差不多是这样写的:if(sqr((int)sqrtl(n))==n){...},把这条语句整理一下,大概是这样:

inline __int64 sqr( int n){ return  static_cast<__int64>(n)*n;} 

inline  int directSqrt( int n) 

     int root =  static_cast< int>(sqrtl(n)); //求n的开方,但是double类型的
     if (sqr(root+ 1) <= n) 
     { 
         return root +  1
     } 
     else  return root; 


inline  bool isSquareNumber( int n) 

   int root = directSqrt(n); 
   return sqr(root) == n; 
}

这个判定方法主要是利用标准库的sqrt函数来求整数平方根(integer square root),正确性不用怀疑,效率也足够。但是如果想要更快呢?通常来说,要判定任意一个数是否是完全平方数,求根的过程在所难免。但是,标准库的sqrt函数是求浮点平方根的,而我们只需要获取整数平方根即可,所以在这里肯定还有性能提升的空间。于是接下来就列举一些计算整数平方根的方法(后两个算法不是我写的),按照计算速度从慢到快依次排列,其中最后一个算法的效率大概是上述直接利用sqrtl的方法的四倍。

1. 线性搜索,直接从小到大依次取数试探,最原始的办法,代码如下:

inline  int linearSqrt( int n) 

     int i =  0
     while (sqr(i) <= n) ++i; 
     return i- 1
}


2. 二分法寻找根

inline  int bisectionSqrt( int n) 

     int l =  0, r = n/ 2 +  1
     while (r - l >  1
     { 
         int m = (l+r)/ 2
         __int64 sqrm = sqr(m); 
         if (sqrm == n) 
         { 
             return m; 
         } 
         else  if (sqrm > n) r = m; 
         else l = m; 
     } 
     
     if (sqr(r) == n) 
     { 
         return r; 
     } 
     else  return l; 
}

3. 牛顿迭代法,令方程 x^2 - n = 0,则求sqrt(n)也即是求这个方程的根,根据牛顿法的迭代公式有:x' = x - (x^2-n)/(2x) = (x^2+n)/(2x),选定合适的初始值迭代,很快就能收敛到根。另外,这个迭代其实也就是不动点迭代的松弛形式,由 x^2= n => x = n/x, 得不动点迭代x' = n/x,加上松弛系数1/2(算数平均),得x'= 1/2 * (x + n/x),即 x' = (x^2+n)/(2x),与前面牛顿法的迭代公式相同。算法的代码如下:

inline  int newtonSqrt( int n) 

     static  int table[ 16] = { 
         11<< 21<< 41<< 61<< 81<< 101<< 121<< 141<< 16
         1<< 181<< 201<< 221<< 241<< 261<< 281<< 30
     }; 
     unsigned i =  0, x0 =  0, x1 =  0
     while (table[i] < n && i!= 16) ++i; x0 =  1 << i; //寻找迭代的初识值
     while (x0 !=  0
     {     
         x1 = static_cast< int>((sqr(x0) + n)/ ( 2*x0)); 
         if (x1 >= x0) 
         { 
             break
         } 
         x0 = x1; 
     } 
     return x0; 
}

4. 逐位确定法,从高到低依次确定根的各个二进制位。见下面这个代码,看起来似乎很飘逸,但实际上算法很简单明晰,具体的解释可以搜索James Ulery的论文:"Computing Integer Square Roots".

inline  int fastIntSqrt( int n) 

     int temp, nHat =  0, b =  0x8000, bshft =  15
     do 
     { 
         if (n >= (temp = (((nHat<< 1)+b)<<bshft--))) 
         { 
             nHat += b; 
             n -= temp; 
         } 
     }  while (b >>=  1); 
     return nHat; 
}

5.建表+牛顿迭代,这个算法号称是要用来替代java数学包中的(int)(java.lang.Math.sqrt(integer))函数的,速度奇快,首先建立了一张强大的表来确定初值,然后最多经过两次牛顿迭代后就得到精确解。

inline  int javaIntSqrt( int n) 

     static  int table[ 256] = { 
         0,     16,   22,   27,   32,   35,   39,   42,   45,   48,   50,   53,   55,   57
         59,    61,   64,   65,   67,   69,   71,   73,   75,   76,   78,   80,   81,   83
         84,    86,   87,   89,   90,   91,   93,   94,   96,   97,   98,   99101102
         103104106107108109110112113114115116117118
         119120121122123124125126128128129130131132
         133134135136137138139140141142143144144145
         146147148149150150151152153154155155156157
         158159160160161162163163164165166167167168
         169170170171172173173174175176176177178178
         179180181181182183183184185185186187187188
         189189190191192192193193194195195196197197
         198199199200201201202203203204204205206206
         207208208209209210211211212212213214214215
         215216217217218218219219220221221222222223
         224224225225226226227227228229229230230231
         231232232233234234235235236236237237238238
         239240240241241242242243243244244245245246
         246247247248248249249250250251251252252253
         253254254255 
     }; 

     int xn; 
     if (n >=  0x7FFEA810return  0xB504
     if (n >=  0x10000) { 
         if (n >=  0x1000000) { 
             if (n >=  0x10000000) { 
                 if (n >=  0x40000000) { 
                     xn = table[n>>  24] <<  8
                 }  else { 
                     xn = table[n>>  22] <<  7
                 } 
             }  else { 
                 if (n>=  0x4000000) { 
                     xn = table[n>>  20] <<  6
                 }  else { 
                     xn = table[n>>  18] <<  5
                 } 
             } 

             xn = (xn +  1 + (n/ xn)) >>  1
             xn = (xn +  1 + (n/ xn)) >>  1
             return ((xn * xn) > n) ? --xn : xn; 
         }  else { 
             if (n>=  0x100000) { 
                 if (n>=  0x400000) { 
                     xn = table[n>>  16] <<  4
                 }  else { 
                     xn = table[n>>  14] <<  3
                 } 
             }  else { 
                 if (n>=  0x40000) { 
                     xn = table[n>>  12] <<  2
                 }  else { 
                     xn = table[n>>  10] <<  1
                 } 
             } 

             xn = (xn +  1 + (n/ xn)) >>  1
             return ((xn * xn) > n) ? --xn : xn; 
         } 
     }  else { 
         if (n>=  0x100) { 
             if (n>=  0x1000) { 
                 if (n>=  0x4000) { 
                     xn = (table[n>>  8]) +  1
                 }  else { 
                     xn = (table[n>>  6] >>  1) +  1
                 } 
             }  else { 
                 if (n>=  0x400) { 
                     xn = (table[n>>  4] >>  2) +  1
                 }  else { 
                     xn = (table[n>>  2] >>  3) +  1
                 } 
             } 
             return ((xn * xn) > n) ? --xn : xn; 
         }  else { 
             if (n>=  0) { 
                 return table[n] >>  4
             } 
         } 
     } 
     return - 1
}

依次计算从1到1000000的每个数的整数平方根,下面分别是上面6个算法所用的时间:

0. 利用标准库sqrtl, directSqrt : 610 ms
1. 线性搜索, linearSqrt : 81256 ms
2. 二分法, bisectionSqrt : 2734 ms
3. 牛顿迭代法, newtonSqrt : 691 ms
4. 逐位确定法, fastIntSqrt : 301 ms
5. 建表+牛顿迭代, javaIntSqrt : 150 ms

可见牛顿迭代法与利用标准库sqrtl的方法速度相当,逐位确定法的速度大概是牛顿迭代法的两倍。当然,最快的还是建表+牛顿迭代,看来以其来替代(int)(java.lang.Math.sqrt(integer))也不是没有道理的。另,以上测试数据相对来说都比较小,可以想象,若是计算接近于2^31这样的大数的整数平方根,逐位确定法和建表+牛顿迭代应该更占据优势。

这篇关于完全平方数的判定及整数平方根的快速求解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/960313

相关文章

Python实现精确小数计算的完全指南

《Python实现精确小数计算的完全指南》在金融计算、科学实验和工程领域,浮点数精度问题一直是开发者面临的重大挑战,本文将深入解析Python精确小数计算技术体系,感兴趣的小伙伴可以了解一下... 目录引言:小数精度问题的核心挑战一、浮点数精度问题分析1.1 浮点数精度陷阱1.2 浮点数误差来源二、基础解决

从入门到精通详解Python虚拟环境完全指南

《从入门到精通详解Python虚拟环境完全指南》Python虚拟环境是一个独立的Python运行环境,它允许你为不同的项目创建隔离的Python环境,下面小编就来和大家详细介绍一下吧... 目录什么是python虚拟环境一、使用venv创建和管理虚拟环境1.1 创建虚拟环境1.2 激活虚拟环境1.3 验证虚

从基础到高级详解Python数值格式化输出的完全指南

《从基础到高级详解Python数值格式化输出的完全指南》在数据分析、金融计算和科学报告领域,数值格式化是提升可读性和专业性的关键技术,本文将深入解析Python中数值格式化输出的相关方法,感兴趣的小伙... 目录引言:数值格式化的核心价值一、基础格式化方法1.1 三种核心格式化方式对比1.2 基础格式化示例

Python ORM神器之SQLAlchemy基本使用完全指南

《PythonORM神器之SQLAlchemy基本使用完全指南》SQLAlchemy是Python主流ORM框架,通过对象化方式简化数据库操作,支持多数据库,提供引擎、会话、模型等核心组件,实现事务... 目录一、什么是SQLAlchemy?二、安装SQLAlchemy三、核心概念1. Engine(引擎)

Python多线程实现大文件快速下载的代码实现

《Python多线程实现大文件快速下载的代码实现》在互联网时代,文件下载是日常操作之一,尤其是大文件,然而,网络条件不稳定或带宽有限时,下载速度会变得很慢,本文将介绍如何使用Python实现多线程下载... 目录引言一、多线程下载原理二、python实现多线程下载代码说明:三、实战案例四、注意事项五、总结引

C#使用Spire.XLS快速生成多表格Excel文件

《C#使用Spire.XLS快速生成多表格Excel文件》在日常开发中,我们经常需要将业务数据导出为结构清晰的Excel文件,本文将手把手教你使用Spire.XLS这个强大的.NET组件,只需几行C#... 目录一、Spire.XLS核心优势清单1.1 性能碾压:从3秒到0.5秒的质变1.2 批量操作的优雅

MySQL 数据库表操作完全指南:创建、读取、更新与删除实战

《MySQL数据库表操作完全指南:创建、读取、更新与删除实战》本文系统讲解MySQL表的增删查改(CURD)操作,涵盖创建、更新、查询、删除及插入查询结果,也是贯穿各类项目开发全流程的基础数据交互原... 目录mysql系列前言一、Create(创建)并插入数据1.1 单行数据 + 全列插入1.2 多行数据

Mybatis-Plus 3.5.12 分页拦截器消失的问题及快速解决方法

《Mybatis-Plus3.5.12分页拦截器消失的问题及快速解决方法》作为Java开发者,我们都爱用Mybatis-Plus简化CRUD操作,尤其是它的分页功能,几行代码就能搞定复杂的分页查询... 目录一、问题场景:分页拦截器突然 “失踪”二、问题根源:依赖拆分惹的祸三、解决办法:添加扩展依赖四、分页

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

使用Redis快速实现共享Session登录的详细步骤

《使用Redis快速实现共享Session登录的详细步骤》在Web开发中,Session通常用于存储用户的会话信息,允许用户在多个页面之间保持登录状态,Redis是一个开源的高性能键值数据库,广泛用于... 目录前言实现原理:步骤:使用Redis实现共享Session登录1. 引入Redis依赖2. 配置R