题目の传送门:https://www.luogu.org/problem/show?pid=1550

精简版题意(本来就精简了不是么):n个点,每个点可以选择打井或从别的有水的点引水,求所有点都有水用的最小费用。(我是不是一说反而更不懂了)

这题其实并不是很难,讲道理贪心都能过。。
令我感到有趣的是这题中一种非常神奇的解题思路。。

首先,如果这个题没有打井的选项,显然是最小生成树
然而我们现在要打井,就不能直接用最小生成树做了。。
殊不知,这题还是一个最小生成树~

这就需要一些小小的、巧妙的办法了。。哦,

新建一个伪节点n+1,向每个点连边,边权是这一点打井的费用!!!

然后我们就可以跑最小生成树了2333
话说是不是只有我一个人觉得这方法很巧妙orz
似乎可以证明?其实证明不看也就不看了。。

在最小生成上,抛开我们新建的伪节点不看,剩下的部分一定是一个或多个连通块,而且这些连通块也满足最小生成树的性质,费用是最小的。而这些连通块中,需要有一个打井费用最小的节点来打井,这一点我们跑最小生成树的时候就连到了伪节点上。
显然,对于点i,如果引水费用小于打井,跑最小生成树时就会连到连通块上,表示选择引水,反之则连到伪节点上,表示打井。因为最小生成树上有伪节点存在,从而保证了至少有一个点与其相连,即至少有一个点选择打井。

十分不严谨的证明完了,大家理解就好,不要深究。。

然后就是代码了,明白了原理就非常简单了。。
尽管图非常稠密,然而我还是跑Kruskal,毕竟数据范围只有N<=300,边数也不会超过10W
其实我不太会Prim(尤其是堆优化Prim)= =

代码:

#include <cstdio>
#include <algorithm>
using std::sort; const int MAXV=303;
const int MAXE=MAXV*MAXV+100; struct edge{
int from,to,data;
}e[MAXE]; int fa[MAXV],tot,cnt,ans; int find(int x){
if(fa[x]!=x) fa[x]=find(fa[x]);
return fa[x];
} inline int gnum(){
int a=0;char c=getchar();bool f=0;
for(;(c<'0'||c>'9')&&c!='-';c=getchar());
if(c=='-') c=getchar(),f=1;
for(;c>='0'&&c<='9';c=getchar()) a=(a<<1)+(a<<3)+c-'0';
if(f) return ~a+1; return a;
} bool operator <(const edge &a,const edge &b){
return a.data<b.data;
} void build(int x,int y,int z){
e[++tot].from=x; e[tot].to=y; e[tot].data=z;
} int main(){
int n=gnum();
for(int i=1;i<=n;i++)
build(i,n+1,gnum()),fa[i]=i;
fa[n+1]=n+1;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
build(i,j,gnum());
sort(e+1,e+tot+1);
for(int i=1;i<=tot;i++){
if(cnt==n) break;
int x=find(e[i].from),y=find(e[i].to);
if(x-y) fa[y]=x,ans+=e[i].data;
}
printf("%d",ans);
}

就这样,虽然跑得慢,但是能过嘛= =

你说贪心?留给你自己思索去吧~~

