1114. Family Property (25)[并查集]

2023-12-02 15:58
文章标签 25 family 查集 property 1114

本文主要是介绍1114. Family Property (25)[并查集],希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 原题: https://www.patest.cn/contests/pat-a-practise/1114

2. 思路:

题意:
找出同一个有关系的家庭里的人数,房产套数及总面积。
思路:
并查集或者dfs算法。
集合问题,并查集优先。


我是先把结点进行合并,
合并好后按照根结点值升序(因为s[i].parent是负值,绝对值是该集合的人数),同时累计该集合的套数及面积。
最后排序输出就好了。
注意:
合并的时候要把编号大的合并到小的那里去。
已AC。

3. 源码:

#include<iostream>
#include<algorithm>//使用sort函数
#include<vector>
using namespace std;struct Node
{Node() : id(-1), parent(-1), sets(0), area(0) {}//父结点初始化为-1,表示自己就是根bool operator<(const Node &b) const//重载比较运算符,先安装父结点值升序,再按照id降序。所以id初始为-1{if (parent != b.parent)return parent < b.parent;elsereturn id > b.id;}int id, parent, sets, area;//分别为结点号, 父结点号, 该人的套数, 面积double avg_area, avg_sets;//平均面积及套数
};
const int Max = 10000;//结点数最大值
Node s[Max];//下标映射结点号int find(int x);//并查集的查找根结点函数
void Union(int root1, int root2);//并查集的合并
bool cmp(const Node &a, const Node &b);//最后输出的排序int main(void)
{//freopen("in.txt", "r", stdin);int N;scanf("%d", &N);vector<Node> data(N);//存储现有结点数for (int i = 0; i < N; i++){vector<int> tnode;//存储每个id下有关系的人int t_id, f, m, k;//分别为当前编号, 父亲编号,目前编号, 孩子数scanf("%d %d %d %d", &t_id, &f, &m, &k);tnode.push_back(f);//插入数组tnode.push_back(m);for (int j = 0; j < k; j++){int child;scanf("%d", &child);tnode.push_back(child);}data[i].id = t_id;s[t_id].id = t_id;scanf("%d %d", &data[i].sets, &data[i].area);for (int j = 0; j < tnode.size(); j++)//对数组统一处理{if (tnode[j] != -1){s[tnode[j]].id = tnode[j];int root1 = find(t_id);int root2 = find(tnode[j]);Union(root1, root2);//合并}}}for (int i = 0; i < N; i++)//累计套数和面积{s[find(data[i].id)].area += data[i].area;s[find(data[i].id)].sets += data[i].sets;}sort(s, s + Max);//排序int cnt = 0;//集合数目vector<Node> re;//存储最终结果for (int i = 0; i < Max; i++){if (s[i].parent < 0 && s[i].id > -1)//计算有效的根结点{s[i].avg_area = (double)s[i].area / (-s[i].parent);s[i].avg_sets = (double)s[i].sets / (-s[i].parent);re.push_back(s[i]);cnt++;}elsebreak;}sort(re.begin(), re.end(), cmp);//对结果排序printf("%d\n", cnt);for (int i = 0; i < cnt; i++){printf("%04d %d %.3lf %.3lf\n", re[i].id, -re[i].parent, re[i].avg_sets,re[i].avg_area);}return 0;
}int find(int x)//并查集的查找根结点函数
{if (s[x].parent < 0)return x;elsereturn (s[x].parent = find(s[x].parent));//利用了路径压缩
}void Union(int root1, int root2)//并查集的合并
{if (root1 < root2)//大的合并到小的{s[root1].parent += s[root2].parent;s[root2].parent = root1;}else if (root2 < root1){s[root2].parent += s[root1].parent;s[root1].parent = root2;}
}bool cmp(const Node &a, const Node &b)//最后输出的排序
{if (a.avg_area != b.avg_area)return a.avg_area > b.avg_area;elsereturn a.id < b.id;
}


这篇关于1114. Family Property (25)[并查集]的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/445933

相关文章

poj 1182 并查集 食物链类

