swustoj---1091

来源:互联网 发布:龙门侧铣头编程 编辑:程序博客网 时间:2024/04/30 08:54

普通并查集+维护二当家人(不知怎么说。) 


要求先会并查集模板;


#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;
}



0 0
原创粉丝点击