普通并查集+维护二当家人(不知怎么说。)
要求先会并查集模板;
#include <stdio.h>
struct node
{
int x;//并查集的数
int m;//这个人拥有的钱
int s;//集合里钱第二多的人
}date[100005];
void init(int n)
{
for(int i = 0;i <= n; i++)//@1 先把集合第二钱多的人初始化为自己,这样的话后面如果第二多钱的人和自己是同一个那么代表他一个
date[i].s = date[i].x = i;//把并查集下标初始化
}
int find1(int x)//并查集寻找钱最多的
{
if(date[x].x==x)
return x;
return date[x].x = find1(date[x].x);
}
int main()
{
int n,i;
while(scanf("%d",&n)!=EOF)
{
for(i = 1;i <= n; i++) scanf("%d",&date[i].m);
init(n);
scanf("%d",&n);
int t1,t2,t3;
while(n--)
{
scanf("%d %d",&t1,&t2);
if(t1==1)//合并俩朋友圈
{
scanf("%d",&t3);
int x = find1(t2);
int y = find1(t3);//找到各自的朋友圈中钱最多的土豪霸霸
if((date[x].m>date[y].m)||(date[x].m==date[y].m&&x<y))//如果x霸霸的钱多于y霸霸或者x和y的钱一样多但是x编号比y小(维护钱最多的且编号最下的)
{
//进了这个if那么y霸霸和y霸霸的人都要成为x霸霸的人;
date[y].x = x;
if(date[x].s==x) date[x].s = y;//如果x霸霸是一个人那么不用说。y霸霸就是二当家
else if((date[date[x].s].m<date[y].m)||(date[date[x].s].m==date[y].m&&y<date[x].s))//如果x霸霸不是一个人那么y和x霸霸的二把手就要来pk.
{
date[x].s = y;
}
}
else//同上,不过这个是x霸霸率众加入y霸霸部落;
{
int t = x;
x = y;
y = t;
date[y].x = x;
if(date[x].s==x) date[x].s = y;
else if((date[date[x].s].m<date[y].m)||(date[date[x].s].m==date[y].m&&y<date[x].s))
{
date[x].s = y;
}
}
}
else//借钱开始
{
if(date[t2].x==t2)//如果要找的编号等于自己那么他自己就是钱最多的那个
{
if(date[t2].s==t2) printf("NO ONE CAN HELP!\r\n");//如果只有他一个那么人借给他。。。(接第四个注释)
else printf("%d\r\n",date[t2].s);//输出第二多钱的
}
else printf("%d\r\n",find1(t2));//找t2所处的朋友圈钱最多的
}
}
}
return 0;
}