首页 > 其他分享 >最小生成树

最小生成树

时间:2024-03-07 20:58:19浏览次数:31  
标签:dist int rint 最小 生成 edge mp

最小生成树

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

相关文章

  • Swagger-自动生成接口文档
    Swagger官网:https://swagger.io/Swagger3是接口文档生成工具依赖pom.xml<dependency> <groupId>com.github.xiaoymin</groupId><artifactId>knife4j-openapi2-spring-boot-starter</artifactId><version>4.4.0</version></......
  • 类以撒的结合的房间生成算法
    类以撒的结合的房间生成算法原链接翻译版本目前本人只实现了房间大小都一样的情况,没有什么2x2,L型,2x1之类的房间放个效果图算法需要注意的房间选择对于当前位置是否有设置房间(是否被占位)当前房间位置是否越界(超出限定范围)当前位置的周围4方向的房间量是否大于1个(为什......
  • 动图演示步骤 Vmware安装Centos-7 最小安装/图形化界面及常见错误参考,基础配置推荐
    程序软件工具安装篇--【Linux】(Vmware/Centos-7)目录程序软件工具安装篇--【Linux】(Vmware/Centos-7)①:文件准备工作虚拟机工具安装文件系统镜像文件②:Vmware安装工作③:Centos安装工作④:Centos安装常见错误⑤:基础配置参考⑥:注意事项①:文件准备工作虚拟机工具安装......
  • oracle11g awr手动生成快照
    您可以手动生成一个快照,以收集Oracle数据库的AWR(AutomaticWorkloadRepository)数据。请按照以下步骤生成一个快照:登录到Oracle数据库实例所在的服务器。切换到具有适当权限的Oracle用户。打开SQL*Plus或其他OracleSQL客户端。运行以下命令来生成快照:EXECDBM......
  • leetcode120. 三角形最小路径和
    leetcode120.三角形最小路径和这道题的关键在于想到dp[i][j]=min(dp[i-1][j-1],dp[i-1][j])+triangle[i][j];太久没做过算法题了,连设一个dp数组都没意识到我的代码classSolution{public:intminimumTotal(vector<vector<int>>&triangle){intsize......
  • 使用BPF之前和之后生成直方图过程的对比
    以bitehist为例:使用BPF之前:1、在内核中:开启磁盘IO事件的插桩观测。2、在内核中,针对每个事件:向perf缓冲区写入一条记录。如果使用了跟踪点技术(推荐方式),记录中会包含关于磁盘IO的几个元数据字段。3、在用户空间:周期性地将所有事件的缓冲区内容复制到用户空间4。在用户空间:......
  • 最小瓶颈路
    Decribe:给定一个\(n\)个点\(m\)条边的无向连通图,编号为\(1\)到\(n\),没有自环,可能有重边,每一条边有一个正权值\(w\)。给出\(q\)个询问,每次给出两个不同的点\(u\)和\(v\),求一条从\(u\)到\(v\)的路径上边权的最大值最小是多少。\(n\leq10^3,m\leq10^5,......
  • 代码随想录算法训练营第二天| 977.有序数组的平方、 209.长度最小的子数组、 59.螺旋
    977.有序数组的平方https://leetcode.cn/problems/squares-of-a-sorted-array/description/publicstaticint[]sortedSquares(int[]nums){intleft=0;intright=nums.length-1;int[]result=newint[nums.length];intwrite=......
  • Python根据坐标半径生成测试点数据
    一、代码#-*-coding:UTF-8-*-importcsvimportrandomimportmathimportdatetimefromfakerimportFaker#定义语言faker_data=Faker(locale='zh_CN')#获取当前时间current_time=datetime.datetime.now()#格式化时间formatted_time=current_time.strft......
  • Autofac的Swashbuckle生成报错 Microsoft.AspNetCore.Mvc.ApiExplorer.EndpointMetada
    错误内容:AnexceptionwasthrownwhileactivatingSwashbuckle.AspNetCore.SwaggerGen.SwaggerGenerator->Microsoft.AspNetCore.Mvc.ApiExplorer.ApiDescriptionGroupCollectionProvider->λ:Microsoft.AspNetCore.Mvc.ApiExplorer.IApiDescriptionProvider[]->......