其他分享
首页 > 其他分享> > 高精度压位模版

高精度压位模版

作者:互联网

文章目录

高精度加法

/***************************************
User:Mandy.H.Y
Language:c++
Problem:luogu1601 A+B Problem(压位)
Algorhithm:High Precision Addition
****************************************/

#include<bits/stdc++.h>

using namespace std;

const int power=4; //压位压四位 
const int base=1e4;
const int maxn=505;

struct num{
	int a[maxn];
	
	num(){memset(a,0,sizeof(a));}
	
	int &operator [](int x){return a[x];}//重载中括号 
	
	num(char *s){
		
		memset(a,0,sizeof(a));
		int len=strlen(s);
		
		a[0]=(len+power-1)/power;//计算位数
		
		for(int i=0,t=0,w=0;i<len;++i,w*=10){
			if(i%power==0) {w=1,++t;}//最好写成这种形式 
			a[t]+=(s[i]-'0')*w;
			
		}
	}
	
	void print(){
		printf("%d",a[a[0]]);
		for(int i=a[0]-1;i>0;--i) printf("%04d",a[i]);//printf("%0*d",power,a[i]); 
	}
}ans,p,q;

num operator +(num &p,num &q){//重载了中括号所以不用const 
	int cur=0,len=max(p[0],q[0]);
	for(int i=1;i<=len;++i){
		cur+=i<=p[0]?p[i]:0;
		cur+=i<=q[0]?q[i]:0;
		p[i]=cur%base;cur/=base;
	}
	if(cur) p[++len]=cur;
	p[0]=len;//注意更新长度 
	return p;
}

char p1[maxn],q1[maxn];
int lenp,lenq;

int main(){
	scanf("%s%s",p1,q1);
	
	lenp=strlen(p1);lenq=strlen(q1);
	reverse(p1,p1+lenp);reverse(q1,q1+lenq);
	
	p=num(p1);q=num(q1);
	
	ans=p+q;
	ans.print();
	return 0;
}

高精度减法

/**************************************
User:Mandy.H.Y
Language:c++
Pronblem:luogu2142 A-B Problem
Algorhithm:High Precision Subtraction 
**************************************/

#include<bits/stdc++.h>

using namespace std;

const int power=4;
const int base=1e4;
const int maxn=1e4+5;

struct num{
	int a[maxn];
	
	num(){memset(a,0,sizeof(a));}
	
	num(char *s,int len){
		memset(a,0,sizeof(a));
		
		a[0]=(len+power-1)/power;
		
		for(int t=0,w=1,i=len-1;i>=0;w=(w<<1)+(w<<3),--i){//倒序省去reverse
			if((len-1-i)%power==0){++t,w=1;}
			a[t]+=(s[i]^48)*w;
		}
	}
	
	void print(){
		printf("%d",a[a[0]]);
		for(int i=a[0]-1;i>0;--i) printf("%0*d",power,a[i]);
	}
}p,q,ans;

bool operator <(const num &p,const num &q){//比较大小 
	if(p.a[0]<q.a[0]) return true;
	if(p.a[0]>q.a[0]) return false;
	for(int i=1;i<=p.a[0];++i){
		if(p.a[i]!=q.a[i]) return p.a[i]<q.a[i];
	}
	return true;
}

num operator -(num &p,num &q){
	int len=max(p.a[0],q.a[0]);
	for(int i=1;i<=len;++i){
		p.a[i]-=i<=q.a[0]?q.a[i]:0;
		if(p.a[i]<0) p.a[i]+=base,p.a[i+1]-=1;//借位 
	}
	while(p.a[0]>1&&p.a[p.a[0]]==0) --p.a[0];
	return p;
}

char p1[maxn],q1[maxn];
int lenp,lenq;

int main(){
	scanf("%s%s",p1,q1);
	lenp=strlen(p1);lenq=strlen(q1);
	p=num(p1,lenp),q=num(q1,lenq);
	if(q<p) ans=p-q;
	else putchar('-'),ans=q-p;
	ans.print();
	return 0;
}

高精度乘法

/*********************************************

User:Mandy.H.Y
Language:c++
Problem:luogu 1303 A*B Problem
Algorithm: High Precision Multiplication

**********************************************/

#include<bits/stdc++.h>

using namespace std;

const int power=4;
const int base=1e4;
const int maxn=2e3+5;

struct num{
	int a[maxn<<1];
	
	num(){memset(a,0,sizeof(a));}
	
