zoukankan      html  css  js  c++  java
  • CodeForces

    题意:给出u,v,p,对u可以进行三种变化: 1.u=(u+1)%p ; 2.u = (u+p-1)%p;  3.u = 模p下的逆元。问通过几步可以使u变成v,并且给出每一步的操作。

    分析:朴素的bfs或dfs会超时或炸栈,考虑用双向bfs头尾同时搜。用map存每个数的访问状态和对应的操作编号,正向搜步长为正,反向搜步长为负。反向搜的时候要注意对应加减操作是反过来的。

    #include<stdio.h>
    #include<iostream>
    #include<cstring>
    #include<iostream>
    #include<queue>
    #include<map>
    #include<string>
    #include<stack>
    using namespace std;
    typedef long long LL;
    const int maxn =10;
    const int INF=0x3f3f3f3f;
    struct Node{
        int step, but;  
        LL pre;
    };
    map<LL,Node> path;
    LL u,v,p;
    
    LL fpow(LL a,LL n)
    {
        LL res=1;
        while(n){
            if(n&1) res =(res*a)%p;
            a = a*a %p;
            n>>=1;
        }
        return res;
    }
    
    void Print(LL a,LL b,int op)
    {
        printf("%d
    ",path[a].step-1-path[b].step);
        stack<int> S;
        while(a!=u){
            Node ap = path[a];
            S.push(ap.but);
            a = ap.pre;
        }
        while(!S.empty()){
            int x =S.top();S.pop();
            printf("%d ",x);
        }
        printf("%d ",op);
        while(b!=v){
            Node bp =path[b];
            printf("%d ",bp.but);
            b = bp.pre;
        }
        puts("");
    }
    
    
    void BFS()
    {
        path.clear();
        path[u]=(Node){1,-1,-1};
        path[v]=(Node){-1,-1,-1};          
        queue<LL> qf,qb;
        qf.push(u); qb.push(v);
        while(!qf.empty()|| !qb.empty()){
            if(!qf.empty()){
                LL x = qf.front(); qf.pop();
                Node xp = path[x];
                LL next = (x+1)%p;
                Node np = path[next];
                if(np.step==0){         //未访问
                    path[next]= (Node){xp.step+1,1,x};
                    qf.push(next);
                }
                else if(np.step<0){     //相遇
                    Print(x,next,1);
                    return;
                }
    
                next = (x+p-1) %p;              //op2
                np = path[next];
                if(np.step==0){
                    path[next]=  (Node){xp.step+1,2,x};
                    qf.push(next);
                }
                else if(np.step<0){
                    Print(x,next,2);
                    return;
                }
                
                next = fpow(x,p-2);
                np = path[next];
                if(np.step==0){
                    path[next] = (Node){xp.step+1,3,x};
                    qf.push(next);
                }
                else if(np.step<0){
                    Print(x,next,3);
                    return;
                }
            }
    
            if(!qb.empty()){
                LL x = qb.front(); qb.pop();
                Node xp = path[x];
    
                LL next = (x+p-1)%p;
                Node np = path[next];
                if(!np.step){
                    path[next] = (Node){xp.step-1,1,x};
                    qb.push(next);
                }
                else if(np.step>0){
                    Print(next,x,1);
                    return;
                }
    
                next = (x+1)%p;
                np = path[next];
                if(!np.step){
                    path[next] = (Node){xp.step-1,2,x};
                    qb.push(next);
                }
                else if(np.step>0){
                    Print(next,x,2);
                    return;
                }
    
                next = fpow(x,p-2);
                np = path[next];
                if(!np.step){
                    path[next] = (Node){xp.step-1,3,x};
                    qb.push(next);
                }
                else if(np.step>0){
                    Print(next,x,3);
                    return;
                }
            }
        }
    }
    
    int main()
    {
         #ifndef ONLINE_JUDGE
             freopen("in.txt","r",stdin);
             freopen("out.txt","w",stdout);
        #endif
        int N,M,tmp,T;
        while(scanf("%lld %lld %lld",&u,&v,&p)==3){
            if(u==v){
                puts("0");
                continue;
            }
            BFS();
        }
        return 0;
    }
    为了更好的明天
  • 相关阅读:
    Windows10关机问题----只有“睡眠”、“更新并重启”、“更新并关机”,但是又不想更新,解决办法
    3ds max学习笔记(九)-- 实例操作(路径阵列)
    3ds max学习笔记(八)-- 实例操作(直行楼梯)
    3ds max学习笔记(七)-- 实例操作(桌子)
    3ds max学习笔记(六)-- 基本操作(建模前奏)
    UE4入门(二)建立和打开项目
    3ds max学习笔记(五)--操作工具
    3ds max 学习笔记(四)--创建物体
    3ds max学习笔记(一)--选择物体
    欧拉回路输出(DFS,不用回溯!)Watchcow POJ 2230
  • 原文地址:https://www.cnblogs.com/xiuwenli/p/9435071.html
Copyright © 2011-2022 走看看