《算法竞赛·快冲300题》每日一题:“简化农场”

news2025/1/10 22:11:57

算法竞赛·快冲300题》将于2024年出版,是《算法竞赛》的辅助练习册。
所有题目放在自建的OJ New Online Judge。
用C/C++、Java、Python三种语言给出代码,以中低档题为主,适合入门、进阶。

文章目录

  • 题目描述
  • 题解
  • C++代码
  • Java代码
  • Python代码

简化农场” ,链接: http://oj.ecustacm.cn/problem.php?id=1879

题目描述

【题目描述】 约翰的农场可以看做一个图,农田代表图中顶点,田间小路代表图中的边,每条边有一定的长度。
   约翰发现自己的农场中最多有三条小路有着相同的长度。
   约翰想删除一些小路使得农场成为一棵树,使得两块农田间只有一条路径。
   约翰想把农场设计成最小生成树,也就是农场道路的总长度最短。
   请帮助约翰找出最小生成树的总长度,同时请计算出总共有多少种最小生成树。
【输入格式】 输入第一行为两个正整数 N 和 M ,表示点数和边数(1 <= N <= 40,000; 1 <= M <= 100,000)。
   接下来 M 行,每行三个整数 ai, bi, ci,表示点 ai 和 bi 之间存在长度为 ci 的无向边。(1 <= ai, bi <= N; 1 <= ci <= 1,000,000)
【输出格式】 输出一行包含两个整数,分别表示最小生成树的长度和最小生成树的数目。
数目对 1,000,000,007 求余.
【输入样例】

4 5
1 2 1
3 4 1
1 3 2
1 4 2
2 3 2

【输出样例】

4 3

题解

   有两种最小生成树(MST)算法:kruskal、prim。Kruskal的思路是对边贪心,“最短的边一定在MST上”;prim的思路是对点贪心,“最近的邻居点一定在MST上”。
   本题描述中比较特殊的地方是:(1)最多有三条小路(边)有相同长度;(2)计算总共有多少种最小生成树。着重点在边上,所以用kruskal算法。
   kruskal算法执行步骤是:(1)对边排序;(2)从最短的边开始,从小到大依次把边加入到MST中;(3)加边的过程中用并查集判断是否产生圈,如果形成了圈就丢弃这个边;(4)所有边处理完后结束,或者加边的数量等于n-1时结束。
   如果所有的边长都不同,那么只有一种最小生成树。题目指出“最多有三条边的长度相同”,从样例可知,有等长的两条边,也有等长的三条边。对边排序时,这些相等的边会挨在一起。
   处理等长边,设cnt是合法(所谓合法,是指这个边加入到MST,不会产生圈)的边的数量,num是这几个等长边有几个能同时加入到MST。sum是最小生成树的数目。
   (1)有两条等长边。
   若cnt=1,只有一条边是合法的,也就是说这条边别无选择,那么sum不变。
   若cnt=2,有2条边合法,继续讨论:
   1)num=1,即这两条等长边只有一条能加入到MST中。那么sum = sumcnt,即sum=sum2。以下图为例,s1和s2是两棵已经加入到MST的子树,它们内部没有圈。现在加两条等长边(x1,y1)、(x2,y2),它们单独加入MST都是合法的,但是同时加入就会形成圈。
在这里插入图片描述
   2)num=2,即这两条等长边都应该加入到MST中。那么sum不变,即sum=sum1。以下图为例。
在这里插入图片描述
   (2)有三个等长边。
   若cnt=1,只有一条边合法,sum不变。
   若cnt=2,有两条边合法,和(1)有两条等长边,且cnt=2的情况一样。
   若cnt=3,有三条边合法,那么:
   1)num = 1,只有一条边能加入到MST中,sum = sum
cnt=sum3。以下图为例,三条边任选一条,有3种情况。
在这里插入图片描述
   2)num = 2,有两条边能加入到MST中,且其中一条边必须加,sum = sum
2。以下图为例,三条边任选两条,有2种情况。
在这里插入图片描述
   3)num = 2,有两条边能加入到MST中,且是任意两条,sum = sum*3。以下图为例,三条边任选两条,有3种情况。
在这里插入图片描述
   3)num = 3,三条边都应该加入到MST中,sum不变。

【重点】 kruskal 。

C++代码

