最小生成树,Kruskal
#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
int fa[105],gra[105][105];
int n,m;
const int maxn=100005;
struct edge
{
int u,v,w;
}e[maxn];
int find(int x)
{
return fa[x]==x?fa[x]:fa[x]=find(fa[x]);
}
void unionxy(int x,int y)
{
int x1=find(x);
int y1=find(y);
if(x1!=y1) fa[x1]=y1;
}
bool cmp(edge a,edge b)
{
return a.w<b.w;
}
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
//freopen("in.txt","r",stdin);
while(cin>>n>>m&&n)
{
for(int i=1;i<=m;i++) fa[i]=i;
for(int i=1;i<=n;i++)
cin>>e[i].u>>e[i].v>>e[i].w;
sort(e+1,e+n+1,cmp);
int cnt=0;
int sum=0;
for(int i=1;i<=n;i++)
{
if(find(e[i].u)!=find(e[i].v))
{
unionxy(e[i].u,e[i].v);
++cnt;
sum+=e[i].w;
}
if(cnt==m-1) break;
}
if(cnt!=m-1)
cout<<"?"<<endl;
else cout<<sum<<endl;
}
return 0;
}