IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 并查集,c/c++描述,完整代码 -> 正文阅读

[C++知识库]并查集,c/c++描述,完整代码

??并查集,是描述了这样一类问题:我们经常需要合并两个集合,或者经常查询某个元素属于哪个集合。频繁进行合并与查询操作,所以叫做并查集。
??问题如下:已知b与d、e与g、a与c、h与i、a与b、e与f、b与c,各自都是亲戚关系。请问:c与d、g与j、h与i是亲戚关系么?
??课本里介绍了并查集,和几个关键函数,但没有给出完整例题与完整代码。参考请教网上资料后,这里也给出了自己的代码,并没有用到递归。但分析思路是一样的。
??几个关键点的处理如下:
??如何存储数据:所有人的数据组成一维数组。数组里每一个人的数据用结构体表示:含有两个成员:int indexRelation,其亲戚在数组里的下表;char name,本人的名字。以区分不同的人。假设没有同名的人。
??如何判断两人是否为亲戚:先由函数createOrFind找到这两个人的在数组里的下表,如果其亲戚是同一个人,则这两人也为亲戚,否则这俩人就不是亲戚。
??如何建立存储所有人的数组:在输入亲戚关系的时候由函数createOrFind完成,如果输入的这个人已经存在于数组,直接返回其在数组里的下标,如果数组里没有包含这个人的信息,则把此人加入数组,再返回此人的数组下标。因为c语言只能建立定长数组。故还需要一个变量记录数组里实际存储人数,即数组实际长度。每输入一对亲戚后,紧接着合并其所在的家族。家族里可以只有一个人,即族长本身。
??如何查找族长:如果一个人的indexRelation成员等于其自身下标,这个人就是族长,其是族里所有人的亲戚。找到几个族长,就是有几个家族。由函数familyNum完成这个思路。
??如何合并家族:两个家族彼此都是亲戚的话,需要进行合并,并选出新的族长,即是别的文章里介绍的树的根。根据两个家族的成员多少,把人少的家族并到人数多的家族,即修改其indexRelation成员的值,存储大家族族长的下标。人少家族和人多家族的成员下标另由中间数组保存。并计算其数组长度,得到各自家族的人数。这个数组是动态变量,随函数unionSet的建立而存在,unionSet执行完毕则中间数组也被回收,不占用内存空间。
??别的地方提到了rank成员,表示家族成员在家族树里的深度。这里没有用到rank,因为已经做到了家族里族长是其他人的直接亲戚,看做家族树的话,树只有两层。族长单独一层,其他亲戚成员在另外同一层。
??各函数功能如下:
??createOrFind:返回每个人在存人数组里的下标,或者建立此人信息后再返回其下标。
??unionSet:合并两个人所在的家族或者称为集合。
??familyNum:统计存在多少家族,或者说统计数组里有几个集合。
??main函数所在源文件代码:

#include<iostream>
using namespace std;
#define MAXSIZE 1000

struct Relation{
	int indexRelation;
	char name;
};

extern int createOrFind(Relation relation[],int & arrayLength,char name);
extern void unionSet(Relation relation[], int& arrayLength,int indexA,int indexB);
extern int familyNum(Relation relation[], int& arrayLength);

int main() {
	Relation relation[MAXSIZE];
	int arrayLength = 0;
	
	int pairNum;
	cout << "input the number of relation pairs : ";
	cin >> pairNum;

	char peopleA, peopleB;
	int indexA, indexB;
	cout << endl;
	for (int i = 1; i <= pairNum; i++) {
		cout << "input the " << i << "th pair relation : ";
		cin >> peopleA >> peopleB;
		indexA = createOrFind(relation,arrayLength,peopleA);
		indexB = createOrFind(relation, arrayLength, peopleB);
		unionSet(relation,arrayLength,indexA,indexB);
	}
	
	cout <<endl<< "how many times you want to query ? ";
	cin >> pairNum;
	for (int i = 1; i <= pairNum; i++) {
		cout << endl;
		cout << "input the " << i << "th pair you want query : ";
		cin >> peopleA >> peopleB;
		indexA = createOrFind(relation, arrayLength, peopleA);
		indexB = createOrFind(relation, arrayLength, peopleB);
		
		if (relation[indexA].indexRelation == relation[indexB].indexRelation)
			cout << "they are relation"<<endl;
		else
			cout << "they are not relation" << endl;
	}

	cout << endl;
	cout <<"there are "<< familyNum(relation, arrayLength)<<" families"<<endl;
	return 0;
}

各函数所在源文件代码:

#include<iostream>
using namespace std;
#define MAXSIZE 1000

struct Relation {
	int indexRelation;
	char name;
};

int createOrFind(Relation relation[], int& arrayLength, char name) {
	for (int i = 0; i < arrayLength; i++)
		if (relation[i].name == name)
			return i;

	relation[arrayLength].indexRelation = arrayLength;
	relation[arrayLength].name = name;
	arrayLength++;
	return arrayLength - 1;
}
void unionSet(Relation relation[], int& arrayLength, int indexA, int indexB) {
	if (relation[indexA].indexRelation == relation[indexB].indexRelation)
		return;

	int indexParentA = relation[indexA].indexRelation;
	int indexParentB = relation[indexB].indexRelation;
	int familyA[MAXSIZE], lengthAFamily = 0, familyB[MAXSIZE], lengthBFamily = 0;
	for (int i = 0; i < arrayLength; i++) {
		if (relation[i].indexRelation == indexParentA) {
			familyA[lengthAFamily] = i;
			lengthAFamily++;
		}
		if (relation[i].indexRelation == indexParentB) {
			familyB[lengthBFamily] = i;
			lengthBFamily++;
		}
	}
	if (lengthAFamily <= lengthBFamily)
		for (int i = 0; i < lengthAFamily; i++)
			relation[familyA[i]].indexRelation = indexParentB;
	else
		for (int i = 0; i < lengthBFamily; i++)
			relation[familyB[i]].indexRelation = indexParentA;
}
int familyNum(Relation relation[], int& arrayLength) {
	int num = 0;
	for (int i = 0; i < arrayLength; i++)
		if (relation[i].indexRelation == i)
			num++;

	return num;
}

测试结果及对应的亲戚关系如下:
在这里插入图片描述
在这里插入图片描述
我认为能解决问题就可以。跟很多地方提供的代码不一样没有merge,union、find函数,也没有递归。
谢谢阅读。

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2021-08-02 10:37:50  更:2021-08-02 10:40:18 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年10日历 -2025/10/27 3:58:32-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码