C语言实现大数运算(长整数的加、减、乘、除)

由于整型数的位数有限,因此整型数不能满足大整数(超长整数)的运算要求 。大整数计算是利用字符串来表示大整数,即用字符串的一位字符表示大整数的一位数值,然后根据四则运算规则实现大整数的四则运算。

简单表述大数运算实现过程:
1、加法:两数末尾对齐,自末端逐个相加,如果有进位则高位加一;
2、减法:比较好两数大小,大数减去小数,两数末尾对齐,自末端逐个相减,减不掉就向高位借一
3、乘法:取其中任意一个数,分别用此数各位中数字与另一个数字相乘,各个位置相乘的结果重叠部分相加,最终考虑进位关系。
4、除法:如果被除数大于被除数,先保持除数和被除数位数相同,然后被除数一次次减去除数,最终得到在此位之下被除数对于除数的倍数,改变位数,得到其他位数下的倍数,最终得到结果。

#include<stdio.h>
#include<stdlib.h>
#include<string.h>

int a[1000],b[1000],c[1000];
 
//返回两者较大值 
int max(int a,int b){
	if(a>b)return a;
	else return b;
}

//将数组设置为0 
int setZero(int*num,int len){
	for(int i=0;i<len;i++)
		num[i]=0;
}

//比较两个数组形式数字大小 
int compare(int*num1,int*num2){
	int i,j;
	for(i=999;num1[i]==0;i--);
	for(j=999;num2[j]==0;j--);
	if(i<j)return -1;
	else if(i>j)return 1;
	else{
		for(;i>=0;i--){
			if(num1[i]>num2[i])
				return 1;
			else if(num1[i]<num2[i])
				return -1;
		}
		return 0;
	}
}

//将一个数组赋给另一个数组 
int copy(int*num1,int*num2){
	int i,j;
	for(i=0;i<1000;i++)
		num1[i]=0;
	for(i=999;num2[i]==0;i--);
	for(j=0;j<=i;j++)
		num1[j]=num2[j];
	return i+1;
}

//大数加法 
void add(char*str1,char*str2){
	setZero(a,1000);
	setZero(b,1000);
	setZero(c,1000);
	printf("%s + %s = ",str1,str2);
		
	int i,carry=0;
	int len1=strlen(str1),len2=strlen(str2),len=max(len1,len2);
	for(i=0;i<len1;i++)
		a[i]=str1[len1-1-i]-'0';
	for(i=0;i<len2;i++)
		b[i]=str2[len2-1-i]-'0';
	for(i=0;i<len;i++){
		c[i]=(a[i]+b[i]+carry)%10;
		carry=(a[i]+b[i]+carry)/10;
	}
	

	if(carry!=0)
		c[len++]=1;
	for(i=len-1;i>=0;i--)
		printf("%d",c[i]);
	printf("\n");
}

//大数减法 
void sub(char*str1,char*str2){
	setZero(a,1000);
	setZero(b,1000);
	setZero(c,1000);
	printf("%s - %s = ",str1,str2);
	
	int i,borrow=0;
	int len1=strlen(str1),len2=strlen(str2),len=max(len1,len2);
	for(i=0;i<len1;i++)
		a[i]=str1[len1-1-i]-'0';
	for(i=0;i<len2;i++)
		b[i]=str2[len2-1-i]-'0';	
	
	//初步根据位数判断两数大小	
	if(len2>len1){ 
		printf("-");
		for(i=0;i<len;i++){
			c[i]=b[i]-a[i]-borrow;
			if(c[i]<0){
				borrow=1;
				c[i]+=10;
			}
			else
				borrow=0;
		}
	}
	else{
		for(i=0;i<len;i++){
			c[i]=a[i]-b[i]-borrow;
			if(c[i]<0&&i+1<len){
				borrow=1;
				c[i]+=10;
			}
			else
				borrow=0;
		}
	}
	//如果发现被减数小于减数,则重新计算,用减数减去被减数,输出负号 
	if(c[len-1]<0){
		printf("-");
		for(i=0;i<len;i++){
			c[i]=b[i]-a[i]-borrow;
			if(c[i]<0&&i+1<len){
				borrow=1;
				c[i]+=10;
			}
			else
				borrow=0;
		}
	}
	
	
	while(c[len-1]==0&&len>1)
		len--;
	for(i=len-1;i>=0;i--)
		printf("%d",c[i]);
	printf("\n");
}

//大数乘法 
void mul(char*str1,char*str2) {
	setZero(a,1000);
	setZero(b,1000);
	setZero(c,1000);
	printf("%s * %s = ",str1,str2);
	
	int i,j,k,carry=0,tmp; 
	int len1=strlen(str1),len2=strlen(str2);
	for(i=0;i<len1;i++)
		a[i]=str1[len1-1-i]-'0';
	for(i=0;i<len2;i++)
		b[i]=str2[len2-1-i]-'0';
	
	for(i=0;i<len1;i++){
		for(k=i,j=0;j<len2;j++,k++){
			c[k]+=a[i]*b[j];
		}
	}
	for(i=0;i<=k;i++) {
		c[i]+=carry;
		tmp=c[i];
		c[i]=tmp%10;
		carry=tmp/10;
	}
	
	for(i=k;c[i]==0;i--);
	for(;i>=0;i--)
		printf("%d",c[i]);
	printf("\n");
}


