首页 > 编程语言 >高精度乘法C++

高精度乘法C++

时间:2024-10-21 10:10:07浏览次数:6  
标签:高精度 int double len C++ l1 ax n3 乘法

1.高精度乘高精度的简单算法

思想:倒置相乘,统一处理进位,还原。

复杂度:$o(n^2)$

// By SnowDream
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=1e5+10;
string s1,s2;
int n1[N],n2[N],n3[N];//n1储存被乘数,n2储存乘数,n3储存积

void mul()
{
    int l1=(int)s1.size();
    int l2=(int)s2.size();
    //读取字符串转换为int类型并倒置储存至数组
    for(int i=0;i<l1;++i)
    {
        n1[l1-1-i]=s1[i]-'0';
    }
    for(int i=0;i<l2;++i)
    {
        n2[l2-1-i]=s2[i]-'0';
    }

    for(int i=0;i<l1;++i)
    {
        for(int j=0;j<l2;++j)
        {
            n3[i+j]+=n1[i]*n2[j];
        }
    }
    //处理进位
    int l_max=l1+l2;
    for(int i=0;i<l_max;++i)
    {
        n3[i+1]+=n3[i]/10;
        n3[i]%=10;
    }
    
    if(n3[l_max])
        l_max++;
    while(n3[l_max-1]>=10)
    {
        n3[l_max]=n3[l_max-1]/10;
        n3[l_max-1]%=10;
        l_max++;
    }
    while(n3[l_max-1]==0&&l_max>1)
        l_max--;

    for(int i=l_max-1;i>=0;i--)
    {
        cout << n3[i];
    }
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cin >> s1 >> s2;
    mul();
    return 0;
}

2.高精度乘高精度FFT优化算法

思想:将两个大整数看作多项式的系数,然后利用FFT算法在$O(n log n)$的时间复杂度内计算出它们的乘积,并最终得到乘积的各位数字。

复杂度:$o(nlog(n))$

具体步骤:

  1. 将输入的两个大整数转换为对应的多项式表示,其中每个数字位作为多项式的系数。
  2. 对两个多项式进行补零操作,使其长度变为2的幂,方便进行FFT运算。
  3. 利用FFT算法对两个多项式进行快速傅里叶变换,得到它们在频域上的表示。
  4. 将两个多项式在频域上进行点乘操作,得到它们的乘积在频域上的表示。
  5. 对乘积进行逆FFT操作,得到乘积多项式在时域上的表示。
  6. 对乘积多项式进行进位处理,并输出最终的乘积结果。
// By SnowDream
#include<bits/stdc++.h>
using namespace std;

#define L(x) (1<< (x))
typedef long long ll;
const int N=1e5+10;

const double PI = acos(-1.0);
string s1,s2;
double ax[N],ay[N],bx[N],by[N];
int n1[N],n2[N],n3[N];

//将一个整数x的二进制表示进行位逆序排列,并返回结果。
// 函数接受两个参数:整数x和表示位数的整数bits
int reverseBits(int x, int bits)
{
    int ret = 0;//存储位逆序排列后的结果,初始化为0
    for(int i=0;i<bits;++i)//循环bits次,对x的二进制表示进行位逆序排列
    {
        ret <<= 1;//将ret左移一位,为下一位的值腾出位置
        ret |=x&1;//将x的最低位与ret进行或操作,将x的最低位值添加到ret的最低位
        x >>=1;//将x右移一位,准备处理下一位
    }
    return ret;
}