#include<bits/stdc++.h>
using namespace std;
#define int long long
const int N = 1e6+10;
const int mod = 1e9+7;
struct Node{int x,y,val;}e[N<<1];
bool cmp(Node a,Node b){return a.val<b.val;}//按边权排序
int n,m;
int s[N];          //并查集
int ans=0,sum=1;   //ans: MST的总长度, sum:最小生成树的数目
int find_set(int x){                         //查询并查集,返回x的根
    if(x != s[x]) s[x] = find_set(s[x]);     //路径压缩
    return s[x];
}
void kruscal(){
	for(int i=1;i<=n;i++) s[i] = i;  //并查集初始化
	sort(e+1,e+m+1,cmp);             //边:升序排序
	for(int i=1;i<=m;){              //遍历所有边,每次处理其中的等长边
		int cnt = 0;                 //这次的等长边中,有几个可以加入MST
		set<pair<int,int> >st;       //set用于存储并去重
		int j;                       //第i~j个边等长
		for(j=i;j<=i+2 && e[i].val==e[j].val;j++){ //枚举等长边,最多3个相同。更新j
			int s1 = find_set(e[j].x);   //边的一个端点属于哪个集?
			int s2 = find_set(e[j].y);   //边的另一个端点属于哪个集?
			if(s1 > s2)  swap(s1,s2);
			if(s1 != s2){                //两个集不等,这个边可以加入到MST中
				cnt ++;                  //cnt: 允许加入MST的边的数量
				st.insert(make_pair(s1,s2));   //这个边的两端点所属的集存到st中
			}
		}                                //第i~j个边是等长的
		int num = 0;
		for(;i<j;i++){                   //开始时第i~j个边是等长的。i=j时退出
			int s1 = find_set(e[i].x);
			int s2 = find_set(e[i].y);
			if(s1 != s2){                //不属于一个集,可以加入到树里
				s[s2] = s1;              //并查集合并
				num++;                   //这几个等长边有num个可以同时加入树
			}
		}
		ans += e[i-1].val*num;    //这几个等长边最后有num个可以加入到MST,计算MST总长
		if(num == 1)  sum = sum*cnt%mod;   //只有一条边能加入树,直接乘以cnt
		if(cnt == 3 && num==2 && st.size() == 2) sum = 2*sum%mod;
		if(cnt == 3 && num==2 && st.size() == 3) sum = 3*sum%mod;
		 //其他情况方案数都不变
	}
}
signed main(){
	scanf("%lld%lld",&n,&m);  //读点,边
	for(int i=1;i<=m;i++)     //存边,用最简单的“边集数组”存边
		scanf("%lld%lld%lld",&e[i].x,&e[i].y,&e[i].val);
	kruscal();                //最小生成树
	printf("%lld %lld\n",ans,sum);
}

Java代码

 

Python代码

  

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/962067.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot自定义消息总线

一、前言 在现代的分布式系统中&#xff0c;消息传递已成为一个非常流行的模式。它使得系统内的不同部分可以松耦合地通信&#xff0c;从而实现更高效、更可靠的应用程序。本博客将介绍SpringBoot如何提供简单易用的消息传递机制&#xff0c;并展示如何自定义消息总线以满足特定…

安装使用 d3graph 时出现 TypeError 的解决方法

使用 python 3.7 pip 22.3.1 在清华镜像源 https://pypi.tuna.tsinghua.edu.cn/simple 安装 d3blocks 1.3.2 时&#xff0c;安装成功后导入包时出错&#xff1a; 观察报错信息可以看到出错的代码&#xff08;902 行&#xff09;使用了类型指定语法&#xff0c;这是最新的 pyth…

stable diffusion实践操作-电脑硬件查看

本文专门开一节写电脑硬件相关的内容&#xff0c;在看之前&#xff0c;可以同步关注&#xff1a; stable diffusion实践操作 正文 1、检查电脑显存的方法&#xff08;win10&#xff09;&#xff1a; 鼠标放在工具栏&#xff0c;单击右键打开“任务管理器”&#xff0c;选择顶…

jmeter 固定定时器

固定定时器&#xff08;Constant Timer&#xff09;是一个定时器元件&#xff0c;可以在线程组中的每个线程之间添加固定的延迟时间。固定定时器会对每个线程的执行进行一定的暂停。 聊一下和线程组中的调度器对线程组执行时长的影响&#xff1a; 相同&#xff1a; 都会影响线…

【线性代数】矩阵求导的本质与分子布局、分母布局的本质(矩阵求导——本质篇)

矩阵求导的本质与分子布局、分母布局的本质&#xff08;矩阵求导——本质篇&#xff09; 说在前面一. 函数与标量、向量、矩阵二. 矩阵求导的本质三. 矩阵求导结果的布局四. 分子布局、分母布局的本质五. 向量变元的实值标量函数 说在前面 我将严谨地说明矩阵求导的本质与分子布…

游戏思考30(补充版):关于逆水寒铁牢关副本、白石副本和技能的一些注释(2023/0902)

前期介绍 我是一名逆水寒的玩家&#xff0c;做一些游戏的笔记当作攻略记录下来&#xff0c;荣光不朽-帝霸来源视频连接 传送门 一、旧版铁牢关&#xff08;非逆水寒老兵服&#xff09; &#xff08;1&#xff09;老一&#xff1a;巨鹰 1&#xff09;机制一&#xff1a;三阵风…

多通道振弦数据记录仪应用桥梁安全监测的关键要点