//大数除法 
void div(char*str1,char*str2){
	setZero(a,1000);
	setZero(b,1000);
	setZero(c,1000);
	printf("%s / %s = ",str1,str2);
	
	int i,j,tmp,tmp2=0,borrow=0,temp[1000]={0},temp2[1000]={0};
	int len1=strlen(str1),len2=strlen(str2);
	for(i=0;i<len1;i++)
		a[i]=str1[len1-1-i]-'0';
	for(i=0;i<len2;i++)
		b[i]=str2[len2-1-i]-'0';
	
	if(compare(a,b)<0){
		printf("0\n");
		return;
	}
	
	while(compare(a,b)>=0){			
		tmp=len1-len2;
		if(tmp==tmp2&&tmp>0)
			tmp--;
		tmp2=tmp;
		
		setZero(temp2,1000);
		for(i=len1-1;i>=tmp;i--)
			temp2[i]=b[i-tmp];
		copy(temp,a);
		
		if(compare(temp,temp2)<0)
				continue;
		for(j=1;;j++){
			borrow=0;
			for(i=tmp;i<len1;i++){
				temp[i]=a[i]-temp2[i]-borrow;
				if(temp[i]<0){
					borrow=1;
					temp[i]+=10;
				}
				else
					borrow=0;
			}
			
			len1=copy(a,temp);			
			c[tmp]=j;
			if(compare(temp,temp2)<0)
				break;
		}
	}
	
	for(i=999;c[i]==0;i--);
	for(;i>=0;i--)
		printf("%d",c[i]);
	printf("\n");
}

int main() {
	add("265151916516519419516","15616651622161651616");
	sub("163165165161984116165","15454874487989819449");
	mul("798794149194949451949","98491516194941631316");
	div("16516498416516516191981916516165","12345678");
}

代码效果:

在这里插入图片描述

  • 52
    点赞
  • 318
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
### 回答1: 在C语言中进行大数运算,一般有以下几种方式: 1. 自己实现高精度计算库:通过定义自己的数据结构(比如用数组表示大整数)和实现基本的大数加减乘除运算,可以实现高精度计算。但是这需要自己实现大量的代码,而且效率可能不如专业的大数计算库。 2. 使用现有的大数计算库:有一些现成的开源大数计算库(比如GNU MP库),可以在C语言中直接调用。使用这些库可以避免自己实现大量的代码,同时也可以获得高效的运算性能。 无论采用哪种方式,都需要了解大数运算的原理和具体实现方式,以及注意处理边界情况和错误情况,以保证计算的正确性和可靠性。 ### 回答2: 大数运算是指处理超过计算机整数位数上限的数值运算C语言是一种基础的编程语言,它没有内置针对大数运算的数据类型和运算符,但可以通过自定义数据结构和算法实现大数运算。 一种常用的实现大数运算的方法是使用数组来表示大数。具体步骤如下: 1. 定义一个足够的数组,用于存储大数的每一位,数组的度应根据需要处理的数值范围确定。 2. 将大数按照逆序的方式存储在数组中,即个位数存储在数组的第一个元素,十位数存储在数组的第二个元素,依次类推。 3. 实现基本的法、法、乘法和除法运算函数。对于法和法,可以从个位数开始逐位相或相,将结果存储在新的数组中,并考虑进位或借位;对于乘法和除法,可以参考手动计算的方法,将运算结果逐步存储在新的数组中。 4. 根据需要,还可以实现其他的运算操作,如取模运算、幂运算等。 需要注意的是,由于大数计算涉及到多位数的运算,所以处理起来相对较慢,需要考虑计算时间和内存占用。同时,大数运算中也需要考虑到运算结果溢出的问题,以及负数运算的处理等。 综上所述,通过使用数组和自定义算法,我们可以利用C语言进行大数运算。这样可以扩展C语言在数值计算方面的能力,应对一些特殊的计算需求。 ### 回答3: 在C语言中进行大数运算可以通过使用字符串来表示和计算大数。以下是一种简单的实现方法: 1. 首先,将要进行运算大数转换成字符串表示。可以使用字符数组来表示,例如char number[100],其中100是一个足够大的度以容纳大数。 2. 定义一个结构体来表示大数,结构体中包含一个字符数组和一个记录大数位数的变量。例如: ```c struct BigInt { char digit[100]; int length; }; ``` 3. 编写函数来实现大数的基本运算,例如法、法、乘法和除法。对于法和运算,可以模仿手动计算时的竖向计算方法,从低位数开始逐位相或相,同时注意进位或借位。 4. 对于乘法和除法运算,可以借助于循环和进位(借位)的操作,逐位相乘或相除,并将结果保存在一个新的字符数组中。 5. 在进行运算时,为了方便操作,可以将字符串的字符顺序进行逆序,从而在计算过程中更容易按照从低位到高位的顺序进行。 6. 编写其他必要的辅助函数,例如比较大小、取反等。 请注意,大数运算是一种复杂的问题,需要考虑到边界情况和错误处理。以上仅是一种简单的实现方法,对于更复杂的大数运算问题还需要进一步的优化和改进。
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值