void fft(double * a, double * b, int n, bool rev)
{
    int bits = 0;// 计算n的二进制表示中位数
    while (1 << bits < n) ++bits;// 找到大于n的最小的2的幂
    for (int i = 0; i < n; i++)
    {
        int j = reverseBits(i, bits);// 将i按照bits位反转后的值赋给j
        if (i < j)// 交换a和b数组中i和j位置的值
        {
            swap(a[i], a[j]);
            swap(b[i], b[j]);
        }
    }
    for (int len = 2; len <= n; len <<= 1)// 迭代每个长度为len的子序列
    {
        int half = len >> 1;// 子序列长度的一半
        double wmx = cos(2 * PI / len), wmy = sin(2 * PI / len);// 计算旋转因子的实部和虚部
        if (rev) wmy = -wmy;// 如果是逆变换,则虚部取相反数
        for (int i = 0; i < n; i += len)// 遍历每个子序列的起始位置
        {
            double wx = 1, wy = 0;// 初始化旋转因子
            for (int j = 0; j < half; j++)// 遍历子序列的前半部分
            {
                double cx = a[i + j], cy = b[i + j];// 获取当前位置的实部和虚部
                double dx = a[i + j + half], dy = b[i + j + half];// 获取对应的另一半的实部和虚部
                double ex = dx * wx - dy * wy, ey = dx * wy + dy * wx;// 计算旋转后的值
                a[i + j] = cx + ex, b[i + j] = cy + ey;// 更新前半部分的值
                a[i + j + half] = cx - ex, b[i + j + half] = cy - ey;// 更新后半部分的值
                double wnx = wx * wmx - wy * wmy, wny = wx * wmy + wy * wmx;// 计算下一个旋转因子
                wx = wnx, wy = wny;// 更新旋转因子
            }
        }
    }
    if (rev)
    {
        for (int i = 0; i < n; i++)
            a[i] /= n, b[i] /= n;// 对结果进行归一化
    }
}

int solve(int l1,int l2,int ans[])
{
    int len = max(l1, l2), ln;
    for(ln=0; L(ln)<len; ++ln);// 找到大于len的最小的2的幂
    len=L(++ln);// 计算2的幂
    for (int i = 0; i < len ; ++i)
    {
        if (i >= l1) ax[i] = 0, ay[i] =0;// 如果i超过l1,则ax[i]和ay[i]赋值为0
        else ax[i] = n1[i], ay[i] = 0;// 否则将n1[i]赋值给ax[i],ay[i]赋值为0
    }
    fft(ax, ay, len, false);// 进行快速傅里叶变换
    for (int i = 0; i < len; ++i)
    {
        if (i >= l2) bx[i] = 0, by[i] = 0;// 如果i超过l2,则bx[i]和by[i]赋值为0
        else bx[i] = n2[i], by[i] = 0;// 否则将n2[i]赋值给bx[i],by[i]赋值为0
    }
    fft(bx, by, len, false);// 进行快速傅里叶变换
    for (int i = 0; i < len; ++i)
    {
        double cx = ax[i] * bx[i] - ay[i] * by[i];// 计算乘积的实部
        double cy = ax[i] * by[i] + ay[i] * bx[i];// 计算乘积的虚部
        ax[i] = cx, ay[i] = cy;// 更新ax和ay数组的值
    }
    fft(ax, ay, len, true);// 进行逆快速傅里叶变换
    for (int i = 0; i < len; ++i)
        ans[i] = (int)(ax[i] + 0.5);// 四舍五入取整
    return len;
}

void mul()
{
    int l;
    int i;
    string ans;
    memset(n3, 0, sizeof(n3));
    int l1 = (int)s1.size();
    int l2 = (int)s2.size();
    for(i = 0; i < l1; i++)
        n1[i] = s1[l1 - i - 1]-'0';
    for(i = 0; i < l2; i++)
        n2[i] = s2[l2-i-1]-'0';
    l = solve(l1,l2, n3);
    for(i = 0; i<l || n3[i] >= 10; i++) // 进位
    {
        n3[i + 1] += n3[i] / 10;
        n3[i] %= 10;
    }
    l = i;
    while(n3[l] <= 0 && l>0) l--; // 检索最高位
    for(i = l; i >= 0; i--)
        cout << n3[i]; // 倒序输出
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cin >> s1 >> s2;
    mul();
    cout << "\n";
    return 0;
}

3.高精度乘单精度

思想:倒置相乘,统一处理进位,再还原。

复杂度:$o(n)$

// By SnowDream
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=1e5+10;
string s1;
int s2;
int n1[N];
void mul()
{
    string ans;
    int l1 = (int)s1.size();
    for(int i=0;i<l1;++i)
    {
        n1[l1-1-i]=s1[i]-'0';
    }
    int w=0;
    for(int i=0;i<l1;++i)
    {
        n1[i]=n1[i]*s2+w;
        w=n1[i]/10;
        n1[i]=n1[i]%10;
    }
    while(w)
    {
        n1[l1++]=w%10;
        w/=10;
    }
    for(int i=l1-1;i>=0;i--)
        cout << n1[i];
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cin >> s1 >> s2;
    mul();
    cout << "\n";
    return 0;
}

