[USACO17FEB]Why Did the Cow Cross the Road III P(CDQ分治)

题意

两列$n$的排列,相同的数连边,如果一对数有交叉且差的绝对值$>k$,则$++ans$,求$ans$

题解

可以把每一个数字看成一个三元组$(x,y,z)$,其中$x$表示在第一列的位置,$y$表示在第二列的位置,$z$表示权值

两条线交叉,就是$x<x‘$且$y>y‘$,又要满足差值的绝对值小于等于$k$,就是$|z-z‘|<=k$

于是就转化为了一个三维偏序问题,直接上CDQ

具体细节请看代码

 1 // luogu-judger-enable-o2
 2 //minamoto
 3 #include<iostream>
 4 #include<cstdio>
 5 #include<algorithm>
 6 #define ll long long
 7 using namespace std;
 8 #define getc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
 9 char buf[1<<21],*p1=buf,*p2=buf;
10 template<class T>inline bool cmax(T&a,const T&b){return a<b?a=b,1:0;}
11 template<class T>inline bool cmin(T&a,const T&b){return a>b?a=b,1:0;}
12 inline int read(){
13     #define num ch-‘0‘
14     char ch;bool flag=0;int res;
15     while(!isdigit(ch=getc()))
16     (ch==‘-‘)&&(flag=true);
17     for(res=num;isdigit(ch=getc());res=res*10+num);
18     (flag)&&(res=-res);
19     #undef num
20     return res;
21 }
22 const int N=1e5+5;
23 int n,k,b[N];
24 ll ans;int c[N];
25 inline void add(int x){
26     for(;x<=n;x+=x&-x) c[x]+=1;
27 }
28 inline void clear(int x){
29     for(;x<=n;x+=x&-x)
30     if(c[x]) c[x]=0;
31     else break;
32 }
33 inline int query(int x){
34     int res=0;x=x>n?n:x;if(x<=0) return 0;
35     for(;x;x-=x&-x) res+=c[x];
36     return res;
37 }
38 struct node{
39     int x,y,z;
40     node(){}
41     node(int x,int y,int z):x(x),y(y),z(z){}
42     inline bool operator <(const node &b)const
43     {return x!=b.x?x<b.x:
44             y!=b.y?y>b.y:
45             z<b.z;}
46 }a[N],p[N];
47 void CDQ(int l,int r){
48     if(l==r) return;
49     int mid=l+r>>1;
50     CDQ(l,mid),CDQ(mid+1,r);
51     for(int j=mid+1,i=l;j<=r;++j){
52         while(i<=mid&&a[i].y>a[j].y) add(a[i++].z);
53         ans+=1ll*query(a[j].z-k-1)+query(n)-query(a[j].z+k);
54     }
55     for(int i=l;i<=mid;++i) clear(a[i].z);
56     for(int i=l,j=l,k=mid+1;i<=r;){
57         if(k>r||(j<=mid&&a[j].y>a[k].y)) p[i++]=a[j++];
58         else p[i++]=a[k++];
59     }
60     for(int i=l;i<=r;++i) a[i]=p[i];
61 }
62 int main(){
63     //freopen("testdata.in","r",stdin);
64     n=read(),k=read();
65     for(int i=1;i<=n;++i){
66         int x=read();b[x]=i;
67     }
68     for(int i=1;i<=n;++i){
69         int x=read();
70         a[i]=node(b[x],i,x);
71     }
72     sort(a+1,a+1+n);
73     CDQ(1,n);
74     printf("%lld",ans);
75     return 0;
76 }

原文地址:https://www.cnblogs.com/bztMinamoto/p/9462193.html

时间: 2024-10-04 05:09:16

[USACO17FEB]Why Did the Cow Cross the Road III P(CDQ分治)的相关文章

洛谷 P3660 [USACO17FEB]Why Did the Cow Cross the Road III G(树状数组)

题目背景 给定长度为2N的序列,1~N各处现过2次,i第一次出现位置记为ai,第二次记为bi,求满足ai<aj<bi<bj的对数 题目描述 The layout of Farmer John's farm is quite peculiar, with a large circular road running around the perimeter of the main field on which his cows graze during the day. Every morn

洛谷 P3662 [USACO17FEB]Why Did the Cow Cross the Road II S

P3662 [USACO17FEB]Why Did the Cow Cross the Road II S 题目描述 The long road through Farmer John's farm has NN crosswalks across it, conveniently numbered 1 \ldots N1…N (1 \leq N \leq 100,0001≤N≤100,000). To allow cows to cross at these crosswalks, FJ in

