最小生成树
AcWing.346 走廊泼水节
简要题意
给定一个 N 个节点的树,要求增加若干条边,把这棵树扩充为完全图,并满足图的唯一最小生成树仍然是这棵树。求增加的边的权值总和最小是多少,保证边权位非负整数。
题目分析
考虑 kruskal 的过程,是把权值从小到大排序,依次扫描每一个边。那么我们想让这棵新的树的最小生成树仍然不变且是唯一的,那么我们的边权应该设为 \(z+1\) ,\(z\) 是当前扫描边的边长。那么两个点之间要比原图多多少条边呢?为 \(|S_x|*|S_y|-1\),\(S_x\) 表示 \(x\) 所在的并查集。所以只需要在原来跑 kruskal 的过程上多维护一个 \(S\) 就可以了。
struct rec
{
int x, y, z;
friend bool operator < (rec a, rec b)
{
return a.z < b.z;
}
} edge[M];
int find(int x)
{
return fa[x] == x ? x : fa[x] = find(fa[x]);
}
int kruscal()
{
sort(edge + 1, edge + n);
int idx = 0; ans = 0;
for (rint i = 1; i <= n; i++) fa[i] = i, s[i] = 1;
for (rint i = 1; i < n; i++)
{
int x = find(edge[i].x);
int y = find(edge[i].y);
if (x == y) continue;
fa[x] = y;
idx++;
ans += (edge[i].z + 1) * (s[x] * s[y] - 1);
s[y] += s[x];
}
if (idx < n - 1) return inf;
return ans;
}
signed main()
{
int T;
cin >> T;
while (T--)
{
cin >> n;
for (rint i = 1; i < n; i++)
{
cin >> edge[i].x >> edge[i].y >> edge[i].z;
}
cout << kruscal() << endl;
}
return 0;
}
AcWing 347. 野餐规划
简要题意
给定一张 \(N\) 个点 \(M\) 条边的无向图,求出无向图的一棵最小生成树,满足 \(1\) 号节点的度数不超过给定的整数 \(S\)。\(N\) 不超过 \(30\).
题目分析
首先,去掉一号节点之后,无向图可能会分成若干个联通块。可以用深度优先遍历划分出图中的每个联通块。设联通块共有 \(T\) 个,若 \(T > S\),则本题无解。
对于每个联通块,在这个联通块内部求出它的最小生成树,然后从联通块中选出一个节点 \(p\) 与 \(1\) 号节点相连,其中无向边 \((1,p)\) 的权值尽量小。
此时,我们已经得到了原无向图的一棵生成树,\(1\) 号节点的度数为 \(T\)。我们还可以尝试改动 \(S-T\),让答案更优。
考虑无向图中从节点 \(1\) 出发的每条边 \((1,x)\) ,边权为 \(z\)。如果 \((1,x)\) 还不在当前的生成树中,那么继续找到当前生成树中从 \(x\) 到 \(1\) 的路径上权值最大的边 \((u,v)\),边权为 \(w\)。求出使得 \(w-z\) 最大的点 \(x_0\)。若 \(x_0\) 对应的 \(w_0-z_0 > 0\),则从树中删掉边 \((u_0,v_0)\),加入边 \((1,x_0)\),答案就会变小 \(w_0-z_0\)。
重复上一步 \(S - T\) 或者直到 \(w_0-z_0<=0\),就得到了题目所求的最小生成树。
int n, s;
int tot;
int g[N][N];
int fa[N];
int block[N], cntb;
map<pair<int, int>, bool> v;
map<string, int> mp;
struct rec
{
int x, y, z;
friend bool operator < (rec a, rec b)
{
return a.z < b.z;
}
} edge[N];
struct node
{
int a, b;
int dist;
} f[N];
int find(int x)
{
if (fa[x] != x) fa[x] = find(fa[x]);
return fa[x];
}
int kruskal()
{
sort(edge + 1, edge + 1 + n);
for (rint i = 1; i <= tot; i++) fa[i] = i;
int ans = 0;
for (rint i = 1; i <= n; i++)
{
int x = find(edge[i].x);
int y = find(edge[i].y);
if (x == 1 || y == 1 || x == y) continue;
fa[x] = y;
v[{edge[i].x, edge[i].y}] = 1;
v[{edge[i].y, edge[i].x}] = 1;
ans += edge[i].z;
}
return ans;
}
void dfs(int x)
{
for (rint y = 2; y <= tot; y++)
{
if (!g[x][y] || block[y]) continue;
block[y] = cntb;
dfs(y);
}
}
void dfs(int x, int father)
{
for (rint y = 2; y <= tot; y++)
{
if (y == father || !v[{x, y}]) continue;
if (~f[y].dist) continue;
if (f[x].dist > g[x][y]) f[y] = f[x];
else f[y] = {x, y, g[x][y]};
dfs(y, x);
}
}
signed main()
{
cin >> n;
mp["Park"] = tot = 1;
for (rint i = 1; i <= n; i++)
{
string a, b;
int c;
cin >> a >> b >> c;
if (!mp[a]) mp[a] = ++tot;
if (!mp[b]) mp[b] = ++tot;
g[mp[a]][mp[b]] = g[mp[b]][mp[a]] = c;
edge[i] = {mp[a], mp[b], c};
}
cin >> s;
for (rint i = 2; i <= tot; i++)
{
if (!block[i])
{
cntb++;
block[i] = cntb;
dfs(i);
}
}
int sum = kruskal();
for (rint i = 1; i <= cntb; i++)
{
int minn = inf, id = 0;
for (rint j = 2; j <= tot; j++)
{
if (block[j] == i)
{
if (g[1][j] && minn > g[1][j])
{
minn = g[1][j];
id = j;
}
}
}
sum += minn;
v[{1, id}] = 1;
v[{id, 1}] = 1;
}
int t = cntb;
while (t < s)
{
s--;
for (rint i = 0; i <= 25; i++) f[i] = {0, 0, -1};
dfs(1, 0);
int maxx = 0, id = 0;
for (rint j = 2; j <= tot; j++)
{
if (g[1][j] && maxx < f[j].dist - g[1][j])
{
maxx = f[j].dist - g[1][j];
id = j;
}
}
if (!maxx) break;
v[{f[id].a, f[id].b}] = v[{f[id].b, f[id].a}] = 0;
v[{1, id}] = v[{id, 1}] = 1;
sum -= maxx;
}
cout << "Total miles driven: " << sum << endl;
return 0;
}
AcWing348. 沙漠之王
简要题意
给这一张 \(N\) 个点 \(M\) 条边的无向图,图中每条边 \(e\) 都有一个收益 \(C_e\) 和一个成本 \(R_e\),求该图的一颗生成树 \(T\), 使树中各边的收益之和除以成本之和,即 \(∑_{e∈T}C_e/∑_{e∈T}R_e\) 最大。\((1<N,M<10000)\)
题目分析
\(x=w/l\) 即 \(w-l*x =0,f(x) = w-l*x\);将边权更改为 \(w-l*x\) 来求生成树
因为 \(f(x)\) 是个单调递减函数,随着 \(x\) 的增大而减少,对于任意一个生成树如果 \(f(x)>0\),则 \(l\) 需要增大 \(f(x)<0\) 否则 \(l\) 需要减小 若要满足 \(f(x)==0\) 恒成立
1.若要 \(x\) 取最大值,则不能存在任意一个生成树 \(f(x)>0\), 否则 \(x\) 还能继续增大,即任意生成树 \(f(x)<=0\) 若存在一个生成树 \(f(x)>0\),则那个生成树的比率一定大于当前 \(x\), \(w/l > x\) 即 \(w-l*x > 0\)
2.若要 \(x\) 取最小值,则不能存在任意一个生成树 \(f(x)<0\),否则 \(x\) 还能继续减小,即任意生成树 \(f(x)>=0\) 若存在一个生成树 \(f(x)<0\),则那个生成树的比率一定小于当前 \(x\), \(w/l < x\) 即 \(w-l*x < 0\)
若要满足 \(f(x)>0\) 恒成立,则最小生成树 \(>0\)
若要满足 \(f(x)<0\) 恒成立,则最大生成树 \(<0\)
此题目求解最小的 \(x\) 值,也就是检查是否所有的生成树 \(f(x)>=0\),即最小生成树 \(>=0\)
如果最小生成树大于 \(0\),所有的生成树都满足 \(f(x)>0\), 尝试增加 \(x\) 得到 \(f(x)=0\)
否则,有生成树不满足这个条件,那么 \(x\) 一定要减少来使所有 \(f(x)>=0\)
double calc(int a, int b)
{
return sqrt((x[a] - x[b]) * (x[a] - x[b]) + (y[a] - y[b]) * (y[a] - y[b]));
}
bool check(double mid)
{
fill(dist, dist + n + 1, dinf);
fill(v, v + n + 1, 0);
dist[1] = 0;
double ans = 0;
for (rint i = 1; i <= n; i++)
{
int x = 0;
for (rint j = 1; j <= n; j++)
{
if (!v[j] && (x == 0 || dist[j] < dist[x])) x = j;
}
v[x] = 1;
ans += dist[x];
for (rint y = 1; y <= n; y++)
{
if (!v[y]) dist[y] = min(dist[y], fabs(w[x] - w[y]) - mid * calc(x, y));
}
}
return ans >= 0;
}
signed main()
{
while (cin >> n && n)
{
for (rint i = 1; i <= n; i++)
{
cin >> x[i] >> y[i] >> w[i];
}
double l = 0, r = 10000000;
double ans;
while ((r - l) > eps)
{
double mid = (l + r) / 2;
if (check(mid)) ans = mid, l = mid;
else r = mid;
}
cout << fixed << setprecision(3) << ans << endl;
}
return 0;
}
AcWing.349. 黑暗城堡
题目大意
问你有多少棵最短路径树
题目分析
这里用的邻接矩阵 Dijkstra
对于已经是最短路的情况,对于任意两个点 \(x,y\) 有 \(dist[y] <= dist[x] + z\)
现在考虑 \(dist[y] <= dist[x] + z\),那么在最短路径生成树中一定不能有这一条边。如果有这一条边,那么 \(y\) 的路径就不是最小的。(因为是树,所以只能是这一个点来对 \(y\) 进行更新)
那么当 \(dist[y]==dist[x]+z\),最短路径生成树里面可以包含这一条边。
1.对于每一个点,都有到达 \(1\) 号点的距离。现在按照距离从小到大对点进行考虑。
2.对于考虑到的 \(i\) 个点,查找已经遍历过的集合,看有多少 \(x\) 满足 \(dist[i]==dist[x]+z\)。这是方案数。使用乘法原理。
int n, m;
int a[N][N];
int dist[N];
int ans = 1;
bool v[N];
pair<int, int> f[N];
signed main()
{
cin >> n >> m;
memset(a, 0x3f, sizeof a);
memset(dist, 0x3f, sizeof dist);
dist[1] = 0;
for (rint i = 1; i <= n; i++) a[i][i] = 0;
for (rint i = 1; i <= m; i++)
{
int x, y, z;
cin >> x >> y >> z;
a[x][y] = a[y][x] = min(a[x][y], z);
}
for (rint i = 1; i <= n; i++)
{
int x = 0;
for (rint j = 1; j <= n; j++)
if (!v[j] && (x == 0 || dist[x] > dist[j])) x = j;
v[x] = 1;
for (rint y = 1; y <= n; y++)
{
if (!v[y]) dist[y] = min(dist[y], dist[x] + a[x][y]);
}
}
for (rint i = 1; i <= n; i++) f[i] = {dist[i], i};
sort(f + 1, f + n + 1);
memset(v, 0, sizeof v);
v[1] = 1;
for (rint i = 2; i <= n; i++)
{
int y = f[i].second;
int cnt = 0;
for (rint x = 1; x <= n; x++)
{
if (v[x] && dist[x] + a[x][y] == dist[y]) cnt++;
}
ans = ans * cnt % mod;
v[y] = 1;
}
cout << ans << endl;
return 0;
}
AcWing.388. 四叶草魔杖
题目大意
给定一张无向图,结点和边均有权值。所有结点权值之和为 \(0\),点权可以沿边传递,传递点权的代价为边的权值。求让所有结点权值化为 \(0\) 的最小代价。
题目分析
容易想到本题与最小生成树有关。一种不难想出的思路是求出原图的最小生成树,将最小生成树上所有边的权值之和作为答案。
但经过思考,可以发现这样得到的不一定是最优解。首先,原图可能并不联通;其次,可以将原图划分为若干个点权之和均为 \(0\) 的子图,在这些子图中分别转移点权,最后将答案合并。这样得到的方案或许会更优。
此时我们发现划分方案不止一种,如何确定最终的方案成了需要解决的最大问题。
注意到本题中 \(N\) 范围较小,允许我们把所有点权和为 \(0\) 的子图(以下简称“合法子图”)的最小生成树全部求出。因此可以先枚举原图点集的所有子集,对于每个点权和为 \(0\) 的点集,用这些点和连接它们的边构造一张合法子图。我们能够轻易求出这些合法子图的最小生成树。但有些合法子图或许并不联通,为避免对之后的求解造成影响,需要把这些子图的最小生成树边权和设为 \(\infty\)。
接下来需要把这些子图中的若干个合并起来,得到全局最优解。与划分的情形相同,合并这些子图的方案也有多种。可以使用 \(DP\) 得到最优解。
具体地,考虑进行类似背包的 \(DP\),将每个合法子图视作可以放入背包的一个物品。设 \(A\)、\(B\) 为两个不同合法子图的点集,合法子图的最小生成树边权和为 \(S\),可以写出如下状态转移方程:
$f_{A \cup B}=min {f_{A\cup B}, f_{A}+S_{B} },A\cap B=\oslash $
最终 \(f_{2^n-1}\) 即为所求的答案。
int n, m;
int a[N], fa[N];
int s[M], f[M], p[M];
struct rec
{
int x, y, z;
friend bool operator < (rec a, rec b)
{
return a.z < b.z;
}
} edge[M];
int find(int x)
{
return fa[x] == x ? x : fa[x] = find(fa[x]);
}
int kruskal(int s)
{
int ans = 0;
for (rint i = 0; i < n; i++) if (s & (1 << i)) fa[i] = i;
for (rint i = 1; i <= m; i++)
{
if (!(s & (1 << (edge[i].x))) || !(s & (1 << (edge[i].y)))) continue;
int x = find(edge[i].x);
int y = find(edge[i].y);
if (x == y) continue;
fa[x] = y;
ans += edge[i].z;
}
int father = -1;
for (rint i = 0; i < n; i++)
{
if (s & (1 << i))
{
if (father == -1) father = find(i);
else if (find(i) != father) return inf;
}
}
return ans;
}
signed main()
{
cin >> n >> m;
for (rint i = 1; i <= n; i++) cin >> a[i];
for (rint i = 1; i <= m; i++) cin >> edge[i].x >> edge[i].y >> edge[i].z;
for (rint i = 1; i < (1 << n); i++)
for (rint j = 0; j < n; j++)
if (i & (1 << j))
s[i] += a[j + 1];
sort(edge + 1, edge + m + 1);
for (rint i = 1; i < (1 << n); i++)
{
if (!s[i]) p[i] = kruskal(i);
f[i] = inf;
}
f[0] = 0;
for (rint i = 1; i < (1 << n); i++)
{
if (s[i]) continue;
for (rint j = 0; j < (1 << n); j++)
{
if (!(i & j)) f[i | j] = min(f[i | j], f[j] + p[i]);
}
}
if (f[(1 << n) - 1] >= inf) puts("Impossible");
else cout << f[(1 << n) - 1] << endl;
return 0;
}
标签:dist,int,rint,最小,生成,edge,mp
From: https://www.cnblogs.com/spaceswalker/p/18059715