小胖的奇偶(Viojs1112)题解

原题:

题目描述

   huyichen和xuzhenyi在玩一个游戏:他写一个由0和1组成的序列。 huyichen选其中的一段(比如第3位到第5位),问他这段里面有奇数个1 还是偶数个1。xuzhenyi回答你的问题,然后huyichen继续问。 xuzhenyi有可能在撒谎。huyichen要检查xuzhenyi的答案,指出在xuzhenyi的第几个回答一定有问题。 有问题的意思就是存在一个01序列满足这个回答前的所有回答,而且不存在序列 满足这个回答前的所有回答及这个回答。

输入格式

第1行一个整数,是这个01序列的长度1000000000)(≤1000000000) 第2行一个整数,是问题和答案的个数(5000)(≤5000)。 第3行开始是问题和答案, 每行先有两个整数,表示你询问的段的开始位置和结束位置。 然后是xuzhenyi的回答。odd表示有奇数个1,even表示有偶数个

输出格式

输出一行,一个数X,表示存在一个01序列满足第1到第X个回答, 但是不存在序列满足第1到第X+1个回答。如果所有回答都没问题,你就输出 所有回答的个数。


很明显,这道题是并查集(不要问我怎么看出来的)。

由于数据太大,需要离散化。

所谓离散化,就是把较大的一组数进行排序,然后对他们根据大小重新赋值,当数的大小不对数据本身有影响,只有大小对其有影响时,可以使用离散化。

离散化的标准姿势见 点这里

然而蒟蒻并不会标准姿势,说说自己的方法:

以本题为例,把所有的数据都存在nd结构体中

void init()
{
    for(int i = 1;i<=m;i++)
    {
        int x,y;
        scanf("%d%d%s",&x,&y,c);
        nd[i].p = x;//数组开三倍,p表示临时变量。一倍存左端点。
        nd[i+m].p = y;//二倍存右端点。
        nd[i].num = i;//每一倍都要给他们打上编号,一会排序时后会乱。
        nd[i+m].num = i;if(c[0]=='e')
        {
            nd[i+m+m].odd = 0;//三倍存奇偶
        }else
        {
            nd[i+m+m].odd = 1;
        }
    }
}

读进来之后进行离散化。

void discretization()
{
    for(int i = 1;i<=2*m;i++)
    {
        if(nd[nd[i].num].l==-1)
        {
            nd[nd[i].num].l = cnt-1;
        }else 
        {
            nd[nd[i].num].r = cnt;
        }
        if(nd[i+1].p!=nd[i].p)
        {
            cnt++; //去重,不一样的时候再+1   
        }
    }
    for(int i = 1;i<=m;i++)
    {
        nd[i].odd = nd[i+m+m].odd;从三倍搬回原数组
    }
}

这样就离散完了。

接下来就是带权并查集的操作。

在合并和路径压缩的时候处理。

思路就是将一个区间右面的点认左面的点为父亲,两端区间合并的时候,根据奇偶性判断,0表示偶数,1表示奇数(有点懒,具体0或1见代码)。

int find(int x)
{
    if(f[x]==x)
    {
        return f[x];
    }
    int fx = find(f[x]);
    g[x] = (g[f[x]]+g[x])%2;
    return f[x] = fx;
}

最后上总代码:

#include<cstdio>
#include<algorithm>
#include<cmath>
#define ll long long
#define N 105550
using namespace std;
char c[20];
ll n;
int m;
int cnt = 1;
struct node
{
    int l;
    int r;
    int num;
    int odd;
    int p;
}nd[N+N+N];
int f[N];
int g[N];
int cmp(node a,node b)
{
    return a.p < b.p;
}
void init()
{
    for(int i = 1;i<=m;i++)
    {
        int x,y;
        scanf("%d%d%s",&x,&y,c);
        nd[i].p = x;
        nd[i+m].p = y;
        nd[i].num = i;
        nd[i+m].num = i;
        nd[i].l = -1;
        nd[i+m].l = -1;
        if(c[0]=='e')
        {
            nd[i+m+m].odd = 0;
        }else
        {
            nd[i+m+m].odd = 1;
        }
    }
}
void discretization()
{
    for(int i = 1;i<=2*m;i++)
    {
        if(nd[nd[i].num].l==-1)
        {
            nd[nd[i].num].l = cnt-1;
        }else 
        {
            nd[nd[i].num].r = cnt;
        }
        if(nd[i+1].p!=nd[i].p)
        {
            cnt++;    
        }
    }
    for(int i = 1;i<=m;i++)
    {
        nd[i].odd = nd[i+m+m].odd;
    }
}
int find(int x)
{
    if(f[x]==x)
    {
        return f[x];
    }
    int fx = find(f[x]);
    g[x] = (g[f[x]]+g[x])%2;
    return f[x] = fx;
}
int ans;
bool flag;
void uion(int x,int y,int i)
{
    int fx = find(x);
    int fy = find(y);
    if(fx==fy)
    {
        if((nd[i].odd==1&&g[x]==g[y])||(nd[i].odd==0&&g[x]!=g[y]))
        {
            ans = i-1;
            flag = 1;
            return ;
        }
        find(y);
        find(x);
    }else
    {
        f[fy] = fx;
        g[fy] = (g[x]+g[y]+nd[i].odd)%2;
        find(x);
        find(y);        
    }
}
void solve()
{
    for(int i = 1;i<=cnt;i++)
    {
        f[i] = i;
    }
    for(int i = 1;i<=m;i++)
    {
        int l = nd[i].l;
        int r = nd[i].r;
        uion(l,r,i);
        if(flag==1)
        {
            printf("%d\n",ans);
            return ;
        }
    }
    printf("%d\n",m);
}
int main()
{
    scanf("%lld",&n);
    scanf("%d",&m);
    init();
    sort(nd+1,nd+1+m+m,cmp);
    discretization();
    solve();
    return 0;
}

 

posted @ 2018-12-04 16:01  lizitong  阅读(746)  评论(0编辑  收藏  举报