	num(char *s,int len){
		memset(a,0,sizeof(a));
		
		a[0]=(len+power-1)/power;
		
		for(int t=0,w=1,i=len-1;i>=0;w=(w<<1)+(w<<3),--i){
			if((len-1-i)%power==0) w=1,++t;
			a[t]+=(s[i]^48)*w;//注意+= 
		}
	}
	
	void print(){
		printf("%d",a[a[0]]);
		for(int i=a[0]-1;i>0;--i) printf("%0*d",power,a[i]);
	}
}ans,p,q;

char p1[maxn],q1[maxn];
int lenp,lenq;

num operator *(const num &p,const num &q){
	num b;
	for(int i=1;i<=p.a[0];++i){
		int cur=0;
		for(int j=1;j<=q.a[0];++j)
			cur+=b.a[i+j-1]+p.a[i]*q.a[j],b.a[i+j-1]=cur%base,cur/=base;
			//cur+=b.a[i+j-1]+p.a[i]*q.a[j]累加 b.a[i+j-1]
		b.a[i+q.a[0]]=cur;
	}
	b.a[0]=q.a[0]+p.a[0];//得到大致位数 
	while(b.a[0]>1&&b.a[b.a[0]]==0) --b.a[0];//除去前导0 
	return b;
}

int main(){
	scanf("%s%s",p1,q1);
	lenp=strlen(p1);lenq=strlen(q1);
	p=num(p1,lenp);q=num(q1,lenq);
	ans=p*q;
	ans.print();
	return 0;
}

整合

/**************************
User:Mandy.H.Y
Language:c++
Problem:
Algorithm: High Precision
***************************/

#include<bits/stdc++.h>

using namespace std;

const int power=4;
const int base=1e4;
const int maxn=2e3+5;

struct num{
	int a[maxn<<1];
	
	num(){memset(a,0,sizeof(a));}
	int &operator [](int x){return a[x];}//重载中括号 
	num(char *s,int len){
		memset(a,0,sizeof(a));
		
		a[0]=(len+power-1)/power;
		
		for(int t=0,w=1,i=len-1;i>=0;w=(w<<1)+(w<<3),--i){
			if((len-1-i)%power==0) w=1,++t;
			a[t]+=(s[i]^48)*w;//注意+= 
		}
	}
	void add(int k) { if (k || a[0]) a[ ++a[0] ] = k; }     //在末尾添加一个数,除法的时候要用到  
    void re() { reverse(a+1, a+a[0]+1); } 
	void print(){
		printf("%d",a[a[0]]);
		for(int i=a[0]-1;i>0;--i) printf("%0*d",power,a[i]);
	}
}ans,p,q;

char p1[maxn],q1[maxn];
int lenp,lenq;


num operator +(num &p,num &q){//重载了中括号所以不用const 
	int cur=0,len=max(p[0],q[0]);
	for(int i=1;i<=len;++i){
		cur+=i<=p[0]?p[i]:0;
		cur+=i<=q[0]?q[i]:0;
		p[i]=cur%base;cur/=base;
	}
	if(cur) p[++len]=cur;
	p[0]=len;//注意更新长度 
	return p;
}


bool operator <(const num &p,const num &q){//比较大小 
	if(p.a[0]<q.a[0]) return true;
	if(p.a[0]>q.a[0]) return false;
	for(int i=1;i<=p.a[0];++i){
		if(p.a[i]!=q.a[i]) return p.a[i]<q.a[i];
	}
	return true;
}

num operator -(num &p,num &q){
	int len=max(p.a[0],q.a[0]);
	for(int i=1;i<=len;++i){
		p.a[i]-=i<=q.a[0]?q.a[i]:0;
		if(p.a[i]<0) p.a[i]+=base,p.a[i+1]-=1;//借位 
	}
	while(p.a[0]>1&&p.a[p.a[0]]==0) --p.a[0];
	return p;
}

num operator / (num &p,num &q)               //注意const不要冲突 
{  
    num x, y;  
    for (int i = p.a[0];i >= 1;--i)                       //从最高位开始取数  
    {  
        y.add(p.a[i]);             //把数添到末尾(最低位),这时候是高位在前,低位在后  
        y.re();                    //把数反过来,变为统一的存储方式:低位在前,高位在后  
        while ( !(y < q) )         //大于等于除数的时候,如果小于的话,其实答案上的该位就是初始的“0”  
            y = y - q, ++x.a[i];   //看能减几个除数,减几次,答案上该位就加几次。  
        y.re();                    //将数反过来,为下一次添数做准备  
    }  
    x.a[0] = p.a[0];  
    while (x.a[0] > 0 && !x.a[x.a[0]]) --x.a[0];  
    return x;  
}  

