• AtCoder Beginner Contest 279 E.Cheating Amidakuji(思维题 swap)


    题目

    长为n(n<=2e5)的数组a,ai在[1,m)(m<=2e5)范围内,

    初始时,有数组b,bi=i,

    对于k=1,2,...,n,分别独立的对b数组执行不包含k的交换(b[ai],b[ai+1])的操作:

    例如,n=5,k=3,则需要对[1,3)∪(3,5]执行操作,

    初始时bi=i,随后,

    交换(b[a1],b[a1+1]),交换(b[a2],b[a2+1]),

    交换(b[a4,b[a4+1]),交换(b[a5],b[a5+1])

    交换后,找到1数值所在的位置,记为Sk,

    对于k=1,2,...,n,分别输出Sk的答案

    思路来源

    aging佬代码

    题解

    注意到交换操作是可结合的、可逆的

    哪个位置经历[i,n]步后换到位置j,可以通过位置j倒序执行[i,n]操作得到

    所以可以先顺序执行一遍,顺序交换,pos记录值当前所在的位置,

    to[i]表示值0在经历了前i步交换后当前的位置

    然后考虑倒序回滚操作,将前缀、后缀拼接起来

    代码

    1. #include
    2. using namespace std;
    3. const int N=2e5+10;
    4. int n,m,a[N],b[N],pos[N],to[N],ans[N];
    5. int main(){
    6. scanf("%d%d",&m,&n);
    7. for(int i=0;i
    8. scanf("%d",&a[i]);
    9. a[i]--;
    10. }
    11. for(int i=0;i
    12. b[i]=pos[i]=i;
    13. }
    14. for(int i=0;i
    15. swap(pos[b[a[i]]],pos[b[a[i]+1]]);
    16. swap(b[a[i]],b[a[i]+1]);
    17. to[i]=pos[0];
    18. }
    19. for(int i=0;i
    20. b[i]=i;
    21. }
    22. for(int i=n-1;i>=0;--i){
    23. int now=(i-1>=0?to[i-1]:0);
    24. ans[i]=b[now];
    25. swap(b[a[i]],b[a[i]+1]);
    26. }
    27. for(int i=0;i
    28. printf("%d\n",1+ans[i]);
    29. }
    30. return 0;
    31. }

  • 相关阅读:
    linux 安装 mini conda,linux下安装 Miniconda
    Vue3中插槽(slot)用法汇总
    HashMap 源码解读(JDK1.8)
    文件夹名称提取到excel,批量提取
    JVM系列之语法糖的味道
    Vue3记录
    SpringCloud文件上传
    MySQL 5.7详细下载安装配置教程(MySQL 5.7安装包)_mysql5.7的安装教程
    image图片之间的间隙消除
    Vulnhub实战-prime1
  • 原文地址:https://blog.csdn.net/Code92007/article/details/128062414