Typesetting math: 100%

题解【CF277E Binary Tree on Plane】

Description

给你平面上 n 个点 (2n400),要求用这些点组成一个二叉树(每个节点的儿子节点不超过两个),定义每条边的权值为两个点之间的欧几里得距离。求一个权值和最小的二叉树,并输出这个权值。

其中,点 i 可以成为点 j 的的父亲的条件是:点 iy 坐标比 jy 坐标大。

如果不存在满足条件的二叉树,输出 1

Solution

(a,b) 表示一条容量为 a ,费用为 b 的边
把每个点 u 拆成两个点入点 u1 和出点 u2
从源点向 u1 连一条 (2,0),意义为限制了 u 只能有两个儿子
u2 向汇点连一条 (1,0) ,意义是限制了 u 最多只有一个父亲
uy>vy 则在 u1v2 之间连一条 (1,Len),其中 Len 是两点之间的距离 (uxvx)2+(uyvy)2
然后跑最小费用最大流完事。

Code

#include <bits/stdc++.h>
#define db double
using namespace std;
const int N = 450; 
const int INF = 1000000000;
int n, k, S, T, vis[N], cnt, f[N * 2], pre[N * 2];
db dis[N * 2];  
struct edge {
  int v, f; db w; edge *next, *rev; 
}pool[N * N], *head[N * 2], *r[N * 2];
struct node {
  int sid, tid;
  db x, y; 
}a[N]; 
inline db Len(db x1, db y1, db x2, db y2) {
  return sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2)); 
}
inline void addedge(int u, int v, int f, db w) {
  edge *p = &pool[++cnt], *q = &pool[++cnt];
  p->v = v, p->f = f, p->w = w, p->next = head[u], head[u] = p;  p->rev = q;  
  q->v = u, q->f = 0, q->w = -w, q->next = head[v], head[v] = q; q->rev = p; 
}
inline bool spfa() {
  for(int i = S; i <= T; i++) pre[i] = -1, dis[i] = INF, r[i] = NULL, vis[i] = 0; 
  queue <int> Q; Q.push(S), dis[S] = 0, vis[S] = 1; f[S] = INF; 
  while(!Q.empty()) {
    int u = Q.front(), v; Q.pop(); vis[u] = 0;
    for(edge *p = head[u]; p; p = p->next) {
      if(p->f && dis[v = p->v] > dis[u] + p->w) {
        dis[v] = dis[u] + p->w; 
        pre[v] = u, r[v] = p; 
        f[v] = min(f[u], p->f); 
        if(!vis[v]) vis[v] = 1, Q.push(v); 
      }
    }
  } return pre[T] != -1; 
} int MF; db MC; 
inline void MCMF() {
  while(spfa()) {
    for(int i = T; i != S; i = pre[i])
      r[i]->f -= f[T], r[i]->rev->f += f[T];
    MF += f[T], MC += 1.0 * dis[T] * f[T]; 
  } 
}
int main() {
  scanf("%d", &n); S = 0, T = 2 * n + 1; 
  for(int i = 1; i <= n; i++) {
    scanf("%lf %lf", &a[i].x, &a[i].y);
    a[i].sid = i * 2 - 1, a[i].tid = i * 2; 
    addedge(S, a[i].sid, 2, 0); 
    addedge(a[i].tid, T, 1, 0); 
  }
  for(int i = 1; i <= n; i++)
    for(int j = 1; j <= n; j++)
      if(i != j && a[i].y > a[j].y) 
        addedge(a[i].sid, a[j].tid, 1, Len(a[i].x, a[i].y, a[j].x, a[j].y)); 
  MCMF(); 
  if(MF == n - 1) printf("%lf\n", MC); 
  else printf("-1\n");
  return 0; 
}
posted @   AcFunction  阅读(714)  评论(0编辑  收藏  举报
编辑推荐:
· 用 .NET NativeAOT 构建完全 distroless 的静态链接应用
· 为什么构造函数需要尽可能的简单
· 探秘 MySQL 索引底层原理,解锁数据库优化的关键密码(下)
· 大模型 Token 究竟是啥:图解大模型Token
· 35岁程序员的中年求职记:四次碰壁后的深度反思
阅读排行:
· 用 .NET NativeAOT 构建完全 distroless 的静态链接应用
· 如何开发 MCP 服务?保姆级教程!
· 1.net core 工作流WorkFlow流程(介绍)
· C# 工业视觉开发必刷20道 Halcon 面试题
· 瞧瞧别人家的限流,那叫一个优雅!
点击右上角即可分享
微信分享提示