AOJ 2095 Nagashi Soumen

問題

空間上にn個の点があり、それぞれの座標は(x[i], y[i], z[i])である。
点をk本以下のパイプでつなぐ。


パイプは、

  • 自由にまげて良い。
  • 分岐したり、合流したりしてはいけない。
  • どこを始点、終点としてもよい。
  • 始点または終点または、途中の任意の点で、空間上の点を通ることができる。
  • z座標が真に大きい点から小さい点にしかつなげない。
  • 全ての点が、どれか1本以上のパイプにつながるようにする。

このとき、パイプの全長の最小値はいくつか、求めよ。

制約条件

n≦100
0≦x[i],y[i],z[i]≦100

方針

次のような費用流で解いた。


各点をIN, OUTの二頂点に分割する。
INからOUTに容量1、コスト0の辺を張る。


パイプでつなげる頂点間をa, bとする。
aのOUTからbのINへ、容量1、コストdist(a, b)の辺を張る。


始点から全てのパイプのINへ容量1、コスト0の辺、
終点から全てのパイプのOUTへ容量1、コスト0の辺を張る。


ここに、更に流量制約INからOUTの最小流量1という条件を付け加えたときの、
最小費用最大流が求めるパイプの全長の最小値。


費用流に最小流量の制約があるときは、蟻本にあるように、
適当に下駄をはかせて、必ずその辺が使われるように処理する。


想定解はDPっぽい。

ソースコード

typedef pair<long double, int> P;
struct edge{
	int to, cap, rev;
	long double cost;
	 
	edge(int to, int cap, long double cost, int rev) :
		to(to), cap(cap), cost(cost), rev(rev){
	}
};
 
const int MAX_V = 210;
int V, prevv[MAX_V], preve[MAX_V];
vector<edge> G[MAX_V];
long double h[MAX_V], dist[MAX_V];

void add_edge(int from, int to, int cap, long double cost){
	G[from].pb(edge(to, cap, cost, G[to].size()));
	G[to].pb(edge(from, 0, -cost, G[from].size() - 1));
}
 
long double min_cost_flow(int s, int t, int f){
	long double res = 0;
	while(f > 0){
		fill(dist, dist + V, INF);
		dist[s] = 0;
		int d = f;
		 
		rep(it, V){
			bool update = 0;
			rep(v, V){
				if(dist[v] == INF) continue;
				rep(i, G[v].size()){
					edge &e = G[v][i];
					if(e.cap > 0 && dist[e.to] > dist[v] + e.cost + EPS){
						dist[e.to] = dist[v] + e.cost;
						prevv[e.to] = v;
						preve[e.to] = i;
						update = 1;
					}
				}
			}
			if(!update) break;
		}
		if(dist[t] == INF) return -INF;
		 
		for(int v = t; v != s; v = prevv[v]){
			d = min(d, G[prevv[v]][preve[v]].cap);
		}
		f -= d;
		res += d * dist[t];
		for(int v = t; v != s; v = prevv[v]){
			edge &e = G[prevv[v]][preve[v]];
			e.cap -= d;
			G[v][e.rev].cap += d;
		}
		NEXT:;
	}
	return res;
}
 
inline long double sq(long double x){
	return x * x;
}
 
const double BIG = 1e6;
int n, k, x[100], y[100], z[100];
 
int main(){
	while(cin >> n >> k, n){
		V = 2 * n + 2;
		rep(i, V) G[i].clear();
		rep(i, n) cin >> x[i] >> y[i] >> z[i];
		rep(i, n){
			add_edge(V - 2, i * 2, 1, 0);
			add_edge(i * 2, i * 2 + 1, 1, -BIG);
			add_edge(i * 2 + 1, V - 1, 1, 0);
		}
		rep(i, n) rep(j, n) if(z[i] > z[j]){
			long double dist = sqrt(sq(x[i] - x[j]) + sq(y[i] - y[j]) + sq(z[i] - z[j]));
			add_edge(i * 2 + 1, j * 2, 1, dist);
		}
		k = min(k, n);
		long double ans = min_cost_flow(V - 2, V - 1, k);
		if(ans == -INF){
			cout << -1 << endl;
		}
		else{
			ans += BIG * n;
			if(ans > BIG - EPS || ans < -EPS) cout << -1 << endl;
			else cout << setiosflags(ios::fixed) << setprecision(20) << ans << endl;
		}
		 
	}
	return 0;
}