`
java-mans
  • 浏览: 11813180 次
文章分类
社区版块
存档分类
最新评论

POJ 1733 并查集 偏移向量

阅读更多

这题也是用到了偏移向量

一个由0,1组成的数字串~~,现在你问一个人一些问题,第i位到第j位的1的个数为奇数还是偶数。他会告诉你答案, 但是答案可能会自相矛盾,现在就是最多能有前几个回答是不矛盾的。

设r[i]表示第1位到第i位的1个数的奇偶状况,r[i] = 0表示有偶数个1,r[i] = 1表示有奇数个1。
那么要是第i位到第j位为偶数个1时,r[i-1] = 1, r[j] = 1 或r[i-1] = 0, r[j] = 0 所以 r[i-1] ^ r[j] = 0
要是为奇数个1时,r[i-1] = 1, r[j] = 0 或 r[i-1] = 0, r[j] = 1所以r[i-1] ^ r[j] = 1

那么我们可以使用并查集,用一个数组d[i]代表r[i]与其祖先的异或值

如果回答的两个区间的端点以前都出现过,那么就判断异或值是否与以前的那个相等

如果没出现过,就直接合并,更新异或值

#include <iostream>
#include <vector>
#include <map>
#include <set>
#include <queue>
#include <stack>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <string>
#include <cstring>
#include <cmath>
#define MAXN 22222
#define MAXM 55555
#define INF 100000000
using namespace std;
int n, m;
int fa[MAXN];
int l[MAXN], r[MAXN], c[MAXN];
int x[MAXN], d[MAXN];
int find(int x)
{
    if(fa[x] == x) return x;
    int t = find(fa[x]);
    d[x] ^= d[fa[x]];
    fa[x] = t;
    return t;
}
bool join(int x, int y, int v)
{
    int fx = find(x);
    int fy = find(y);
    if(fx == fy) return ((d[x] ^ d[y]) == v);
    else
    {
        fa[fx] = fy;
        d[fx] = d[x] ^ d[y] ^ v;
        return 1;
    }
}
int bin(int low, int high, int v)
{
    while(low <= high)
    {
        int mid = (low + high) >> 1;
        if(x[mid] == v) return mid;
        else if(x[mid] > v) high = mid - 1;
        else low = mid + 1;
    }
    return -1;
}
int main()
{
    char s[15];
    scanf("%d%d", &n, &m);
    int cnt = 0;
    for(int i = 1; i <= m; i++)
    {
        scanf("%d%d%s", &l[i], &r[i], s);
        x[cnt++] = l[i] - 1;
        x[cnt++] = r[i];
        if(s[0] == 'e') c[i] = 0;
        else c[i] = 1;
    }
    sort(x, x + cnt);
    cnt = unique(x, x + cnt) - x;
    for(int i = 1; i < MAXN; i++) fa[i] = i, d[i] = 0;
    int fg = 0;
    for(int i = 1; i <= m; i++)
    {
        int L = bin(0, cnt - 1, l[i] - 1) + 1;
        int R = bin(0, cnt - 1, r[i]) + 1;
        if(!join(L, R, c[i]))
        {
            fg = 1;
            printf("%d\n", i - 1);
            break;
        }
    }
    if(!fg) printf("%d\n", m);
    return 0;
}


分享到:
评论

相关推荐

    poj2492并查集应用的扩展

    poj2492 A Bug's Life并查集应用的扩展,希望可以给大家带来用处

    POJ 1988 简单并查集,

    根据给定的信息,本文将详细解释“POJ 1988 简单并查集”中的核心知识点,包括并查集的基本概念、代码实现以及在本题中的具体应用。 ### 并查集基本概念 并查集是一种用于处理一些不交集的合并及查询问题的数据...

    数据结构--并查集(Union-Find Sets)

    此外,它也是算法竞赛中的常见题目类型,例如POJ等在线编程平台上的题目就经常涉及并查集的使用。 综上所述,并查集是一种非常实用的数据结构,它在处理集合的连接与查找问题时具有高效性和灵活性,对于理解和掌握...

    并查集(Union Find set)基础

    并查集基础 acm 算法 poj oi 并查集基础.ppt

    1089_bingchaji.rar_1089_bingchaji.rar _并查集

    在给定的标题“1089_bingchaji.rar_1089_bingchaji.rar _并查集”和描述“POJ1089 并查集可以解决 并查集加路径压缩”中,我们可以看到这是一个关于使用并查集解决特定问题的案例,可能来自于编程竞赛或练习。...

    并查集问题

    1. poj2524.cpp:这是一个POJ(Problem Setter's Online Judge)的编程题目,通常涉及到特定的算法问题,可能需要利用并查集解决连通性或路径查找问题。 2. hdoj1233最小生成树,克鲁斯卡尔.cpp:最小生成树是图论...

    并查集C++实现

    这份代码用C++实现了经典算法并查集,来源于poj题目1182

    并查集总结

    以POJ 1988为例,这是一个经典的并查集应用题,题目要求计算每个木箱上面还有多少木箱。通过并查集的查找和合并操作,结合路径压缩,可以有效地解决这一问题。 ```cpp #include #include void unionu(int x, int...

    poj 1611 The Suspects 代码

    poj 1611 The Suspects 代码 并查集的应用

    POJ1011-Sticks

    一种常见的方法是使用深度优先搜索(DFS)或广度优先搜索(BFS)来构建线段之间的连接,并通过并查集(Disjoint Set)数据结构来判断是否存在不相交的集合。 1. **数据结构**:首先,我们需要用数组或者链表存储...

    POJ算法题目分类

    * 简单并查集:简单并查集是指解决问题的简单并查集算法。 * 哈希表和二分查找等高效查找法:哈希表和二分查找等高效查找法是指解决问题的高效查找算法,如 poj3349、poj3274、POJ2151、poj1840、poj2002、poj2503。...

    详解并查集(0基础可飞速掌握其基本原理).pdf

    ### 并查集详解 #### 一、并查集简介 并查集是一种非常重要的数据结构,主要用于处理一些不相交集合的合并及查询问题。它具有高效且简洁的特点,在算法竞赛以及实际应用中有着广泛的应用场景。比如在网络连接、...

    POJ 1679 练习克鲁斯卡尔kruskal 算法

    在Java环境下,你可以使用ArrayList来存储边以及它们的权重,用HashSet或者TreeSet对边进行排序,同时利用并查集(Disjoint Set)进行环路检测。 【文件名称列表】: Main.java 这表明你需要在Main.java文件中编写...

    poj题目分类

    * 并查集的高级应用:例如 poj1703、poj2492。 * KMP 算法:例如 poj1961、poj2406。 4. 搜索: * 最优化剪枝和可行性剪枝。 * 搜索的技巧和优化:例如 poj3411、poj1724。 * 记忆化搜索:例如 poj3373、poj...

    poj训练计划.doc

    - 并查集的高级应用:如`poj1703, poj2492`。 - **搜索** - 最优化剪枝和可行性剪枝:如`poj3411, poj1724`。 - **动态规划** - 复杂的动态规划:如`poj1191, poj1054`。 - 记录状态的动态规划:如`poj3254, ...

    POJ 1861 Network

    【标题】"POJ 1861 Network"是一个经典的计算机科学问题,它涉及到图论中的网络连接,并要求我们利用并查集(Disjoint Set)数据结构来检测图中的环路,同时应用快速排序算法来求解最小生成树。这个问题在ACM(国际...

    poj各种分类

    Prim算法和Kruskal算法分别基于贪心策略和并查集数据结构,用于在带权图中找到连接所有顶点的最小总权重的树结构。 #### 拓扑排序 适用于有向无环图,帮助分析任务依赖关系,如poj1094所示。 #### 匹配算法 包括...

    西工大新版POJ100题合集

    《西工大新版POJ100题合集》是一个针对西北工业大学计算机科学与技术专业学生的编程练习资源,包含了100个不同难度级别的题目源代码。这些题目源自POJ(Problem Online Judge)在线编程评测系统,是学习C语言编程和...

    POJ分类POJ分类POJ分类POJ分类POJ分类POJ分类POJ分类

    - **解释**:并查集是一种用于处理集合合并及查询操作的数据结构。 #### 2. 特殊优化技巧 - **例题**:poj3393, poj1472, poj3371, poj1027, poj2706 - **解释**:特殊优化技巧通常涉及特定问题的优化算法,如位...

Global site tag (gtag.js) - Google Analytics