W11(Union)

7-1 部落

分数 25

全屏浏览题目

切换布局

作者 陈越

单位 浙江大学

在一个社区里,每个人都有自己的小圈子,还可能同时属于很多不同的朋友圈。我们认为朋友的朋友都算在一个部落里,于是要请你统计一下,在一个给定社区中,到底有多少个互不相交的部落?并且检查任意两个人是否属于同一个部落。

输入格式:

输入在第一行给出一个正整数N(≤104),是已知小圈子的个数。随后N行,每行按下列格式给出一个小圈子里的人:

K P[1] P[2] ⋯ P[K]

其中K是小圈子里的人数,P[i](i=1,⋯,K)是小圈子里每个人的编号。这里所有人的编号从1开始连续编号,最大编号不会超过104。

之后一行给出一个非负整数Q(≤104),是查询次数。随后Q行,每行给出一对被查询的人的编号。

输出格式:

首先在一行中输出这个社区的总人数、以及互不相交的部落的个数。随后对每一次查询,如果他们属于同一个部落,则在一行中输出Y,否则输出N

输入样例:

4
3 10 1 2
2 3 4
4 1 5 7 8
3 9 6 4
2
10 5
3 7

输出样例:

10 2
Y
N
#include <bits/stdc++.h>
#define N 10010
using namespace std;

int s[N];//初始化为0
int ps[N];
int num = 0;//总人数

int Find(int x)
{
	if (x == ps[x])
		return x;
	return ps[x] = Find(ps[x]);
}

void Unionroot(int x, int y)
{
	int a, b;
	a = Find(x);
	b = Find(y);
	if (a != b)
		ps[a] = b;
}

int main()
{
	int n, k, x, y, sum = 0;//sum记录根数,根的对应数组值为其编号本身,其余编号对应数组值为其父结点编号
	memset(s, 0, sizeof(s));
	for (int i = 1; i < N; i++)
		ps[i] = i;

	cin >> n;
	for (int i = 0; i < n; i++)
	{
		cin >> k;
		cin >> x;
		if (s[x] == 0)
		{
			s[x] = 1;
			num++;
		}
		for (int j = 1; j < k; j++)
		{
			cin >> y;
			if (s[y] == 0)
			{
				s[y] = 1;
				num++;
			}
			Unionroot(x, y);
		}
	}
	for (int i = 1; i < N; i++)
	{
		if (s[i] && i == Find(i))
			sum++;
	}
	cout << num << ' ' << sum << endl;
	int m;
	cin >> m;
	for (int i = 0; i < m; i++)
	{
		cin >> x >> y;
		if (Find(x) == Find(y))
			cout << "Y" << endl;
		else
			cout << "N" << endl;
	}
	return 0;
}

 

7-3 文件传输

分数 25

全屏浏览题目

切换布局

作者 陈越

单位 浙江大学

当两台计算机双向连通的时候,文件是可以在两台机器间传输的。给定一套计算机网络,请你判断任意两台指定的计算机之间能否传输文件?

输入格式:

首先在第一行给出网络中计算机的总数 N (2≤N≤104),于是我们假设这些计算机从 1 到 N 编号。随后每行输入按以下格式给出:

I c1 c2  

其中I表示在计算机c1c2之间加入连线,使它们连通;或者是

C c1 c2    

其中C表示查询计算机c1c2之间能否传输文件;又或者是

S

这里S表示输入终止。

输出格式:

对每个C开头的查询,如果c1c2之间可以传输文件,就在一行中输出"yes",否则输出"no"。当读到终止符时,在一行中输出"The network is connected."如果网络中所有计算机之间都能传输文件;或者输出"There are k components.",其中k是网络中连通集的个数。

输入样例 1:

5
C 3 2
I 3 2
C 1 5
I 4 5
I 2 4
C 3 5
S

输出样例 1:

no
no
yes
There are 2 components.
#include <bits/stdc++.h>
#define N 10010
using namespace std;

int s[N];//初始化为0
int ps[N];

