【小算法整理】- 高精度运算

高精度运算

| 存储结构

//大整数类型 结构体 
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;  
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值