标签:高精度,int,double,len,C++,l1,ax,n3,乘法
From: https://www.cnblogs.com/snowdreamxue/p/18488670

相关文章

  • C++基础与实用技巧第三课:内存管理与性能优化
    第二章:C++基础与实用技巧第三课:内存管理与性能优化1.动态内存的管理策略与技巧动态内存管理是C++编程的核心部分之一,合理管理内存可以极大提高程序的性能和稳定性。在C++中,动态内存的分配和释放通常使用new和delete运算符,但由于手动管理内存容易引入错误,因此建议使用现代C+......
  • 【c++篇】:解析c++类--优化编程的关键所在(一)
    文章目录前言一.面向过程和面向对象二.c++中的类1.类的引入2.类的定义3.类的封装和访问限定符4.类的作用域5.类的实例化6.类对象模型三.`this`指针1.`this`指针的引出2.`this`指针的特性3.C语言和c++实现栈Stack的对比前言在程序设计的广袤宇宙中,C++以其强大的功能......
  • C++回调
    目录1、回调(callback):函数指针回调:函数对象回调:Lambda表达式回调:2、对象绑定器(Binders):3、包装器(Wrappers):1、回调(callback):在C++中,回调(callback)是指一种将一个函数作为参数传递给另一个函数,并在该函数执行的过程中执行传递进来的函数的机制。回调通常用于实现一种灵活的、可扩展的......
  • 高精度加法
    #include<iostream>#include<vector>usingnamespacestd;vector<int>add(stringa,stringb){vector<int>a1,b1;for(intj=a.length()-1;j>=0;j--)a1.push_back(a[j]-'0');for(intj=......
  • cuda core实现两个128x128 float矩阵乘法demo
    #include<stdio.h>#include<cuda_runtime.h>//128x128->__global__voidmm(float*a,float*b,float*c){//8x8个方块,每个方块16x16extern__shared__floatbuf[];float*a_local=buf;float*b_local=buf+16*128;for(inti=......
  • C++ constexp vs const
    C++constexpvsconstconstexpr是在C++11标准中引入的关键字,目的是为编译时常量提供更强大的支持。它允许某些表达式在编译期进行求值,从而提高性能和优化能力。下面详细说明它与const的区别。constexpr和const的区别特性constexprconst引入版本C++11C++......
  • Chromium 中chrome.contextMenus扩展接口实现分析c++
    一、chrome.contextMenus使用 chrome.contextMenus API向GoogleChrome的上下文菜单中添加项。您可以选择从右键菜单中添加的对象类型,例如图片、超链接和页面。权限contextMenus您必须在扩展程序的清单中声明 "contextMenus" 权限,才能使用该API。此外,您应指定一个......
  • 【蓝桥杯】C++ 第20场 小白入门赛
    一、四个亲戚题目四个亲戚 题目分析字面意思:Daiyu+‘kind’代码#include<iostream>usingnamespacestd;intmain(){cout<<"Daiyu'kind'";return0;}二、黛玉泡茶题目黛玉泡茶 题目分析1.我们可以c2.然后c3.计算c,如果不能,整除后的答案还要加1 ......
  • 基于C++的 BP/CNN神经网络算法(不调包)
    目前玩深度学习的小伙伴,上来就是使用现有的深度学习框架(TensorFlow,keras,pytorch,caffe),增加网络层,就像搭积木似的,看似方便,实则有时不利于个人能力发展,要知道现在公司需要的算法工程师,不仅仅只是会搭积木(这种工作,入门几个月的人就可以干了),而是要深入底层,能优化代码,能自己搭。本文......
  • qt图像算法—图像的缩放之c++实现(不调包)
     1.基本原理  图像的缩放一般使用插值算法,而本章将介绍两种常用插值算法:最临近插值法和双线性插值法  1.最临近插值法  将浮点数的位置坐标,进行四舍五入找到原图像的整型坐标即可,具体操作可见下面的公式,其中原图像坐标为(x,y),输出图像坐标为(i,j),比例系数为fx和fy。......