C++大数模板

来源:互联网 发布:菜鸟网络航空港区地址 编辑:程序博客网 时间:2024/06/06 05:04

转自http://blog.csdn.net/hackbuteer1/article/details/6595881

分别使用C++中的运算符重载的方法来实现大数之间的数学运算,包括加法、减法、乘法、除法、n次方、取模、大小比较、赋值以及输入流、输出流的重载。。

并且使用这个大数模板,顺利AC了HDOJ上的1134这个题目的Catalan数计数问题。。

http://acm.hdu.edu.cn/showproblem.php?pid=1134


大数模板的代码如下:


#include<iostream>#include<algorithm>#include<cstdio>#include<cstring>using namespace std;#define MAXN 9999#define MAXSIZE 10#define DLEN 4class BigNum{private:int a[500];    //可以控制大数的位数int len;       //大数长度public:BigNum(){ len = 1;memset(a,0,sizeof(a)); }   //构造函数BigNum(const int);       //将一个int类型的变量转化为大数BigNum(const char*);     //将一个字符串类型的变量转化为大数BigNum(const BigNum &);  //拷贝构造函数BigNum &operator=(const BigNum &);   //重载赋值运算符,大数之间进行赋值运算friend istream& operator>>(istream&,  BigNum&);   //重载输入运算符friend ostream& operator<<(ostream&,  BigNum&);   //重载输出运算符BigNum operator+(const BigNum &) const;   //重载加法运算符,两个大数之间的相加运算BigNum operator-(const BigNum &) const;   //重载减法运算符,两个大数之间的相减运算BigNum operator*(const BigNum &) const;   //重载乘法运算符,两个大数之间的相乘运算BigNum operator/(const int   &) const;    //重载除法运算符,大数对一个整数进行相除运算BigNum operator^(const int  &) const;    //大数的n次方运算int    operator%(const int  &) const;    //大数对一个int类型的变量进行取模运算bool   operator>(const BigNum & T)const;   //大数和另一个大数的大小比较bool   operator>(const int & t)const;      //大数和一个int类型的变量的大小比较void print();       //输出大数};BigNum::BigNum(const int b)     //将一个int类型的变量转化为大数{int c,d = b;len = 0;memset(a,0,sizeof(a));while(d > MAXN){c = d - (d / (MAXN + 1)) * (MAXN + 1);d = d / (MAXN + 1);a[len++] = c;}a[len++] = d;}BigNum::BigNum(const char*s)     //将一个字符串类型的变量转化为大数{int t,k,index,l,i;memset(a,0,sizeof(a));l=strlen(s);len=l/DLEN;if(l%DLEN)len++;index=0;for(i=l-1;i>=0;i-=DLEN){t=0;k=i-DLEN+1;if(k<0)k=0;for(int j=k;j<=i;j++)t=t*10+s[j]-'0';a[index++]=t;}}BigNum::BigNum(const BigNum & T) : len(T.len)  //拷贝构造函数{int i;memset(a,0,sizeof(a));for(i = 0 ; i < len ; i++)a[i] = T.a[i];}BigNum & BigNum::operator=(const BigNum & n)   //重载赋值运算符,大数之间进行赋值运算{int i;len = n.len;memset(a,0,sizeof(a));for(i = 0 ; i < len ; i++)a[i] = n.a[i];return *this;}istream& operator>>(istream & in,  BigNum & b)   //重载输入运算符{char ch[MAXSIZE*4];int i = -1;in>>ch;int l=strlen(ch);int count=0,sum=0;for(i=l-1;i>=0;){sum = 0;int t=1;for(int j=0;j<4&&i>=0;j++,i--,t*=10){sum+=(ch[i]-'0')*t;}b.a[count]=sum;count++;}b.len =count++;return in;}ostream& operator<<(ostream& out,  BigNum& b)   //重载输出运算符{int i;cout << b.a[b.len - 1];for(i = b.len - 2 ; i >= 0 ; i--){cout.width(DLEN);cout.fill('0');cout << b.a[i];}return out;}BigNum BigNum::operator+(const BigNum & T) const   //两个大数之间的相加运算{BigNum t(*this);int i,big;      //位数big = T.len > len ? T.len : len;for(i = 0 ; i < big ; i++){t.a[i] +=T.a[i];if(t.a[i] > MAXN){t.a[i + 1]++;t.a[i] -=MAXN+1;}}if(t.a[big] != 0)t.len = big + 1;elset.len = big;return t;}BigNum BigNum::operator-(const BigNum & T) const   //两个大数之间的相减运算{int i,j,big;bool flag;BigNum t1,t2;if(*this>T){t1=*this;t2=T;flag=0;}else{t1=T;t2=*this;flag=1;}big=t1.len;for(i = 0 ; i < big ; i++){if(t1.a[i] < t2.a[i]){j = i + 1;while(t1.a[j] == 0)j++;t1.a[j--]--;while(j > i)t1.a[j--] += MAXN;t1.a[i] += MAXN + 1 - t2.a[i];}elset1.a[i] -= t2.a[i];}t1.len = big;while(t1.a[len - 1] == 0 && t1.len > 1){t1.len--;big--;}if(flag)t1.a[big-1]=0-t1.a[big-1];return t1;}BigNum BigNum::operator*(const BigNum & T) const   //两个大数之间的相乘运算{BigNum ret;int i,j,up;int temp,temp1;for(i = 0 ; i < len ; i++){up = 0;for(j = 0 ; j < T.len ; j++){temp = a[i] * T.a[j] + ret.a[i + j] + up;if(temp > MAXN){temp1 = temp - temp / (MAXN + 1) * (MAXN + 1);up = temp / (MAXN + 1);ret.a[i + j] = temp1;}else{up = 0;ret.a[i + j] = temp;}}if(up != 0)ret.a[i + j] = up;}ret.len = i + j;while(ret.a[ret.len - 1] == 0 && ret.len > 1)ret.len--;return ret;}BigNum BigNum::operator/(const int & b) const   //大数对一个整数进行相除运算{BigNum ret;int i,down = 0;for(i = len - 1 ; i >= 0 ; i--){ret.a[i] = (a[i] + down * (MAXN + 1)) / b;down = a[i] + down * (MAXN + 1) - ret.a[i] * b;}ret.len = len;while(ret.a[ret.len - 1] == 0 && ret.len > 1)ret.len--;return ret;}int BigNum::operator %(const int & b) const    //大数对一个int类型的变量进行取模运算{int i,d=0;for (i = len-1; i>=0; i--){d = ((d * (MAXN+1))% b + a[i])% b;}return d;}BigNum BigNum::operator^(const int & n) const    //大数的n次方运算{BigNum t,ret(1);int i;if(n<0)exit(-1);if(n==0)return 1;if(n==1)return *this;int m=n;while(m>1){t=*this;for( i=1;i<<1<=m;i<<=1){t=t*t;}m-=i;ret=ret*t;if(m==1)ret=ret*(*this);}return ret;}bool BigNum::operator>(const BigNum & T) const   //大数和另一个大数的大小比较{int ln;if(len > T.len)return true;else if(len == T.len){ln = len - 1;while(a[ln] == T.a[ln] && ln >= 0)ln--;if(ln >= 0 && a[ln] > T.a[ln])return true;elsereturn false;}elsereturn false;}bool BigNum::operator >(const int & t) const    //大数和一个int类型的变量的大小比较{BigNum b(t);return *this>b;}void BigNum::print()    //输出大数{int i;cout << a[len - 1];for(i = len - 2 ; i >= 0 ; i--){cout.width(DLEN);cout.fill('0');cout << a[i];}cout << endl;}int main(){int i,n;BigNum x[101];      //定义大数的对象数组x[0]=1;for(i=1;i<101;i++)x[i]=x[i-1]*(4*i-2)/(i+1);while(~scanf("%d",&n) && n!=-1){x[n].print();}return 0;}




