zoukankan      html  css  js  c++  java
  • [HNOI2010]物品调度

    题目描述

    现在找工作不容易,Lostmonkey费了好大劲才得到fsk公司基层流水线操作员的职位。流水线上有n个位置,从0到n-1依次编号,一开始0号位置空,其它的位置i上有编号为i的盒子。Lostmonkey要按照以下规则重新排列这些盒子。

    规则由5个数描述,q,p,m,d,s,s表示空位的最终位置。

    首先生成一个序列c,c0=0,ci+1=(ci*q+p) mod m。

    接下来从第一个盒子开始依次生成每个盒子的最终位置posi,posi=(ci+d*xi+yi) mod n,xi,yi是为了让第i个盒子不与之前的盒子位置相同的由你设定的非负整数,且posi还不能为s。

    如果有多个xi,yi满足要求,你需要选择yi最小的,当yi相同时选择xi最小的。这样你得到了所有盒子的最终位置,现在你每次可以把某个盒子移动到空位上,移动后原盒子所在的位置成为空位。

    请问把所有的盒子移动到目的位置所需的最少步数。

    题解

    这道题有两问,如果解出第一问,第二问就比较简单了,直接统计每组置换的环长就好了。

    c[i]+x*d+y这个东西,可以看做先是找到了一个数w,然后一直往后跳d个位置(%n),直到找到第一个空的位置停下,这个东西可以想到用并查集维护。

    然后y这个东西我们考虑如何优化枚举过程,如果当前环被填满了,那么就连向下一个点。

    细节超级多,到底是%gcd还是%n还是数它在那个环里要想清楚,写错一个地方就凉凉。

    代码

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #define N 100002
    #define int long long 
    using namespace std;
    int ans,fx[N],fy[N],t,n,q,p,s,m,d,c[N],pos[N],id[N];
    bool vis[N];
    inline int rd(){
        int x=0;char c=getchar();bool f=0;
        while(!isdigit(c)){if(c=='-')f=1;c=getchar();}
        while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
        return f?-x:x;
    }
    inline int findx(int x){return fx[x]=fx[x]==x?x:findx(fx[x]);}
    inline int findy(int x){return fy[x]=fy[x]==x?x:findy(fy[x]);}
    inline int gcd(int x,int y){return y?gcd(y,x%y):x;}
    signed main(){
        t=rd();
        while(t--){
            memset(vis,0,sizeof(vis));
            memset(id,-1,sizeof(id));
            n=rd();s=rd();q=rd();p=rd();m=rd();d=rd();
            for(int i=0;i<n;++i)fx[i]=i;int g=gcd(d,n);
            for(int i=0;i<n;++i)fy[i]=i;
            for(int i=0;i<g;++i){
                int x=i;
                while(id[x]==-1)id[x]=i,x=(x+d)%n;
            }
            fx[s]=(s+d)%n;if(fx[s]==s)fy[id[s]]=(id[s]+1)%g;
            c[0]=0;ans=0;
            for(int i=1;i<n;++i)c[i]=(c[i-1]*q+p)%m;
            for(int i=1;i<n;++i){
                c[i]%=n;
                int y=findy(id[c[i]]);
                int x=((y-id[c[i]]+g)%g+c[i]+n)%n;pos[i]=findx(x);
                int xp=(pos[i]+d)%n,xx=findx(pos[i]),yy=findx(xp);
                if(xx==yy)fy[findy(id[pos[i]])]=findy((id[pos[i]]+1)%g);
                else fx[xx]=yy;
            }
            pos[0]=s;
            for(int i=0;i<n;++i)if(i!=pos[i]&&!vis[i]){
                int x=i,num=0;
                while(!vis[x])vis[x]=1,num++,x=pos[x];
                if(!i)ans+=num-1;else ans+=num+1;
            }
            printf("%lld
    ",ans);
        }
        return 0;    
    } 
  • 相关阅读:
    免费音频录制及处理软件 Audacity
    centos7设置程序开机启动方案
    tomcat开启前或者关闭前执行清理任务 servlet基础知识解决
    BigDecimal比较大小及判0处理
    File文件夹操作创建层级文件夹
    centos7设置activemq开机启动
    tomcat关闭时无法清理资源解决方案
    java数据类型和C++的对应关系 SDK开发
    centos7查询开机启动项及设置服务为开机自启动
    Entity Framework 教程
  • 原文地址:https://www.cnblogs.com/ZH-comld/p/10227240.html
Copyright © 2011-2022 走看看