高精度除法分析及代码

  算法分析

   高精度除法, 这个和加减乘一样,我们都要从手算的角度入手。举一个例子,比如 524134 除以 123。结果是4261 

第一位4的来源是 我们把 524和123对齐,然后进行循环减法,循环了4次,余32,将32134的前三位321继续和123对齐,循环减法2次,余75,把7534的前三位753和123对齐,循环减法6次,余15,将154和123对齐,只能减1次,所以结果是4 2 6 1。

      把上述过程程序化

1.把A,B两个数存入char数组 0下标表示的是最高位
2.把A的前lenB位和B对齐进行大小比较
3.如果2的比较结果里A的前lenB位大,那么就进行循环减法,直到它比B小,循环的次数记作s[0]表示最终结果的最高位
4.如果2的比较结果里A的前lenB位小,什么也不做.
5.把B整体向后以一位,和A的最高位对齐(最高位可能暂时是0) 把s的下标迭代+1 表示进行下一位的计算
6.不断地比较,直到当B的尾部和A的尾部对齐时,说明A的最后lenB位也进行了循环减法算数,所以得到了结果.终止程序
//以上提到的lenB都是最开始的B的长度,后来由于移位会导致增大

过程如下

A:524134/

B:123/

循环4次减法

A:032134/

B:0123/             -----循环错位 前方补0

循环2次减法

A:007534/

B:00123/

循环6次减法

A:000154/

B:000123/

循环1次减法

 

   代码如下:(代码是除数后补0法)

 #include <iostream>

#include <cstring>

#include <cstdio>

#define MAX 10001

using namespace std;

 

char A[MAX]={'0'},B[MAX]={'0'};

int lenA=0,lenB=0;

int res[MAX]={0};

bool isGreater(){

    return strncmp(A,B,lenB)>=0;//注意 lenB会因为移位而增大,所以不必担心

}

//把A当前的最高位和B的最高位对齐后相减

void sub(){

    //首先要找到A的最高位 和B进行对齐,由于在while循环里已经进行了大小的判断,此时最高位指的就是真正的最高位

    int i, begin = 0;

    for (i = 0; A[i]=='0'; ++i)

        begin = i;

    //此时begin和i都是最高位的下标 已经是和B对齐的了

    for (; i < lenB; ++i)

        A[i] = A[i] - B[i] +'0'; //注意 如果用 -= 的话 应该是 A[i] -= B[i] - '0'

    //开始借位 把负数补全 从右向左借

    for (int i = lenB-1; i >0; --i) if(A[i]<'0')

    {

        A[i]+=10;

        A[i-1]--;

    }

}

 

int main(int argc, char const *argv[])

