poj 3708


题目描述:

抽象一下就是:先给出一个100位的大整数.转换成d<100进制的数.然后有一个同样的目标数.d进制下位数要一样.每次转换i对应i位,最高位用a,低位用b.然后问最少转换几次能够到目标数.

题解:

第一步是高精度的进制转换.第二步是100个数,每个数求出来第一次转成功的次数,以及循环数(注意,感觉题目有毛病,其实就是第二次遇到目标数-第一次的次数就是循环数,竟然可能没有循环.标准的应该是随便找个循环就行).之后解线性同余模方程.

重点:

高精度转换+找循环和次数+线性同余模方程

代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
#include <cmath>
#include <ctype.h>
#include <limits.h>
#include <cstdlib>
#include <algorithm>
#include <vector>
#include <queue>
#include <map>
#include <stack>
#include <set>
#include <bitset>
#define CLR(a) memset(a, 0, sizeof(a))
#define REP(i, a, b) for(ll i = a;i < b;i++)
#define REP_D(i, a, b) for(ll i = a;i <= b;i++)

typedef long long ll;

using namespace std;

struct P
{
    ll a, b;
};

const ll maxn = 100+10;
const ll eps = 5000;
ll res_m[maxn], res_k[maxn], ans[maxn], start[maxn];
char m[maxn], k[maxn];
ll d;
ll a[maxn], b[maxn];
ll vis[maxn], B[maxn], M[maxn];

ll gcd(ll a, ll b)
{
    if(b==0)
    {
        return a;
    }
    return gcd(b, a%b);
}
ll ext_gcd(ll a, ll b, ll &x, ll &y)
{
    if(b==0)
    {
        x = 1;
        y = 0;
        return a;
    }
    ll r = ext_gcd(b, a%b, x, y);
    ll t = x;
    x = y;
    y = t - (a/b*y);
    return r;
}

void trans(char s[], ll base1, ll base2, ll res[])//从高位开始除,答案保留在res中,并且是倒序
{
    //CLR(res);
    res[0] = 0;
    start[0] = strlen(s);
    REP(i, 0, start[0])
    {
        start[i+1] = s[i]-'0';
    }
    while(start[0] >= 1)
    {
        ll y = 0;
        ans[0] = start[0];
        REP_D(i, 1, start[0])
        {
            y = y*base1 + start[i];
            ans[i] = y/base2;
            y %= base2;
        }
        res[++res[0]] = y;
        ll i = 1;
        while(i <= ans[0]&&ans[i]==0)
        {
            i++;
        }
        start[0] = 0;
        while(i<=ans[0])
        {
            start[++start[0]] = ans[i];
            i++;
        }

    }
}

P mo_fangchengzu()//方程组
{
    ll x = 0, m=1;
    REP_D(i, 1, res_k[0])
    {
        ll a = m, b = ((B[i]-x)%M[i] + M[i])%M[i];
        ll d = gcd(a, M[i]);
        if(b%d!= 0)
        {
            return (P){0, -1};
        }
        else
        {
            ll t, temp;
            a /= d;
            b /= d;
            M[i] /= d;
            ext_gcd(a, M[i], t, temp);
            t = (t%M[i]+M[i])%M[i];
            t = (t*b)%M[i];
            x=x+m*t;
            m *= M[i];
            x %= m;
        }
    }
    return (P){x, m};
}

ll solve()//注意是ll...wa了好久这个地方
{
    trans(m, 10, d, res_m);
    trans(k, 10, d, res_k);
//    REP_D(i, 1, res_m[0])
//    {
//        prllf("%d ", res_m[i]);
//    }
//    prllf("\n");
    if(res_m[0]!=res_k[0])
    {
        return -1;
    }
    for(ll i = 1; i <= res_k[0]; i++)
    {
        //CLR(vis);
        ll from = res_m[i];
        ll to = res_k[i];
        //vis[from]=1;
        ll now = from;
        ll cnt = 0;
        ll ans = -1;
        while(1)
        {
            if(now == to&&ans==-1)//找循环
            {
                ans = cnt;
            }
            else if(now==to)
            {
                M[i] = cnt - ans;
                break;
            }

            //vis[now] = 1;
            cnt++;
            if(cnt > eps)
                return -1;
            if(i != res_k[0])
                now = b[now];
            else
                now = a[now];
        }
        //M[i] = cnt;
        if(ans==-1)
        {
            return -1;
        }
        B[i] = ans;
    }
    P out = mo_fangchengzu();
    if(out.b==-1)
    {
        return -1;
    }
    return out.a;
}


int main()
{
    //freopen("4Din.txt", "r", stdin);
    //freopen("4Dout.txt", "w", stdout);
    while(scanf("%I64d", &d) && d != -1)
    {
        REP_D(i, 1, d - 1)
        {
            scanf("%I64d", &a[i]);
        }
        REP_D(i, 0, d - 1)
        {
            scanf("%I64d", &b[i]);
        }
        scanf("%s%s", m, k);
        ll t = solve();
        if(t==-1)
        {
            printf("NO\n");
        }
        else
        {
            printf("%I64d\n", t);
        }
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值