题意: 有n只动物,分别编号1....n。所有动物都属于A,B,C中的一种,已知A吃B,B吃C,C吃A。 按顺序给出下面两种共K条信息: 1. x 和 y 属于同一类。 2. x 吃 y 。 然而这些信息可能会出错,有可能有的信息和之前给出的信息矛盾,也有的信息可能给出的 x 和 y 不在n的范围内。 求k条信息中有多少条是不正确的。 解析: 对于每只动物,创建3个元素 i

POJ1988带权并查集

M u v 将u所在的堆移动到v所在的堆的上面 C u 求u的下面有多少块 带权并查集 import java.io.BufferedReader;import java.io.InputStream;import java.io.InputStreamReader;import java.io.PrintWriter;import java.math.BigInteger;i

POJ1703带权并查集

D: u v  u与v在不同的集合 A: u v  查询u与v的关系 1)压缩路径过程        fu->root   0  1  u-fu 0                 0  1   1                 1  0 2)合并过程 fu->fv      u->fu   0 1   v->fv 0            1 0 1            0

【JavaScript】LeetCode:21-25

文章目录 21 最大子数组和22 合并区间23 轮转数组24 除自身以外数组的乘积25 缺失的第一个正数 21 最大子数组和 贪心 / 动态规划贪心:连续和(count)< 0时,放弃当前起点的连续和,将下一个数作为新起点,这里提供使用贪心算法解决本题的代码。动态规划:dp[i]:以nums[i]为结尾的最长连续子序列(子数组)和。 dp[i] = max(dp[i - 1]

vue 父组件调用子组件的方法报错,“TypeError: Cannot read property ‘subDialogRef‘ of undefined“

vue 父组件调用子组件的方法报错,“TypeError: Cannot read property ‘subDialogRef’ of undefined” 最近用vue做的一个界面,引入了一个子组件,在父组件中调用子组件的方法时,报错提示: [Vue warn]: Error in v-on handler: “TypeError: Cannot read property ‘methods

Cannot read property ‘length‘ of null while opening vscode terminal

同一问题地址:Cannot read property ‘length’ of null while opening vscode terminal 问题描述 One day, 我在ubuntu 18.04下用vscode打开一个项目,并想和往常一样在vscode使用终端,发现报错Cannot read property 'length' of null。 解决 打开setting.jso

并查集基础与简单扩展应用

并查集 基础题目路径压缩 扩展应用扩展题目1扩展题目2 并查集的结构是一棵树 并查集有两种功能,一种是判断两个元素是否在同一集合,第二种是合并两个集合 并查集的实现需要记录每个节点的父亲节点 判断两个元素是否在同一集合,即判断两个元素的祖宗节点是否是一个节点(祖宗代表整棵树的根节点) 合并两个集合,即将任意一个集合祖宗的爸爸改为另一个集合的祖宗 基础题目 一共有

nyoj99(并查集+欧拉路+dfs)

单词拼接 时间限制: 3000 ms  |  内存限制: 65535 KB 难度: 5 描述 给你一些单词,请你判断能否把它们首尾串起来串成一串。 前一个单词的结尾应该与下一个单词的道字母相同。 如 aloha dog arachnid gopher tiger rat   可以拼接成:aloha.arachnid.dog.gopher.rat.tiger 输入 第一行是一个整

nyoj42(并查集解决欧拉回路)

一笔画问题 时间限制: 3000 ms  |  内存限制: 65535 KB 难度: 4 描述 zyc从小就比较喜欢玩一些小游戏,其中就包括画一笔画,他想请你帮他写一个程序,判断一个图是否能够用一笔画下来。 规定,所有的边都只能画一次,不能重复画。   输入 第一行只有一个正整数N(N<=10)表示测试数据的组数。 每组测试数据的第一行有两个正整数P,Q(P<=1000,Q<

【HDU】5222 Exploration(并查集+拓扑排序)

对于无向边使用并查集合并成一个集合,对于有向边使用判断是否存在环 唯一无语的地方就是看输入是百万级的,加个输入挂跑9s,不加挂跑了5s #include<cstdio>#include<cstring>#include<vector>#include<algorithm>using namespace std;#pragma comment(linker, "/STACK:102