BZOJ 1604 [Usaco2008 Open]Cow Neighborhoods 奶牛的邻居 Treap

题意:链接

方法: Treap

解析:

前几道资格赛的题水的不行,这道Gold的题就够分量辣。

首先这个曼哈顿距离啥的肯定能做文章,怎么转化是个问题,自己玩了一会没玩出来,就查了查曼哈顿距离的转化,发现这个玩意转化之后就变得有思路多了,所以这数学本领还是很重要啊=-=

先看曼哈顿距离的定义

|x1?x2|+|y1?y2|

拆绝对值

x1?x2+y1?y2或x1?x2+y2?y1

x2?x1+y1?y2或x2?x1+y2?y1

即|x1+y1?(x2+y2)|或|x1?y1?(x2?y2)|

设x1+y1为x′,x1?y1为y′

则|x1′?x2′|或|y1′?y2′|

所以原要求1转化为

max(|x1′?x2′|,|y1′?y2′|)<=c

这种二维的东西显然排序一下降一维。

按x’排序后,维护一个x’的队列,再对y’维护一个平衡树就行了。

至于要求2,即是并查集,也就是说平衡树每一次拿出来前驱后继维护下并查集即可。

y’显然可能重复,又维护并查集我们需要拿出来标号,所以平衡树需要多维护一个no,所以再删除的时候我们要找到v与no都跟要删除的目标节点相同的节点删除。

