BZOJ_1601_[Usaco2008_Oct]_灌水_(最小生成树_Kruskal)

描述



http://www.lydsy.com/JudgeOnline/problem.php?id=1601

有\(n\)个田地需要灌溉,每个田地可以自己引水,花费为\(w[i]\),或者连接其他被灌溉的田地,花费为\(p[i][j]\),求最小花费.

分析



我第一眼看以为是dp,发现不对...

如果田地不能自己引水只能和其他田地连接,就是裸的最小生成树.

这里加了一个可以自己引水,相当于加了一个和所有点都有连边,且权值分别为\(w[i]\)的结点,然后求\(n+1\)个点的最小生成树即可.

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3
 4 const int maxn=300+5;
 5 struct edge{
 6     int u,v,w;
 7     edge(int u=0,int v=0,int w=0):u(u),v(v),w(w){}
 8     bool operator < (const edge &a) const { return w<a.w; }
 9 }g[maxn*maxn];
10 int n,ect,ans;
11 int f[maxn];
12 inline int read(int &x){ x=0;int k=1;char c;for(c=getchar();c<‘0‘||c>‘9‘;c=getchar())if(c==‘-‘)k=-1;for(;c>=‘0‘&&c<=‘9‘;c=getchar())x=x*10+c-‘0‘;return x*=k; }
13 inline int find(int x){ return x==f[x]?x:f[x]=find(f[x]); }
14 int main(){
15     read(n);
16     for(int i=1,w;i<=n;i++) g[++ect]=edge(0,i,read(w)), f[i]=i;
17     for(int i=1;i<=n;i++)for(int j=1,w;j<=n;j++){
18         read(w);
19         if(j<i) g[++ect]=edge(i,j,w);
20     }
21     sort(g+1,g+ect+1);
22     for(int i=1,j=n;i<=ect&&j;i++){
23         int fu=find(g[i].u),fv=find(g[i].v);
24         if(fu!=fv) f[fu]=fv, j--, ans+=g[i].w;
25     }
26     printf("%d\n",ans);
27     return 0;
28 }

时间: 2024-10-19 17:18:03

BZOJ_1601_[Usaco2008_Oct]_灌水_(最小生成树_Kruskal)的相关文章

【Usaco2008 Oct】灌水 (最小生成树)

题目描述 Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij(1<=pij<=100000,pij=pji,pii=0). 计算Farmer John所需的最少代价. 算法 最小生成树 思路 每块农田都有两种操作方法: 1.在农田上修水库 2.和别的农田连接 我们如果想要简化

【最小生成树】Bzoj1601[Usaco2008 Oct]灌水

Description Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij(1<=pij<=100000,pij=pji,pii=0). 计算Farmer John所需的最少代价. Sulotion 直接搞好像不太好做?于是新设一个点,作为总水源,和i点边权为wi,然后直接求

BZOJ1601: [Usaco2008 Oct]灌水

1601: [Usaco2008 Oct]灌水 Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 1280  Solved: 839[Submit][Status] Description Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij(1

1601: [Usaco2008 Oct]灌水

1601: [Usaco2008 Oct]灌水 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 1342  Solved: 881 [Submit][Status] Description Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij

sql面试题(学生表_课程表_成绩表_教师表)

sql面试题(学生表_课程表_成绩表_教师表) 原帖链接:http://bbs.csdn.net/topics/280002741 表架构 Student(S#,Sname,Sage,Ssex) 学生表 Course(C#,Cname,T#) 课程表 SC(S#,C#,score) 成绩表 Teacher(T#,Tname) 教师表 建表语句 CREATE TABLE student ( s# INT, sname nvarchar(32), sage INT, ssex nvarchar(8)

bzoj1601 灌水

Description Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij(1<=pij<=100000,pij=pji,pii=0). 计算Farmer John所需的最少代价. Input *第一行:一个数n *第二行到第n+1行:第i+1行含有一个数wi *第n+2行到

BZOJ 1601 USACO 2008 Oct. 灌水

[Description] Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij(1<=pij<=100000,pij=pji,pii=0). 计算Farmer John所需的最少代价. [题解] 新建一个节点0,把各个点与0点连边,边权为在该点上建水库的代价 然后跑一边最小生

bzoj 1601 灌水

题目大意: 决定把水灌到n块农田,农田被数字1到n标记 把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库 建造一个水库需要花费wi,连接两块土地需要花费Pij. 计算所需的最少代价 思路: 新开一个节点与每个农田连一条长为wi的边 然后直接最小生成树 1 #include<iostream> 2 #include<cstdio> 3 #include<cmath> 4 #include<cstdlib> 5 #include<cstr

[Usaco2008 Oct]灌水

题目描述 Farmer John已经决定把水灌到他的n(1<=n<=300)块农田,农田被数字1到n标记.把一块土地进行灌水有两种方法,从其他农田饮水,或者这块土地建造水库. 建造一个水库需要花费wi(1<=wi<=100000),连接两块土地需要花费Pij(1<=pij<=100000,pij=pji,pii=0). 计算Farmer John所需的最少代价. 输入格式 *第一行:一个数n *第二行到第n+1行:第i+1行含有一个数wi *第n+2行到第2n+1行:第