0 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 淘宝换货没收到怎么办 淘宝卖货让人换货了怎么办 淘宝换货关闭了怎么办 物流透露客户信息怎么办 淘宝信息泄漏后怎么办 淘宝信息泄漏了怎么办 天猫贷款还不上怎么办 车贷下不来定金怎么办 建行车贷不通过怎么办 天猫介入失败怎么办 天猫投诉没用怎么办 天猫被投诉商标侵权怎么办 虚假发货有天猫红包怎么办 淘宝代购是假货怎么办 闲鱼对方不在怎么办 小米商城退款慢怎么办 小米手机第三方拿货是怎么办 oppo手机卡被锁怎么办 下巴粉刺特别多怎么办 苹果6sp手机卡怎么办 苹果手机无服务怎么办 京东买电脑没发票怎么办 买东西发票丢了怎么办 在天猫上买了假货怎么办 苹果发票丢了怎么办 iphone8屏幕摔了怎么办 在手机店买到翻新机怎么办 信用卡网上买东西退款怎么办 在唯品会买到假的护肤品怎么办 天猫买东西发票怎么办 支付宝无法收款怎么办 买到苹果假货怎么办 16周岁怎么办手机分期 淘宝打假扣分了怎么办 买到不合格食品怎么办 买到不合格面膜怎么办 买假货怎么处理怎么办 咸鱼买手机被骗怎么办 华为p9超级卡怎么办 花呗分期退货怎么办 华为荣耀10网咯好卡怎么办