多通道振弦数据记录仪应用桥梁安全监测的关键要点 随着近年来桥梁建设和维护的不断推进&#xff0c;桥梁安全监测越来越成为公共关注的焦点。多通道振弦数据记录仪因其高效、准确的数据采集和处理能力&#xff0c;已经成为桥梁安全监测中不可或缺的设备。本文将从以下几个方面…

JavaScript基础02

JavaScript 基础 文章目录 JavaScript 基础运算符算术运算符赋值运算符自增/自减运算符比较运算符逻辑运算符运算符优先级 语句表达式和语句分支语句if 分支语句if双分支语句if 多分支语句三元运算符&#xff08;三元表达式&#xff09;switch语句&#xff08;了解&#xff09;…

孙哥Spring源码第16集

第16集 refresh()-prepareBeanFactory分析 【视频来源于&#xff1a;B站up主孙帅suns Spring源码视频】 1、设置类加载器 2、设置SpringEL表达式 解析器 3、设置内置的属性编辑器 &#xff08;类型转换器&#xff09; 3.1、如何实现类型转化&#xff1f; 1、Converter 2、P…

Ubuntu 20.04 Server配置网络

0&#xff0c;环境 服务器&#xff1a; Intel(R) Xeon(R) Gold 6248R CPU 3.00GHz 96核 网卡&#xff1a; 多网卡 1&#xff0c; 镜像下载 http://old-releases.ubuntu.com/releases/ubuntu-20.04.1-desktop-amd64.iso 2&#xff0c; 系统安装--具体步骤就不贴出来&#…

Navicat 强大的数据模型功能 | 面向数据库设计、架构和数据资产梳理等使用场景

数据模型是用来描述数据、组织数据和对数据进行操作的一组概念和定义。根据不同的应用需求&#xff0c;数据模型可以分为概念模型、逻辑模型和物理模型。这些数据模型帮助数据库设计人员设计和管理数据库&#xff0c;以满足用户的需求。 Navicat 强大的数据模型功能主要适用于…

Linux c++开发-03-使用CMake组织工程

一、简单文件的编译 有如下的目录结构&#xff1a; 其中 helloworld.cpp如下&#xff1a; #include <iostream> using namespace std; int main() {printf("hello world my name is Ty!");return 0; }CMakeLists.txt如下&#xff1a; cmake_minimum_requir…

cocos creator配置终端调试

在launch.json里添加"preLaunchTask":“CocosCreator compile” 在cocos creator里选择开发者&#xff0c;visual studio code工作流&#xff0c;选择添加编译任务。 添加 settings.json {"files.exclude":{"**/.git": true,"**/.DS_Sto…

数据结构与算法(三)线性表

线性表定义 线性表&#xff08;List&#xff09;&#xff1a;零个或多个数据元素的有限序列。 首先它是一个序列&#xff0c;其次&#xff0c;线性表强调是有限的。 前驱元素&#xff1a;若A元素在B元素的前面&#xff0c;则称A为B的前驱元素 后继元素&#xff1a;若B元素在…

【代码随想录day23】不同路径

题目 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的路径&#xff1f; 示…

明茨伯格的人际关系角色理论

明茨伯格的人际关系角色理论是由社会心理学家明茨伯格&#xff08;William Schutz&#xff09;在20世纪50年代提出的一种关于人际关系的理论。该理论主要探讨了人际关系中的三个角色&#xff1a;包容性、控制性和亲密性。这些角色代表了人们在互动中所表现出的行为和需求。下面…

ClickHouse进阶(六):副本与分片-2-Distributed引擎

进入正文前&#xff0c;感谢宝子们订阅专题、点赞、评论、收藏&#xff01;关注IT贫道&#xff0c;获取高质量博客内容&#xff01; &#x1f3e1;个人主页&#xff1a;含各种IT体系技术,IT贫道_Apache Doris,大数据OLAP体系技术栈,Kerberos安全认证-CSDN博客 &#x1f4cc;订阅…

Qt各个版本下载及安装教程(离线和非离线安装)

Qt各个版本下载链接&#xff1a; Index of /archive/qthttps://download.qt.io/archive/qt/ 离线安装 &#xff0c;离线安装很无脑&#xff0c;下一步下一步就可以。 我离线下载 半个小时把2G的exe下载下来了

使用boost::geometry::union_ 合并边界(内、外)- 方案一

使用boost::geometry::union_ 合并边界&#xff08;内、外&#xff09;&#xff1a;方案一 结合 boost::geometry::read_wkt() 函数 #include <iostream> #include <vector>#include <boost/geometry.hpp> #include <boost/geometry/geometries/point_x…

linux C编程 获取系统时间

1.clock_gettime #include<time.h> int clock_gettime(clockid_t clk_id,struct timespec *tp); struct timespec {time_t tv_sec; /* 秒*/long tv_nsec; /* 纳秒*/ }clk_id : CLOCK_BOOTTIME&#xff0c;以系统启动时间为时间原点的时间体系&#xff0c;不受其它因素的…