zoukankan      html  css  js  c++  java
  • 洛谷 1966 火柴排队——思路

    题目:https://www.luogu.org/problemnew/show/P1966

    首先假想是不是两边都按升序排列是差的和最小的。

      举了几个例子发现没问题。具体分析了一下,如果有两个相邻元素在两列的对应位置都是升序,交换一下只会不变或变得更差。(可以分类看,比如a1<b1,a2<b1或a1<b1,b1<a2<b2等等)

    所以就是对应排名的值要放在对应位置。这就是题目给的值两两不同的原因。离散化一下,题目转化为有两个1~n的排列,使它们对应位置的值相等,最少要交换多少次。

    之前做过题,知道相邻交换一下可以使逆序对数减且仅减1。怎么转化?

      考虑发现两行的操作完全是等价的。也就是一行的一个操作可以等价转换成另一行的一个操作。这样只用操作一行即可。

      就想到可以以一行的值的位置为基准给另一行的值重新赋值,赋成这个值在第一行中的位置。这样问题就转化成第二行中有多少个逆序对。树状数组求解即可。

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    const int N=1e5+5,mod=99999997;
    int n,a[N],b[N],tp[N],ps[N],f[N],ans;
    int rdn()
    {
      int ret=0,fx=1; char ch=getchar();
      while(ch>'9'||ch<'0') {if(ch=='-') fx=-1; ch=getchar();}
      while(ch>='0'&&ch<='9') ret=(ret<<3)+(ret<<1)+ch-'0',ch=getchar();
      return ret*fx;
    }
    void upd(int &x){x-=(x>=mod?mod:0);}
    int query(int x)
    {
      int ret=0;
      for(;x;x-=(x&-x)) ret+=f[x],upd(ret);
      return ret;
    }
    void add(int x)
    {
      for(;x<=n;x+=(x&-x)) f[x]++;
    }
    int main()
    {
      n=rdn();
      for(int i=1;i<=n;i++) a[i]=tp[i]=rdn();
      sort(tp+1,tp+n+1);
      for(int i=1;i<=n;i++) a[i]=lower_bound(tp+1,tp+n+1,a[i])-tp;
      for(int i=1;i<=n;i++) b[i]=tp[i]=rdn();
      sort(tp+1,tp+n+1);
      for(int i=1;i<=n;i++) b[i]=lower_bound(tp+1,tp+n+1,b[i])-tp,ps[b[i]]=i;
      for(int i=1;i<=n;i++)
        {
          ans+=query(n-ps[a[i]]+1); upd(ans);
          add(n-ps[a[i]]+1);
        }
      printf("%d
    ",ans);
      return 0;
    }
  • 相关阅读:
    很简单的字节转换函数
    PHP获取用户操作系统信息
    PHP调用COM获得服务器硬件信息
    杂碎记录
    Math类使用记录
    hbase命令使用记录
    shell脚本学习
    多个job存依赖关系如何使用
    hbase的API并且使用多个rowkey分段直接读取数据
    shell学习记录
  • 原文地址:https://www.cnblogs.com/Narh/p/9646833.html
Copyright © 2011-2022 走看看