`
SaraWon
  • 浏览: 43298 次
  • 性别: Icon_minigender_2
  • 来自: 北京
社区版块
存档分类
最新评论

POJ 1182 食物链 初识并查集

阅读更多
Description

动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形。A吃B, B吃C,C吃A。
现有N个动物,以1-N编号。每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种。
有人用两种说法对这N个动物所构成的食物链关系进行描述:
第一种说法是"1 X Y",表示X和Y是同类。
第二种说法是"2 X Y",表示X吃Y。
此人对N个动物,用上述两种说法,一句接一句地说出K句话,这K句话有的是真的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。
1) 当前的话与前面的某些真的话冲突,就是假话;
2) 当前的话中X或Y比N大,就是假话;
3) 当前的话表示X吃X,就是假话。
你的任务是根据给定的N(1 <= N <= 50,000)和K句话(0 <= K <= 100,000),输出假话的总数。

Input

第一行是两个整数N和K,以一个空格分隔。
以下K行每行是三个正整数 D,X,Y,两数之间用一个空格隔开,其中D表示说法的种类。
若D=1,则表示X和Y是同类。
若D=2,则表示X吃Y。

Output

只有一个整数,表示假话的数目。

Sample Input

100 7
1 101 1
2 1 2
2 2 3
2 3 3
1 1 3
2 3 1
1 5 5

Sample Output

3

刚开始写的代码总是超时,于是就在网上搜到了一个简洁的解法,就是利用并查集,以前完全没有接触过这个概念,感觉理解起来还是有些困难。

并查集是一种典型的树形数据结构,用于处理一些不相交集合的合并和查询问题。
它的主要操作有初始化、查找和合并。

食物链这道题的解题思想是把有关联的动物都放到一个集合中,因为有相同的根节点,且知道两者分别与根节点的关系,所以这两者的关系也就可以确定了,如果不在一个集合中,则需要将两个集合合并。

具体实现步骤如下:

Father[x]表示x的根节点 rank[x]表示x与根节点的关系
初始化:把每个动物都看做一个集合,且每个集合的根节点就是自己,即Father[x]=x
查找根节点:如果father[x]直接是x,则直接返回;否则,x的根节点是father[x]的根节点,递归求解,同时由于father[x]在变,需要修改rank[x]
合并集合:如果a和b不在同一个集合,需要合并,可以将a的根节点的父节点指向b的根节点

#include <stdio.h>
 
/* father[x]表示x的根节点 */
int father[50005];
 
/*
rank[x]表示father[x]与x的关系
rank[x] == 0 表示father[x]与x是同类
rank[x] == 1 表示x吃father[x]
rank[x] == 2 表示father[x]吃x
*/
int rank[50005];
 
/* 初始化集合 */
void Make_Set(int x)
{
	father[x] = x;
}
 
/* 查找x所在的集合 */
int Find_Set(int x)
{
	int t;
	if (father[x] == x) return x;
	t = father[x];
	father[x] = Find_Set(father[x]);
	/* 因为压缩时根节点改变,必须更新father[x]与x的关系 */
	rank[x] = (rank[t] + rank[x]) % 3;
	return father[x];
}
 
/* 合并a和b */
void Union_Set(int a, int b, int len)
{
	int ra = Find_Set(a);
	int rb = Find_Set(b);
	/* 将集合ra合并到集合rb上 */
	father[ra] = rb;
	/* 更新father[ra]与ra的关系 */
	rank[ra] = (rank[b] - rank[a] + 3 + len) % 3 ;
}
 
int main()
{
	int i, n, m;
	int d, x, y;
	int rx, ry;
	int sum = 0;
	scanf("%d%d", &n, &m);
	for (i = 1; i <= n; i++)
	{
		Make_Set(i);
	}
	while(m--)
	{
		scanf("%d%d%d", &d, &x, &y);
		if (x > n || y > n || (d == 2 && x == y))
		{
			sum++;
		}
		else
		{
			/* 求出x和y所在的集合rx和ry */
			rx = Find_Set(x);
			ry = Find_Set(y);
			/* 若在同一个集合则可确定x和y的关系 */
			if (rx == ry)
			{
				if((rank[x] - rank[y] + 3) % 3 != d - 1)
				{
					sum++;
				}
			}
			/* 无法确定关系时按照规则合并节点 */
			else
			{
				Union_Set(x, y, d - 1);
			}
		}
	}
	printf("%d\n", sum);
	return 0;	
}


分享到:
评论

相关推荐

    poj 食物链

    poj 食物链代码.带权并查集,关键是找到其中的一些关系式.

    poj2492并查集应用的扩展

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

    POJ 1988 简单并查集,

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

    并查集C++实现

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

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

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

    poj 1611 The Suspects 代码

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

    并查集问题

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

    1089_bingchaji.rar_1089_bingchaji.rar _并查集

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

    并查集总结

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

    并查集(Union Find set)基础

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

    POJ1011-Sticks

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

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

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

    poj训练计划.doc

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

    poj题目分类

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

    POJ算法题目分类

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

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

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

    poj各种分类

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

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

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

    POJ2513-Colored Sticks【TrieTree+MergeSet+EulerPath】

    【标题】"POJ2513-Colored Sticks【TrieTree+MergeSet+EulerPath】"涉及的是一道编程竞赛题目,主要考察参赛者的数据结构与算法应用能力,特别是Trie树(字典树)、并查集(MergeSet)以及欧拉路径(Euler Path)这...

Global site tag (gtag.js) - Google Analytics