biginteger大数运算:从O(n³)到O(n²)的跨越

本文涉及的产品
实时计算 Flink 版,5000CU*H 3个月
智能开放搜索 OpenSearch行业算法版,1GB 20LCU 1个月
实时数仓Hologres,5000CU*H 100GB 3个月
简介: 这篇文章讲述了作者优化大数乘法算法的过程,从起初的O(n³)时间复杂度降低到O(n²)。优化的关键在于使用`int`类型临时存储,避免了进位检查,提升了运算速度。作者提供了代码示例,并通过测试验证了算法的性能,特别是在处理长串数字时效果显著。最后,文章强调了算法设计和数据结构选择的重要性,并提到将继续探索更高效的算法,如Karatsuba算法。

文章是我在2011年,写在某sdn的。搬运过来。


在编程面试中,大数运算题常常作为一道难题出现,考验着开发者对算法效率的理解与掌握。本文将分享一次从O(n³)到O(n²)时间复杂度优化的实战经验,通过具体案例,带你领略算法优化的魅力。

背景介绍

大数运算,尤其是大数乘法,因其涉及大量位数,传统计算机中的基本数据类型(如int, long等)往往无法满足需求。因此,实现一个能够处理任意长度数字的乘法算法显得尤为重要。

初始尝试:笔算方法的直接映射

最初,我采用了一种直观的方法,将手算乘法的过程直接转换为代码实现。虽然保证了计算的正确性,但其时间复杂度高达O(n³),显然无法应对大规模数据的运算需求。

算法优化:向O(n²)迈进

经过一番研究与思考,我发现了更高效的算法策略,将时间复杂度成功降至O(n²)。关键在于利用int类型的临时存储,避免了每次累加后的进位检查,而是统一在所有累加完成后进行。这一改进显著提升了运算速度,尤其是在处理长串数字时效果尤为明显。

实现细节

在具体实现上,我采用了动态分配的整型数组r来存储中间结果,随后通过循环处理进位,确保最终结果的准确性。此外,还特别注意了非零索引的查找,以避免不必要的零前缀。

测试与验证

在一系列测试中,该优化后的算法展现出色的性能。以两个2000位数的乘法为例,平均耗时仅为90多毫秒,而200位数的乘法则仅需1毫秒左右,充分验证了O(n²)复杂度的预期表现。

代码示例

以下是优化后的大数乘法函数实现,包括了必要的辅助函数与主程序逻辑:

Cpp

#include <iostream>
#include <assert.h>
#include <stdio.h>
#include <time.h>
#include <windows.h>
#include <malloc.h>
 
 
using namespace std;
 
const int MAX = 2001;
 
char num1[MAX];
char num2[MAX];
char result[2*MAX];
 
 
void  SafeGetNumFromStr ( char* num, char* str);
char* BigIntMultiply ( const char * const a, const char * const b , char* const lresult);
void  multiply( const char *a, const char *b);
 
int main(int argc, char *argv[])
{
  //test speed
  cout<<"\n\nspeed test... Number of digits is : "<<MAX-1<<"\n";
  int i;
  const int TEST_TIME = 20;
  srand((unsigned)time(NULL));
  for(i = 0;i<MAX;i++)
  {
    num1[i] = 0;
    num2[i] = 0;
  }
  //create data with random
  for(i = 0; i< MAX - 1; i++)
  {
    num1[i] = rand()%10 + '0';
    num2[i] = rand()%10 + '0';
  }
 
  SYSTEMTIME wtm;
  GetLocalTime(&wtm);
  long long time_start = wtm.wMilliseconds + wtm.wSecond * 1000;
  cout<<num1<<endl;
  cout<<"*\n";
  cout<<num2<<endl;
  for(i = 0; i<TEST_TIME; i++)
  {
    BigIntMultiply(num1,num2,result);
  }
  GetLocalTime(&wtm);
  cout<<"Result is:\n";
  cout<<result<<endl;
  double tmv = (double)(wtm.wMilliseconds + wtm.wSecond * 1000 - time_start);
  cout<<"Test Over. "<<TEST_TIME<<" loops use time: "<<tmv<<" ms\n";
  cout<<"     Each One Time Use: "<<tmv/(double)TEST_TIME<<" ms\n\n\n";
  
 
 
  //test validation
  cout<<"Validation work...\n";
  long long  testNum1;
  long long  testNum2;
  int testI;
  for(testNum1 = 0;testNum1<1000000000000000;testNum1 = (testNum1+1)*181+1)
    for(testI= 0;testI<200; testI++)
    {
      char a[2*MAX];
      char b[2*MAX];
      testNum2 = (testNum1+testI)<0?0:testNum1+testI;
      for(i = 0;i<MAX;i++)
      {
        num1[i] = 0;
        num2[i] = 0;
      }
      itoa(testNum1,a,10);
      itoa(testNum2,b,10);
      SafeGetNumFromStr(num1,a);
      SafeGetNumFromStr(num2,b);
      BigIntMultiply(num1,num2,result);
 
      if(8 == testNum2%10)
        if(testNum1*testNum2 == atoi(result))
          cout<<testNum1<<" * "<<testNum2<<"  ==  "<<testNum1*testNum2<<"    Correct!\n";
        else
          cout<<testNum1<<" * "<<testNum2<<"  Result:"<<result<<"\n";
    }
  
  
  
 
  //free test
  cout<<"Now ..... Free Test!\n";
 
  while(1)
  {
    char a[2*MAX];
    char b[2*MAX];
    cout<<"\n\ninput long integer for A"<<endl;
    cin>>a;
    cout<<"input long integer for B"<<endl;
    cin>>b;
 
    //get data
    SafeGetNumFromStr(num1,a);
    SafeGetNumFromStr(num2,b);
    cout<<endl<<endl;
    cout<<num1;
    cout<<"  *  ";
    cout<<num2;
    cout<<endl;
    BigIntMultiply(num1,num2,result);
    cout<<"Result is:"<<endl;
    cout<<result;
  }
 
  system("pause");
  return 0;
}
 
 
 
