D - New Friends(AtCoder Beginner Contest 350)

题目链接:

D - New Friends (atcoder.jp)

题目大意:

题目解析:

 题目的大致意思: 假如A和B是朋友 B和C也是朋友 那么当A和C不是朋友的时候 可以通过B让A和C也成为朋友 问你增加了多少对的朋友关系

题目分析:

咱们可以从图论去考虑 当这一群是一个连通块 那么这一群点(人) 都是可以通过这个连通块去成为朋友的 那么假如这个连通块有N个人 那么就会有 N * (N - 1) / 2 条边(朋友关系)  那么全部的连通块减去之前的M条原有的朋友关系就是答案 注意开long 存取答案

那么问题来了 怎么去看他们是不是在一个连通块 并查集出手了

复习一下 并查集让点y到点x的连通下 那么就是p[find(y)] = find(x) 直接就过去了

代码:

import java.util.*;

// 我认为这个题是并查集的题
public class D {
	public static int[] p = null; // 表示的是父亲
	public static void main(String[] args) {
		var sc = new Scanner(System.in);
		int n = sc.nextInt();
		int m = sc.nextInt();
		p = new int[n + 10];
		for(int i = 1; i <= n; i ++ ) {
			p[i] = i; // 刚开始的祖宗 都是自己自己
		}
		long ret = 0;
		int mm = m;
		
		var mp = new HashMap<Integer, Long>();
		var se = new TreeSet<Integer>();
		while(m -- != 0 ) {
			int x = sc.nextInt();
			int y = sc.nextInt();
			// 这里理解为y加到x的节点下面
			p[find(y)] = find(x); // 该不会是这里出问题了吧
		}
		
		for(int i = 1; i <= n; i ++ ) {
			int x = p[find(i)];
			if(!mp.containsKey(x)) {
				mp.put(x, 0l);
			}
			long t = mp.get(x) + 1;
			mp.put(x, t);
			se.add(x); // 这里面存的是 都是祖宗
		}
		for(int i : se) {
			//System.out.print("i = " + i + "\n");
			//System.out.print("x = " + mp.get(i) + "\n");
			long tt = mp.get(i);
			long t2 = mp.get(i) - 1;
			
			ret += tt * t2 / 2;
			//System.out.print("ret = " + ret + "\n");
		}
		ret -= mm;
		System.out.print(ret);
	}
	public static int find(int x) {
		if(x != p[x])p[x] = find(p[x]);
		return p[x];
	}
}

运行的结果:

 因为long问题和并查集认祖宗的问题 出现了几次wa

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

FindYou.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值