| 存储结构
//大整数类型 结构体 struct bign { int d[maxn]; int len; bign() { //构造函数初始化 len = 0; memset(d, 0, sizeof(d)); } };| 字符串 转为 大整数
//把字符串表示的大整数转换成bign类型 bign change(char str[]) { bign c; for(int i = strlen(str) - 1; i >= 0; i--) { //逆序保存,因为操作都是从低位到高位进行 c.d[c.len++] = str[i] - '0'; } return c; }| 大整数比较
//比较函数 (均正) int compare(bign a, bign b) { if(a.len > b.len) return 1; //a大 else if(a.len < b.len) return -1; //a小 else { for(int i = a.len - 1; i >= 0; i--) if(a.d[i] > b.d[i]) return 1; else if(a.d[i] < b.d[i]) return -1; return 0; //相等 } }| 高精度加法
/*高精度加法 思路:每一步: 将该位上的两个数与进位相加,结果的个位作为该位结果,十位作为新的进位 预处理:若一正一负,就转为做高精度减法;若两个负数,则绝对值相加,结果再补负号; */ bign add(bign a, bign b) { int carry = 0; bign c; for(int i = 0; i < a.len || i < b.len; i++) { //以较长的为界限(较短的高位全为0) int temp = a.d[i] + b.d[i] + carry; c.d[c.len++] = temp % 10; carry = temp / 10; } if(carry != 0) { //最后还有进位,直接赋值给最高位 c.d[c.len++] = carry; } return c; }| 高精度减法
//高精度减法 /* 思路:从低位开始,每一位判断是否够减,够减直接减,不够减从高一位借1(高位减1,被减位加10) 最后记得消去差的前导零 并 至少保留一位; 预处理 :如果被减数<减数,则交换变量,最后加负号; */ bign minus(bign a, bign b) { //默认a > b ;且均非负 (其他情况使用前先做预处理) int carry = 0; bign c; for(int i = 0; i < a.len || i < b.len; i++) { if(a.d[i] < b.d[i]){ a.d[i + 1]--; a.d[i] += 10; } c.d[c.len++] = a.d[i] - b.d[i]; } while(c.len - 1 > 0 && c.d[c.len - 1] == 0) c.len--; return c; }| 高精度 * 低精度
/* 高精度 * 低精度 思路:用高精度的每一位去乘低精度整体,再与进位相加,所得结果的个位作为该位结果, 其他高位所有部分作为新的进位 预处理:若有负数,则记录负号,取他们的绝对值带入运算 */ bign multi(bign a, int b) { int carry = 0; bign c; for(int i = 0; i < a.len; i++) { int temp = b * a.d[i] + carry; c.d[c.len++] = temp % 10; carry = temp / 10; } while(carry != 0){ //乘法进位不止是个位数 c.d[c.len++] = carry % 10; carry /= 10; } return c; }| 高精度 / 低精度
/*高精度除法 思路:每一步:(上一步的余数*10 + 该步的位)作为被除数,够除则除,不够除商零; 最后:记得消去商中的前导零,同时至少保留一位 预处理: 函数中多传一个参数引用,用于返回余数 (设为全局变量也行) */ bign divide(bign a, int b, int& r) { //r记录最后的余数 bign c; c.len = a.len; //被除数和商的每一位是一一对应的,因此先令长度相等 for(int i = a.len - 1; i >= 0; i--) { r = r * 10 + a.d[i]; //和上一位余数组成被除数 if(r < b) c.d[i] = 0; //不够除,商0 else { //够除 c.d[i] = r / b; //商 r = r % b; //新余数 } } //消去商中的前导零 同时至少保留一位 while(c.len - 1 > 0 && c.d[c.len - 1] == 0) c.len--; return c; }| 输出大整数
//输出 void print(bign a) { for(int i = a.len - 1; i >= 0; i--) printf("%d", a.d[i]); //注意:最后并没有输出换行 }应用样例:大整数除以int,返回商和余数
int main() { char str[1010]; int k, r; scanf("%s %d", str, &k); bign a = change(str); a = divide(a, k, r); print(a); printf(" %d\n", r); return 0; }* * * ——— 整理自《算法笔记》 ———-
待整理:POJ - 1220
【转自discuss】
【精简版】
int i,l,k,a,b,T,t[555],A[555]; char s[555],d[555]; main(){ for(scanf("%d",&T);T--;){ scanf("%d%d%s",&a,&b,s); for(k=i=strlen(s);0<i--;) t[k-1-i]=s[i]-(s[i]<58?48:s[i]<97?55:61); for(l=0;k;){ for(i=k;1<i--;){ t[i-1]+=t[i]%b*a; t[i]/=b; } A[l++]=t[0]%b; t[0]/=b; for(;0<k&&!t[k-1];k--); } for(d[l]=i=0;i<l;i++) d[l-1-i]=A[i]+(A[i]<10?48:A[i]<36?55:61); printf("%d %s\n%d %s\n\n",a,s,b,d); } }【解释版】
/* Problem: POJ.1220 number base conversion Time: 2017//3/12 题意:可选符号[0-9,A-Z,a-z],其中[A-Z]表示[10-35],[a-z]表示[36-61] */ #include<stdio.h> #include<string.h> #include<iostream> /* * 任意进制转换: 其过程跟十进制 转 二进制一样 * 先将数除2,得出的余数为二进制的一位,大神将商直接保存在原来的数组里面 * 当长度为0,即商为0时,结束。其他进制之间的转换同理。 */ const int maxn = 1000; int B[maxn]; // 保存转换前 旧进制 下的每一位整数值(10进制) int A[maxn]; //保存转换后 新进制 下每一位整数值(10进制)?? char original_str[maxn]; //保存读入的原始字符串 char new_str[maxn]; //保存转换后的字符串 int old_system, new_system; //要转换的两种进制 void solve() { int i, len, k; len = strlen(original_str); //字符串转为 int 逆序 数组 for(i = len; i >= 0; --i) { //[注意a-z表示36-61] B[len - 1 - i] = original_str[i] - (original_str[i] < 58 ? 48 : original_str[i]<97?55:61); } //为什么最高位还要转换?B[-1]不会越界吗? for(k = 0; len > 0;) { /*作用: 截取B[]的一位(高位),以新进制保存在A[] * 从高位(B的末尾)依次把每一位先对新进制取余(本位表示不了?那为什么要传给低位呢?因为A[k]从低往上递增, * 这样多出的部分会最后累加在B[0]中,最后会赋值给在新的A[k]), * 再乘以旧进制,累加给前一位(相当于低位),(相当于把多余的数值从高位蜕下来了,一步步的保存在了A[]中) * 最后得到的是: 用新进制除得的商 * 一次循环,得到一位数,保存在A[k]中 */ for(i = len; i >= 1; --i) { //B[len] = 0 ? //B已然是逆序了,为什么这儿有逆序操作,与其这样,之前为什么要保存成逆序?? // 这样B末尾(相当于高位)可以缩短(截取后,最高位数值变为0) B[i-1] += (B[i] % new_system * old_system); B[i] /= new_system; } A[k++] = B[0] % new_system; B[0] /= new_system; while(len > 0 && !B[len-1]) //消去高位0 同时保持至少有一位 len--; //一位一位把B[]截取完(就是十进制转二进制的竖式除法的推广) } //清空最后一个字符,作为新字符串结束的边界(覆盖上一轮保存的值) new_str[k] = NULL; for(i=0; i<k; i++) //把A中保存的int转换为char new_str[k-1-i] = A[i] + (A[i] < 10 ? 48: A[i] < 36 ? 55 : 61); } int main() { int num_of_input; // freopen("in.txt","r",stdin); scanf("%d",&num_of_input); while(num_of_input--) { scanf("%d%d%s", &old_system, &new_system, original_str); solve(); printf("%d 进制: %s\n%d 进制: %s\n\n", old_system, original_str, new_system, new_str); } system("pause"); return 0; }