{

    cin>>A>>B;

    lenA = strlen(A);    

    lenB = strlen(B);

    //若A小于B 直接输出0结束.

    if(lenA<lenB or (lenA==lenB and !isGreater())){

        cout<<0<<endl;return 0;

    }

    int res_i = 0; //标记res数组正在处理的下标 0是最高位

    while(1){ //当最后对齐时break

        //开始对A和B的前lenB位进行循环减法

        

        while(isGreater()){

            sub();

            ++res[res_i];

        }

        ++res_i;

        //若上一次的余数小于B的位数,则cot为0没有问题

        if(lenA==lenB)    //必须在减法循环结束之后立刻检验 因为移位之后lenB发生了变化 而且也没有必要

            break;

        //对B进行向右移动 注意lenB要增加1

        int j;

        for ( j = lenB++; j > 0 ; --j)

            B[j]=B[j-1];

        B[j]='0';

    }

    bool flag = true;//表示以后有0不输出

    for (int i = 0; i < res_i; ++i)

    {

        if(flag and res[i]!=0)

            flag = false;//已经遇到了 第一个不是0的数 表示以后有0输出

        if(res[i]==0){

            if(!flag)

                cout<<0;

        }else

            cout<<res[i];

 

    }

    cout<<endl;

    return 0;

}

  • 9
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: GMP (GNU Multiple Precision Arithmetic Library) 是一个开源的多精度数学库,支持高精度的整数和浮点数运算。Golang 有一个内置的多精度数学包(math/big),它使用了 GMP 库来实现高精度运算。 Golang 的 math/big 包提供了一些基本的数学运算,包括加法、减法、乘法、除法和模运算。它还提供了一些高级的函数,如幂运算、对数、三角函数和取整函数。 Golang 的 math/big 包非常适用于处理高精度的数学运算,如货币计算、加密、科学计算等。它的精度可以达到任意精度,而且运算速度也相当快。 总的来说,Golang 的 GMP 模型是一个非常优秀的多精度数学库,适用于各种高精度数学运算的场景。 ### 回答2: Golang中的gmp模型指的是Goroutine、Mutex、和Channel三个概念的组合使用。 首先,Goroutine是轻量级的线程,可以通过go关键字启动。Goroutine利用multiplexer调度器实现并发执行。它不仅启动速度快,而且占用的内存资源较少,可以高效地处理大量任务。 接下来,Mutex是一种互斥量,用于保护共享资源的访问。在多个Goroutine并发执行的情况下,Mutex可以保证在同一时间只有一个Goroutine能够访问共享资源,避免数据竞争的问题。 最后,Channel是一种用于在Goroutine之间通信的管道。它可以实现Goroutine之间的同步和数据传递。Channel提供了发送和接收操作,可以在不同的Goroutine之间发送和接收数据,确保数据的安全和一致性。通过Channel,Goroutine可以并发地进行通信和协作,实现更高效的并发编程。 Golang的gmp模型的优点是简洁而高效。Goroutine提供了轻量级的并发模型,使得并发编程变得简单而高效。Mutex在保护共享资源的同时,避免了死锁的问题。Channel则提供了通信和同步的机制,保证了数据的安全和一致性。 此外,通过使用gmp模型,我们可以充分利用多核处理器的计算能力,提高程序的并发性能。Goroutine的调度器能够智能地利用多核处理器的资源,实现任务的并发执行,提高程序的运行效率。 总结来说,Golang的gmp模型提供了一种简洁而高效的并发编程模型,使得我们能够轻松地实现并发编程,并充分利用多核处理器的计算能力。通过合理地使用Goroutine、Mutex和Channel,我们能够编写出高效且可靠的并发程序。 ### 回答3: gmp是golang中的一种高性能并发模型,它是一种轻量级的基于goroutine的消息传递模型。与传统的共享内存模型相比,gmp模型更加安全和简单。 gmp模型的核心思想是通过goroutine之间的通信来实现任务的执行和数据的传递。在gmp模型中,每个任务都可以作为一个goroutine来执行,并与其他goroutine进行消息的传递。这种消息传递是通过通道(channel)来实现的,通道是goroutine之间的连接,用于传递数据和控制消息的通信方式。通过通道,可以实现任务之间的同步和协作,有效地避免了传统共享内存模型中的数据竞争和死锁等问题。 在gmp模型中,每个goroutine都可以独立运行,无需关心其他goroutine的存在和状态。这样可以极大地简化程序的编写和维护,提高了代码的可读性和可维护性。此外,gmp模型还提供了丰富的并发原语,如锁、等待组、原子操作等,以支持更复杂的并发控制和同步操作。 相比于其他并发模型,gmp模型的优势在于其简单性和易用性。并且由于golang本身对并发模型的良好支持,因此在实际使用中,gmp模型可以帮助开发者更轻松地编写高效且具有良好并发性能的程序。 然而,gmp模型也存在一些局限性。由于通道是goroutine之间的连接,因此通道的创建和关闭操作等极易受到限制。另外,在某些情况下,使用共享内存模型可能更加高效,因为通道的消息传递机制可能会引入一定的开销。因此,在选择并发模型时,开发者需要根据具体的应用场景和性能需求进行权衡。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值