题目描述
有一个M行N列的点阵,相邻两点可以相连。一条纵向的连线花费一个单位,一条横向的连线花费两个单位。某些点之间已经有连线了,试问至少还需要花费多少个单位才能使所有的点全部连通。
输入格式
第一行输入两个正整数m和n。
以下若干行每行四个正整数x1,y1,x2,y2,表示第x1行第y1列的点和第x2行第y2列的点已经有连线。输入保证|x1−x2|+|y1−y2|=1。
输出格式
输出使得连通所有点还需要的最小花费。
样例
样例输入
复制2 2
1 1 2 1
样例输出
复制3
数据范围与提示
【数据规模】
30%数据:n*m≤1000
100%数据:m,n≤1000
_____________________________________________________________________________
写作不易,点个赞呗!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
_____________________________________________________________________________
#include<bits/stdc++.h>
#define PII pair<int,int>
using namespace std;
int n,m,ans,cnt;
vector<PII> mest[1000005];
bool f[1000005];
void min_tree(){
priority_queue<PII,vector<PII>,greater<PII> > Q;
memset(f,0,sizeof(f));
Q.push({0,1});
while(!Q.empty()){
int a=Q.top().first;
int b=Q.top().second;
Q.pop();
if(f[b])continue;
f[b]=true;
ans+=a;
for(auto i:mest[b]){
if(f[i.second])continue;
Q.push({i.first,i.second});
}
}
}
int main(){
cin>>n>>m;
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
if(j<m){
mest[j+(i-1)*m].push_back({2,j+(i-1)*m+1});
mest[j+(i-1)*m+1].push_back({2,j+(i-1)*m});
}
if(i<n){
mest[j+(i-1)*m].push_back({1,j+i*m});
mest[j+i*m].push_back({1,j+(i-1)*m});
}
}
}
int x,y,z,w;
while(cin>>x>>y>>z>>w){
mest[(x-1)*m+y].push_back({0,(z-1)*m+w});
mest[(z-1)*m+w].push_back({0,(x-1)*m+y});
}
min_tree();
cout<<ans;
}