int Find(int x)
{
	if (x == ps[x])
		return x;
	return ps[x] = Find(ps[x]);
}

void Unionroot(int x, int y)
{
	int a, b;
	a = Find(x);
	b = Find(y);
	if (a != b)
		ps[a] = b;
}

int main()
{
	int n, sum = 0;
	cin >> n;
	for (int i = 1; i < N; i++)
		ps[i] = i;
	memset(s, 0, sizeof(s));
	for (int i = 1; i <= n; i++)
		s[i] = 1;
	while (1)
	{
		char op;
		int c1, c2;
		cin >> op;
		if (op == 'S')
			break;
		cin >> c1 >> c2;
		if (op == 'I')
			Unionroot(c1, c2);
		else
		{
			if (Find(c1) == Find(c2))
				cout << "yes"; //如果在同一集合说明连接
			else
				cout << "no";
			cout << endl;
		}
	}
	for (int i = 1; i < N; i++)
	{
		if (s[i] && i == Find(i))
			sum++;
	}
	if (sum == 1)
		cout << "The network is connected.";
	else
		cout << "There are " << sum << " components.";

}

 

7-4 社交集群

分数 25

全屏浏览题目

切换布局

作者 陈越

单位 浙江大学

当你在社交网络平台注册时,一般总是被要求填写你的个人兴趣爱好,以便找到具有相同兴趣爱好的潜在的朋友。一个“社交集群”是指部分兴趣爱好相同的人的集合。你需要找出所有的社交集群。

输入格式:

输入在第一行给出一个正整数 N(≤1000),为社交网络平台注册的所有用户的人数。于是这些人从 1 到 N 编号。随后 N 行,每行按以下格式给出一个人的兴趣爱好列表:

Ki​: hi​[1] hi​[2] ... hi​[Ki​]

其中Ki​(>0)是兴趣爱好的个数,hi​[j]是第j个兴趣爱好的编号,为区间 [1, 1000] 内的整数。

输出格式:

首先在一行中输出不同的社交集群的个数。随后第二行按非增序输出每个集群中的人数。数字间以一个空格分隔,行末不得有多余空格。

输入样例:

8
3: 2 7 10
1: 4
2: 5 3
1: 4
1: 3
1: 4
4: 6 8 1 5
1: 4

输出样例:

3
4 3 1
#include<bits/stdc++.h>
using namespace std;
int p[1009], user[1009], hobby[1009]; 

int find(int x){  //找祖先 
	if(p[x] == x){
		return x;
	}
	return p[x] = find(p[x]);
}

void merge(int x, int y){  //合并 
	int xx = find(x), yy = find(y);
	if(xx != yy){
		p[yy] = xx;
	}
}

bool cmp(int x, int y){  //因为题目说非曾序,所以两个数相等时不用交换顺序 
	return x > y;
}

int main()
{
	for(int i=1; i<=1000; i++){
		p[i] = i;
	}
	int n,k,h,num=0;
	char ch;
	cin >> n;
	for(int i=1; i<=n; i++){
		cin >> k >> ch >> user[i]; //user[i]:用第一个兴趣来代表每一个人 
		for(int j=1; j<=k-1; j++){
			cin >> h;
			merge(user[i], h);
		}
	}
	for(int i=1; i<=n; i++){
		hobby[find(user[i])]++;  //计算每个圈子多少人 
	}
	for(int i=1; i<=1000; i++){
		if(hobby[i]!=0) num++;  //统计圈子个数 
	}
	sort(hobby, hobby+1001, cmp);
	cout << num << endl;
	for(int i=0; i<num; i++){
		if(i!=0) cout << " ";
		cout << hobby[i];
	}
	return 0;
}

 

7-2 秀恩爱分得快

分数 25

全屏浏览题目

切换布局

作者 陈越

单位 浙江大学

古人云:秀恩爱,分得快。