(前驱写挫WA一次= =!

代码:

#include <ctime>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 100010
using namespace std;
int fa[N];
int q[N];
int tot;
int root;
int cnt[N];
struct node
{
    int l,r,w,v,no,rnd,siz;
}tr[N];
void pushup(int rt)
{
    tr[rt].siz=tr[tr[rt].l].siz+tr[tr[rt].r].siz+tr[rt].w;
}
void rturn(int &rt)
{
    int t=tr[rt].l;
    tr[rt].l=tr[t].r;
    tr[t].r=rt;
    tr[t].siz=tr[rt].siz;
    pushup(rt);
    rt=t;
}
void lturn(int &rt)
{
    int t=tr[rt].r;
    tr[rt].r=tr[t].l;
    tr[t].l=rt;
    tr[t].siz=tr[rt].siz;
    pushup(rt);
    rt=t;
}
void insert(int &rt,int v,int no)
{
    if(!rt)
    {
        rt=++tot;
        tr[rt].siz=1,tr[rt].no=no,tr[rt].rnd=rand();
        tr[rt].v=v,tr[rt].w=1;
        return;
    }
    tr[rt].siz++;
    if(v<=tr[rt].v)
    {
        insert(tr[rt].l,v,no);
        if(tr[tr[rt].l].rnd<tr[rt].rnd)rturn(rt);
    }else
    {
        insert(tr[rt].r,v,no);
        if(tr[tr[rt].r].rnd<tr[rt].rnd)lturn(rt);
    }
}
void del(int &rt,int v,int no)
{
    if(!rt)return;
    tr[rt].siz--;
    if(tr[rt].v==v&&tr[rt].no==no)
    {
        if(tr[rt].l*tr[rt].r==0){rt=tr[rt].l+tr[rt].r;return;}
        else
        {
            if(tr[tr[rt].l].rnd<tr[tr[rt].r].rnd)
            {
                rturn(rt);
                del(rt,v,no);
            }else
            {
                lturn(rt);
                del(rt,v,no);
            }
        }
    }else if(v<tr[rt].v)
    {
        del(tr[rt].l,v,no);
    }else del(tr[rt].r,v,no);
}
int ans;
void q_pre(int rt,int v)
{
    if(!rt)return;
    if(v>=tr[rt].v)
    {
        ans=rt;
        q_pre(tr[rt].r,v);
    }else q_pre(tr[rt].l,v);
}
void q_sub(int rt,int v)
{
    if(!rt)return;
    if(v<tr[rt].v)
    {
        ans=rt;
        q_sub(tr[rt].l,v);
    }else q_sub(tr[rt].r,v);
}
struct point
{
    int x,y;
}pt[N];
int n,c;
int find(int x)
{
    if(x!=fa[x])return fa[x]=find(fa[x]);
    return x;
}
int cmp(point a,point b)
{
    if(a.x==b.x)return a.y<b.y;
    return a.x<b.x;
}
int main()
{
    scanf("%d%d",&n,&c);
    for(int i=1;i<=n;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        pt[i].x=x+y,pt[i].y=x-y;
        fa[i]=i;
    }
    sort(pt+1,pt+n+1,cmp);
    int head=1,tail=0;
    for(int i=1;i<=n;i++)
    {
        while(head<=tail&&pt[i].x-pt[q[head]].x>c)
        {
            del(root,pt[q[head]].y,q[head]);
            head++;
        }
        ans=0;
        q_pre(root,pt[i].y);
        int tmp=ans;
        if(tmp!=0)
        {
            if(pt[i].y-tr[tmp].v<=c)
            {
                if(find(i)!=find(tr[tmp].no))
                {
                    fa[find(i)]=find(tr[tmp].no);
                }
            }
        }
        ans=0;
        q_sub(root,pt[i].y);
        tmp=ans;
        if(tmp!=0)
        {
            if(tr[tmp].v-pt[i].y<=c)
            {
                if(find(i)!=find(tr[tmp].no))
                {
                    fa[find(i)]=find(tr[tmp].no);
                }
            }
        }
        insert(root,pt[i].y,i);
        q[++tail]=i;
    }
    int ma=0,print=0;
    for(int i=1;i<=n;i++)
    {
        int fx=find(i);
        if(!cnt[fx])
        {
            print++;
        }
        cnt[fx]++;
        if(cnt[fx]>ma)ma=cnt[fx];
    }
    printf("%d %d\n",print,ma);
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-10 14:14:31

BZOJ 1604 [Usaco2008 Open]Cow Neighborhoods 奶牛的邻居 Treap的相关文章

bzoj 1604: [Usaco2008 Open]Cow Neighborhoods 奶牛的邻居——排序+贪心+set

Description 了解奶牛们的人都知道,奶牛喜欢成群结队.观察约翰的N(1≤N≤100000)只奶牛,你会发现她们已经结成了几个"群".每只奶牛在吃草的时候有一个独一无二的位置坐标Xi,Yi(l≤Xi,Yi≤[1..10^9]:Xi,Yi∈整数.当满足下列两个条件之一,两只奶牛i和j是属于同一个群的: 1.两只奶牛的曼哈顿距离不超过C(1≤C≤10^9),即lXi - xil+IYi - Yil≤C. 2.两只奶牛有共同的邻居.即,存在一只奶牛k,使i与k,j与k均同属一个群.

bzoj 1604: [Usaco2008 Open]Cow Neighborhoods 奶牛的邻居 曼哈顿生成树

大致题意:统计平面上由曼哈顿距离小于等于c的点对组成联通块的个数. 曼哈顿生成树的模板题.有关讲解:http://blog.csdn.net/acm_cxlove/article/details/8890003 #include<iostream> #include<cstdio> #include<algorithm> #include<cstring> using namespace std; #define MAXN 100100 #define MA

【BZOJ】1604: [Usaco2008 Open]Cow Neighborhoods 奶牛的邻居

[算法]并查集+平衡树+数学 [题解] 经典哈夫曼距离转切比雪夫距离. 哈夫曼距离:S=|x1-x2|+|y1-y2| 即:max(x1-x2+y1-y2,x1-x2-y1+y2,-x1+x2+y1-y2,-x1+x2-y1+y2) 令X1=x1+y1,Y1=x1-y1, 切比雪夫距离:S=max(|X1-X2|,|Y1-Y2|). 转化为

bzoj1604[Usaco2008 Open]Cow Neighborhoods 奶牛的邻居*

bzoj1604[Usaco2008 Open]Cow Neighborhoods 奶牛的邻居 题意: n只牛,牛结成群当且仅当两只牛曼哈顿距离≤c或存在第三头牛使两头牛与它的曼哈顿距离都≤c,求最大的群和群数.n≤100000 题解: 好神啊.先把曼哈顿距离转成切比雪夫距离,(x,y)转为(x+y,x-y).然后按x坐标排序,用两个指针维护使x坐标差值≤c,同时将新插入的y坐标放入set,每次在set里查找出与当前y差值不超过c的最大y,将这两个点合并成一个集合,用并查集维护. 代码: 1 #

【bzoj1604】[Usaco2008 Open]Cow Neighborhoods 奶牛的邻居 并查集+Treap/STL-set

题目描述 了解奶牛们的人都知道,奶牛喜欢成群结队.观察约翰的N(1≤N≤100000)只奶牛,你会发现她们已经结成了几个“群”.每只奶牛在吃草的时候有一个独一无二的位置坐标Xi,Yi(l≤Xi,Yi≤[1..10^9]:Xi,Yi∈整数.当满足下列两个条件之一,两只奶牛i和j是属于同一个群的: 1.两只奶牛的曼哈顿距离不超过C(1≤C≤10^9),即lXi - xil+IYi - Yil≤C. 2.两只奶牛有共同的邻居.即,存在一只奶牛k,使i与k,j与k均同属一个群. 给出奶牛们的位置,请计算

bzoj 1623: [Usaco2008 Open]Cow Cars 奶牛飞车

1623: [Usaco2008 Open]Cow Cars 奶牛飞车 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 325  Solved: 223[Submit][Status][Discuss] Description 编号为1到N的N只奶牛正各自驾着车打算在牛德比亚的高速公路上飞驰.高速公路有M(1≤M≤N)条车道.奶牛i有一个自己的车速上限Si(l≤Si≤1,000,000). 在经历过糟糕的驾驶事故之后,奶牛们变得十分小心,避免碰撞的发生

BZOJ 1623 [Usaco2008 Open]Cow Cars 奶牛飞车:贪心

题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1623 题意: 编号为1到N的N只奶牛正各自驾着车打算在牛德比亚的高速公路上飞驰.高速公路有M(1≤M≤N)条车道.奶牛i有一个自己的车速上限Si(l≤Si≤1,000,000). 在经历过糟糕的驾驶事故之后,奶牛们变得十分小心,避免碰撞的发生. 每条车道上,如果某一只奶牛i的前面有K只奶牛驾车行驶,那奶牛i的速度上限就会下降K*D个单位,也就是说,她的速度不会超过Si - k*D(O≤D

1623: [Usaco2008 Open]Cow Cars 奶牛飞车

1623: [Usaco2008 Open]Cow Cars 奶牛飞车 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 291  Solved: 201[Submit][Status][Discuss] Description 编号为1到N的N只奶牛正各自驾着车打算在牛德比亚的高速公路上飞驰.高速公路有M(1≤M≤N)条车道.奶牛i有一个自己的车速上限Si(l≤Si≤1,000,000). 在经历过糟糕的驾驶事故之后,奶牛们变得十分小心,避免碰撞的发生

[BZOJ] 1616: [Usaco2008 Mar]Cow Travelling游荡的奶牛

1616: [Usaco2008 Mar]Cow Travelling游荡的奶牛 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1312  Solved: 736[Submit][Status][Discuss] Description 奶牛们在被划分成N行M列(2 <= N <= 100; 2 <= M <= 100)的草地上游走,试图找到整块草地中最美味的牧草.Farmer John在某个时刻看见贝茜在位置 (R1, C1),恰好T