【学术篇】洛谷1550——打井Watering Hole的更多相关文章

  1. 洛谷P1550 [USACO08OCT]打井Watering Hole

    P1550 [USACO08OCT]打井Watering Hole 题目背景 John的农场缺水了!!! 题目描述 Farmer John has decided to bring water to ...

  2. Kruskal || BZOJ 1601: [Usaco2008 Oct]灌水 || Luogu P1550 [USACO08OCT]打井Watering Hole

    题面:P1550 [USACO08OCT]打井Watering Hole 题解:无 代码: #include<cstdio> #include<cstring> #includ ...

  3. bzoj1601 / P1550 [USACO08OCT]打井Watering Hole(堆优化prim)

    P1550 [USACO08OCT]打井Watering Hole   对于自己建水库的情况,新建一个虚拟结点,和其他点的边权即为自建水库的费用 这样问题就转化为一个裸最小生成树问题了. 这里用堆优化 ...

  4. Luogu P1550 打井Watering Hole

    P1550 [USACO08OCT]打井Watering Hole 题目背景 John的农场缺水了!!! 题目描述 Farmer John has decided to bring water to ...

  5. LCT总结——概念篇+洛谷P3690[模板]Link Cut Tree(动态树)(LCT,Splay)

    为了优化体验(其实是强迫症),蒟蒻把总结拆成了两篇,方便不同学习阶段的Dalao们切换. LCT总结--应用篇戳这里 概念.性质简述 首先介绍一下链剖分的概念(感谢laofu的讲课) 链剖分,是指一类 ...

  6. 题解——洛谷P1550 [USACO08OCT]打井Watering Hole(最小生成树,建图)

    题面 题目背景 John的农场缺水了!!! 题目描述 Farmer John has decided to bring water to his N (1 <= N <= 300) pas ...

  7. 洛谷 题解 P1550 【[USACO08OCT]打井Watering Hole】

    本题看似很难,实际上思路非常简单--如果你想通了. 首先有一个问题:图中有几个点?大部分的人会回答\(n\)个点.错了,有\(n+1\)个. 多出来的那个点在哪?关键在于你要理解每一个决策的意义.实际 ...

  8. 洛谷P1550打井

    打井 题目 该题是一个最小生成树的好题,但是比起一般的最小生成树来说他不仅仅有各个井相连,而且还要和地下水相连,所以地下水我们也可以看成一口井. 代码 #include <bits/stdc++ ...

  9. luogu P1550 [USACO08OCT]打井Watering Hole

    题目背景 John的农场缺水了!!! 题目描述 Farmer John has decided to bring water to his N (1 <= N <= 300) pastur ...

随机推荐

  1. NX二次开发-UF_OBJ_cycle_by_name遍历名字

    使用前自己要看好名字是加在body,还是face,还是curve,或者其他,别加错了. NX9+VS2012 #include <uf.h> #include <uf_obj.h&g ...

  2. NX二次开发-UFUN获取图层的状态UF_LAYER_ask_status

    NX11+VS2013 #include <uf.h> #include <uf_ui.h> #include <uf_layer.h> UF_initialize ...

  3. NPAPI插件开发新手容易遇到的问题

    在网上找了一个npdemo的例子,编译了一下在FireFox运行正常,在Chrome下就是不行,也没任何提示. 折腾了好久,最后发现是rc文件 支持语言编码问题 NPAPI插件开发详细记录:用VS20 ...

  4. gvim 安装YouCompleteMe插件

    可以参考:YouCompleteMe#full-installation-guide 可以直接下载: http://pan.baidu.com/s/1dDIq2Al 密码: si5q 确保vim支持p ...

  5. Adobe Fireworks CS6 win64的安装

    网页三大剑客之一    FW的安装 本人也是找了半天才找到的. (没有视频)这里先感谢原帖给我的链接https://blog.csdn.net/qq_38053395/article/details/ ...

  6. Mysql 插入数据,随机事件选择

    在拼写sql的 时候,mysql字段如果需要添加当前时间可以用NOW() 函数 // String sql = ("insert into tablename(content, create ...

  7. myeclipse问题

    eclipse使用过程中发现汉字太小,几乎不可辨识. 更改办法:eclipse界面依次选择“window”–“preference”–“general”–“appearance”–“color and ...

  8. JS函数 函数的作用,可以写一次代码,然后反复地重用这个代码。

    什么是函数 函数的作用,可以写一次代码,然后反复地重用这个代码. 如:我们要完成多组数和的功能. var sum; sum = 3+2; alert(sum); sum=7+8 ; alert(sum ...

  9. 随笔记录 grub引导故障修复 2019.8.7

    系统备份: [root@localhost ~]# mkdir /abc [root@localhost ~]# mount /dev/sdb1 /abc [root@localhost ~]# dd ...

  10. next() 与 nextLine() 区别

    next() 与 nextLine() 区别 next(): 1.一定要读取到有效字符后才可以结束输入. 2.对输入有效字符之前遇到的空白,next() 方法会自动将其去掉. 3.只有输入有效字符后才 ...