Why Did the Cow Cross the Road III(树状数组)

Why Did the Cow Cross the Road III 时间限制: 1 Sec  内存限制: 128 MB提交: 65  解决: 28[提交][状态][讨论版] 题目描述 The layout of Farmer John's farm is quite peculiar, with a large circular road running around the perimeter of the main field on which his cows graze during

[USACO17FEB] Why Did the Cow Cross the Road I P (树状数组求逆序对 易错题)

题目大意:给你两个序列,可以序列进行若干次旋转操作(两个都可以转),对两个序列相同权值的地方连边,求最少的交点数 记录某个值在第一个序列的位置,再记录第二个序列中某个值 在第一个序列出现的位置 ,求逆序对数量即可 本以为是一道逆序对水题,结果被卡了20分.看了题解才恍然大悟,实际上,序列可以旋转 ≠ 序列成环,由于逆序对的特殊性(并不适用于环),故不能把一个序列单独旋转看成它们的相对移动,正着旋转一个序列≠反着旋转另一个序列(更详细证明可以看洛谷) 所以我们要对两个序列再反着进行一次同样的操作

bzoj 4990 [USACO17FEB] Why Did the Cow Cross the Road II P (树状数组优化DP)

题目大意:给你两个序列,你可以两个序列的点之间连边 要求:1.只能在点权差值不大于4的点之间连边 2.边和边不能相交 3.每个点只能连一次 设表示第一个序列进行到 i,第二个序列进行到 j,最多连的边数,容易得到方程: 不连边: 连边: 实际是这样的,每个位置如果想连边,就要从能连边的位置之前找最大值,即 直接转移不可取,由于最多只从9个位置转移,我们可以缩减一维,用记录b序列进行到位置 j 的最大连边数,再用树状数组维护的最大前缀和方便转移 1 #include <bits/stdc++.h>

BZOJ4994 [Usaco2017 Feb]Why Did the Cow Cross the Road III 树状数组

欢迎访问~原文出处--博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ4994 题意概括 给定长度为2N的序列,1~N各处现过2次,i第一次出现位置记为ai,第二次记为bi,求满足ai<aj<bi<bj的对数. n<=100000(这个数据范围是我凑出来的,但是我没试过更小的范围,BZOJ上没写数据范围(截止2017-08-24)) 题解 水题,开一个树状数组在线解决. 比如我们顺着扫过去,当到达一个 bj 时,我们求满足条件的 ai,bi 个数,其实就

[BZOJ4994] [Usaco2017 Feb]Why Did the Cow Cross the Road III(树状数组)

传送门 1.每个数的左右位置预处理出来,按照左端点排序,因为左端点是从小到大的,我们只需要知道每条线段包含了多少个前面线段的右端点即可,可以用树状数组 2.如果 ai < bj < bi, 也就是说在两个i之间只有一个j那么对答案的贡献为1,所以可以用树状数组,当第一次出现一个数的时候,那个位置+1,当第二次出现的时候,第一次出现的位置再-1,也就是把它对答案的贡献去掉,同样是树状数组统计答案 #include <cstdio> #include <iostream>

【bzoj4994】[Usaco2017 Feb]Why Did the Cow Cross the Road III 树状数组

题目描述 给定长度为2N的序列,1~N各处现过2次,i第一次出现位置记为ai,第二次记为bi,求满足ai<aj<bi<bj的对数 样例输入 4 3 2 4 4 1 3 2 1 样例输出 3 题解 树状数组 WH说是CDQ分治直接把我整蒙了... 把所有数按照第一次出现位置从小到大排序,然后扫一遍.此时一定是满足$a_j>a_i$的. 那么只需要求出$(a_j,b_j)$中以前出现过的$b_i$的数目.使用树状数组维护即可. 时间复杂度$O(n\log n)$ #include &l

Why Did the Cow Cross the Road III HYSBZ - 4991 -CDQ-逆序数

HYSBZ - 4991 题意: 第一列 1-n的排列 ,第二列 1-n的排列.  相同数字连边  ,问  有多少组 数字 是有交点的并且 绝对值之差>K思路:处理一下 1-n 在第一列的位置,1-n在第二列的位置.按照第一列的位置从小到大排序,然后 进行cdq分治,因为现在第一列已经是递增序列了,如果在第二列中出现了递减那么这两个数就有交点,分治解决,递归左区间 的必然第一列必然小于递归右区间.所以只处理左区间对右区间的影响,两段小区间分别按照 b 从大到小排序,然后 统计 左区间的b  比右