互联网上每天都有大量人发布大量照片,我们通过分析这些照片,可以分析人与人之间的亲密度。如果一张照片上出现了 K 个人,这些人两两间的亲密度就被定义为 1/K。任意两个人如果同时出现在若干张照片里,他们之间的亲密度就是所有这些同框照片对应的亲密度之和。下面给定一批照片,请你分析一对给定的情侣,看看他们分别有没有亲密度更高的异性朋友?

输入格式:

输入在第一行给出 2 个正整数:N(不超过1000,为总人数——简单起见,我们把所有人从 0 到 N-1 编号。为了区分性别,我们用编号前的负号表示女性)和 M(不超过1000,为照片总数)。随后 M 行,每行给出一张照片的信息,格式如下:

K P[1] ... P[K]

其中 K(≤ 500)是该照片中出现的人数,P[1] ~ P[K] 就是这些人的编号。最后一行给出一对异性情侣的编号 A 和 B。同行数字以空格分隔。题目保证每个人只有一个性别,并且不会在同一张照片里出现多次。

输出格式:

首先输出 A PA,其中 PA 是与 A 最亲密的异性。如果 PA 不唯一,则按他们编号的绝对值递增输出;然后类似地输出 B PB。但如果 A 和 B 正是彼此亲密度最高的一对,则只输出他们的编号,无论是否还有其他人并列。

输入样例 1:

10 4
4 -1 2 -3 4
4 2 -3 -5 -6
3 2 4 -5
3 -6 0 2
-3 2

输出样例 1:

-3 2
2 -5
2 -6

输入样例 2:

4 4
4 -1 2 -3 0
2 0 -3
2 2 -3
2 -1 2 
-3 2

输出样例 2:

-3 2
#include <bits/stdc++.h>
using namespace std;
const int N=1005;
int n,m,k,kk,xb[N],a[505],p,x,y;
double g[N][N];
string ct[N],t;
int toi(string s){//将字符串转化为正整数 
	int x=0;
	if(s[0]=='-')for(int i=1;i<s.size();++i)x*=10,x+=s[i]-'0';
	else for(int i=0;i<s.size();++i)x*=10,x+=s[i]-'0';
	return x;
}
int main(){
	cin>>n>>m;
	while(m--){
		cin>>k;kk=k;p=0;
		while(kk--){
			cin>>t;int tn=toi(t);
			if(t[0]=='-')xb[tn]=1;
			else xb[tn]=2;
			a[p++]=tn;
		}
		double k1=1.0/k;
		for(int i=0;i<k-1;++i){
			for(int j=i+1;j<k;++j){
				if(xb[a[i]]==xb[a[j]])continue;
				g[a[i]][a[j]]+=k1;g[a[j]][a[i]]=g[a[i]][a[j]];
			}
		}
	}
	cin>>t;x=toi(t);
	if(t[0]=='-')xb[x]=1;else xb[x]=2;
	cin>>t;y=toi(t);
	if(t[0]=='-')xb[y]=1;else xb[y]=2;
	for(int i=0;i<n;++i)if(xb[i]==1)ct[i]="-";else ct[i]="";//ct[i]存储编号i的人的性别符合,方便输出 
	double ma=0,mb=0;//x,y的最大亲密度 
	for(int i=0;i<n;++i)if(xb[i]!=xb[x])ma=max(ma,g[x][i]);
	for(int i=0;i<n;++i)if(xb[i]!=xb[y])mb=max(mb,g[y][i]);
	if(ma==mb&&g[x][y]==ma)cout<<ct[x]<<x<<" "<<ct[y]<<y<<endl;
	else {
		for(int i=0;i<n;++i)if(g[x][i]==ma&&xb[x]!=xb[i])cout<<ct[x]<<x<<" "<<ct[i]<<i<<endl;
		for(int i=0;i<n;++i)if(g[y][i]==mb&&xb[y]!=xb[i])cout<<ct[y]<<y<<" "<<ct[i]<<i<<endl;
	}	
	return 0;
} 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

102101222_张凯权

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

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

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

打赏作者

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

抵扣说明:

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

余额充值