zoukankan      html  css  js  c++  java
  • 【并查集】连接格点C++

    连接格点
    描述
    
    有一个M行N列的点阵,相邻两点可以相连。一条纵向的连线花费一个单位,一条横向的连线花费两个单位。某些点之间已经有连线了,试问至少还需要花费多少个单位才能使所有的点全部连通。
    
    
    输入
    第一行输入两个正整数m和n, 其中 1 <= n,m <= 1000。
    
    以下若干行每行四个正整数x1,y1,x2,y2,表示第x1行第y1列的点和第x2行第y2列的点已经有连线。输入保证|x1-x2|+|y1-y2|=1。
    
    
    输出
    输出使得连通所有点还需要的最小花费。
    
    
    输入样例 1 
    
    2 2
    1 1 2 1
    输出样例 1
    
    3
    

    这道题明显使用并查集来完成,但是需要把由(x,y)组成的坐标转成一个数字,所以就令点(x,y)为点(x-1)*m+y即可,其他的就是再普通不过的并查集来完成了。

    AC代码:

    #include<bits/stdc++.h>
    using namespace std;
    int n,m,a,b,c,d,ans;
    int fa[1000000+10];
    int convert(int x,int y)
    {
    	return x*m+y-m;
    }
    void init()
    {
    	for(int i=1;i<=n;i++)
    	{
    		for(int j=1;j<=m;j++)
    		{
    			int con=convert(i,j);
    			fa[con]=con;
    		}
    	}
    }
    int get(int x)
    {
    	if(fa[x]==x)return x;
    	int r=get(fa[x]);
    	fa[x]=r;
    	return r;
    }
    bool merge(int x,int y)
    {
    	int r1=get(x);
    	int r2=get(y);
    	if(r1==r2)return 0;
    	else fa[r1]=r2;
    	return 1;
    }
    int main()
    {
    	cin>>n>>m;
    	init();
    	while(cin>>a>>b>>c>>d)
    	{
    		bool meiyong=merge(convert(a,b),convert(c,d));
    	}
    	for(int j=1;j<=m;j++)
    	{
    		for(int i=1;i+1<=n;i++)
    		{
    			if(merge(convert(i,j),convert(i+1,j)))
    			{
    				ans++;
    			}
    		}
    	}
    	for(int j=1;j<=m;j++)
    	{
    		for(int i=1;i+1<=n;i++)
    		{
    			if(merge(convert(i,j),convert(i,j+1)))
    			{
    				ans+=2;
    			}
    		}
    	}
    	cout<<ans<<endl;
    	return 0;
    }
    

    ov.

    个人博客地址: www.moyujiang.com 或 moyujiang.top
  • 相关阅读:
    Spring 控制器层如何调用DAO层
    Spring 工程分层
    spring boot工程如何启用 热启动功能
    Spring 视图层如何显示验证消息提示
    Sping POJO中如何添加验证规则和验证消息提示
    Spirng 分层,增加数据访问对象层
    Spring A 标签链接使用
    Spring 控制器重定向
    课程详情页之后台
    课程详情页之前台
  • 原文地址:https://www.cnblogs.com/moyujiang/p/11167794.html
Copyright © 2011-2022 走看看