num operator *(const num &p,const num &q){
	num b;
	for(int i=1;i<=p.a[0];++i){
		int cur=0;
		for(int j=1;j<=q.a[0];++j)
			cur+=b.a[i+j-1]+p.a[i]*q.a[j],b.a[i+j-1]=cur%base,cur/=base;
		b.a[i+q.a[0]]=cur;
	}
	b.a[0]=q.a[0]+p.a[0];//得到大致位数 
	while(b.a[0]>1&&b.a[b.a[0]]==0) --b.a[0];//除去前导0 
	return b;
}

int main(){
	
	/*
	scanf("%s%s",p1,q1);
	lenp=strlen(p1);lenq=strlen(q1);
	p=num(p1,lenp);q=num(q1,lenq);
	ans=p*q;
	ans.print();
	return 0;
	*/
	
	
	scanf("%s%s",p1,q1);
	
	lenp=strlen(p1);lenq=strlen(q1);
	
	p=num(p1,lenp);q=num(q1,lenq);
	
	ans=p+q;
	ans.print();
	return 0;
	
	
	
	/*
	
	scanf("%s%s",p1,q1);
	lenp=strlen(p1);lenq=strlen(q1);
	p=num(p1,lenp),q=num(q1,lenq);
	if(q<p) ans=p-q;
	else putchar('-'),ans=q-p;
	ans.print();
	return 0;

	*/
	
}

练习

luogu1255

luogu1255

/************************
User:Mandy.H.Y
Language:c++
Problem:luogu 1255 Staircase
Algorithm:High Precision Addition
************************/

#include<bits/stdc++.h>

using namespace std;

const int power=4;
const int base=1e4;
const int maxn=5005;

struct num{
	int a[maxn];
	num(){memset(a,0,sizeof(a));}
	int &operator [](int x){return a[x];}
	void print(){
		printf("%d",a[a[0]]);
		for(int i=a[0]-1;i>0;--i) printf("%04d",a[i]);
	}
}ans[5];

int n;

num operator +(num &p,num &q){
	int cur=0,len=max(p[0],q[0]);
	for(int i=1;i<=len;++i){
		cur+=i<=p[0]?p[i]:0;
		cur+=i<=q[0]?q[i]:0;
		p[i]=cur%base,cur/=base;
	}
	p[0]=len;//注意继承长度 
	if(cur) p[++p[0]]=cur;
	return p;
}

int main(){
	scanf("%d",&n);
	if(n==1) {puts("1");return 0;}
	if(n==2) {puts("2");return 0;}
	ans[1].a[0]=1,ans[2].a[0]=1;ans[1].a[1]=1,ans[2].a[1]=2;
	for(int i=3;i<=n;++i){
		ans[i%3]=ans[(i+1)%3]+ans[(i+2)%3];
	}
	ans[n%3].print();
	return 0;
}

luogu1604

luogu1604

/***************************
User:Mandy.H.Y
Language:c++
Problem:luogu1604 Planet
Algorithm:High Precision
***************************/

#include<bits/stdc++.h>

using namespace std;

const int power=1;
const int maxn=2e4+5;

int n,base;

struct num{
	int a[maxn];
	num(){memset(a,0,sizeof(a));}
	int &operator [](int x){return a[x];}
	num(char *s,int len){
		a[0]=len;
		for(int i=len-1;i>=0;--i){
			if(isdigit(s[i])) a[len-i]=s[i]^48;
			else a[len-i]=s[i]-'A'+10;
		}
	}
	void print(){
		for(int i=a[0];i>0;--i){
			if(a[i]<10) printf("%d",a[i]);
			else printf("%c",a[i]-10+'A');
		}
	}
}p,q,ans;

char p1[maxn],q1[maxn];
int lenp,lenq;

num operator +(num &p,num &q){
	int cur=0,len=max(p[0],q[0]);
	for(int i=1;i<=len;++i){
		cur+=i<=p[0]?p[i]:0;
		cur+=i<=q[0]?q[i]:0;
		p[i]=cur%base,cur/=base;
	}
	if(cur) p[++len]=cur;
	p[0]=len;
	return p;
}

int main(){
	scanf("%d",&base);
	scanf("%s%s",p1,q1);
	lenp=strlen(p1);lenq=strlen(q1);
	p=num(p1,lenp);q=num(q1,lenq);
	ans=p+q;
	ans.print();
	return 0;
}

部分引自:https://www.cnblogs.com/hnqw1214/p/6351321.html

标签:q1,p1,const,高精度,int,模版,压位,num,maxn
来源: https://blog.csdn.net/weixin_44584560/article/details/89405691