void SafeGetNumFromStr( char* num, char* str)
{
  memset(num,0,sizeof(num[0])*MAX);
  int i;
  int index = 0;
  for(i=0;i<2*MAX && index < MAX;i++)
  {
    if(str[i] <= '9' && str[i] >='0')
      num[index++] = str[i];
    if('\0'==str[i])
      break;
  }
  assert( 0 != index );
}
 
 
 
char* BigIntMultiply ( const char * const a, const char * const b , char* const lresult)
{
  int i,j;
  int la = strlen(a);
  int lb = strlen(b);
  int rlen = la+lb;
  int* r = (int*)calloc( rlen, sizeof(int) );
 
  for(i = 0;i < lb; i++)
    for(j = 0; j < la; j++)
      r[rlen - 1 - i - j] += (b[lb - i - 1] - '0') * (a[la - j - 1] - '0');
 
  //then is there carry on current number
  for(j = rlen - 1; j >= 1; j--)
    if(r[j] > 9)
    {
      r[j-1] += r[j] / 10;
      r[j] %= 10;
    }
  //find first none_zero_index
  for(i = 0; 0 == r[i]; i++){}
    
  //mem cpy
  for(j=0; i< rlen; i++,j++)
    lresult[j] = r[i]+'0';
  lresult[j]='\0';
 
  free(r);
  return lresult;
}


image.png


image.png


image.png



总结与展望

通过本次算法优化之旅,我们不仅实现了从O(n³)到O(n²)的飞跃,更深刻理解了算法设计与数据结构选择的重要性。在后续的探索中,我们还将继续挑战更高效的大数运算算法,如Karatsuba算法,进一步提升计算效率。

注意事项

值得注意的是,尽管优化后的算法在多数情况下表现出色,但在极端条件下,如数字位数接近int的最大表示范围时,仍存在溢出风险。因此,在实际应用中,还需考虑更高级的数据类型或库支持,以确保算法的鲁棒性与通用性。

通过本文的分享,希望能激发你对算法优化的兴趣

相关文章
BigInteger大数字方法的解释及使用
BigInteger大数字方法的解释及使用
124 0
|
5月前
|
缓存 安全 Java
【揭秘】String vs StringBuilder vs StringBuffer:三大字符串类的秘密较量!你真的知道何时该用哪个吗?
【8月更文挑战第19天】探讨Java中`String`、`StringBuilder`与`StringBuffer`的区别及应用场景。`String`不可变,适合做哈希表键或多线程共享。`StringBuilder`支持动态修改字符串,适用于单线程环境以提高性能。`StringBuffer`与`StringBuilder`功能相似,但线程安全。示例代码展示各类型的基本用法。选择哪种类型取决于具体需求和性能考量。
72 0
|
7月前
|
安全 C语言 C++
数与字符串相互转化
数与字符串相互转化
33 1
|
Java
Java 中大数的处理方案BigInteger和BigDecimal类的使用
Java 中大数的处理方案BigInteger和BigDecimal类的使用
103 0
|
8月前
|
存储 Java C++
java中基本数据类型转化以及字符串之间的转化
java中基本数据类型转化以及字符串之间的转化
|
8月前
|
存储 编解码
字符串转ASSII字符串相互处理
字符串转ASSII字符串相互处理
64 0
运用BigInteger进行整数之间的高精度的加减乘除运算
运用BigInteger进行整数之间的高精度的加减乘除运算
113 0
|
Java 索引
java中大数的计算BigInteger和BigDecimal两个类的常用方法
java中大数的计算BigInteger和BigDecimal两个类的常用方法
112 0
|
Rust 算法 Python
【密码学】杂谈-字节数组和int之间的转换
本文还是来随便聊一聊,我们在去看一些密码学算法的结构的过程当中,我们经常的会发现,这些结构内部的数据的处理方式并不都是根据字节来处理的,有可能他们对于数据的处理用的u32或者说是u64,之前我们说了,在计算机的内部,最小的单位是字节,那么我们怎么将这个字节处理成为结构当中需要的u32或者u64呢?本文接下来就来聊一下他们之间的转换过程(还是老样子,只考虑无符号数)
【密码学】杂谈-字节数组和int之间的转换