下午花了三个小时肝这道题,心态差点爆炸!

下面是分析:

 1 题目要求:
2 求最小生成树
3 但是
4 - a是从1号点开始的 --> 如果以后的某个点比一号高,则不可能到达
5 - a只能从高往低处滑
6 - 可能某两个景点没有轨道连接 -->没有处在一个联通快里?
7
8 生成树的要求:
9 - 树根高度最大
10 - 子节点的高度必须比父亲结点小
11 - 边权和尽可能小
12
13 自己造的样例:
14 输入:
15 4 4
16 4 2 3 1
17 1 2 3
18 2 3 2
19 2 4 4
20 3 4 1
21
22 输出:
23 3 7

从题解上爬的Solotion:

 1 为保证我们只会由高到低,我们就只建立由高向低的单向边即可。
2
3 对于建立出来的图A,由1点开始宽搜,将扩展到的点和边加入一个新图B,
4 所有扩展到的点便是能到达的最多点。
5
6 我们再在这个新图上跑Kruskal求最小生成树,求得最短距离。
7
8 对于排序部分,为保证有尽可能多的点在最小生成树里,
9 我们按终点的高度为第一关键字从大到小排序,边长为第二关键字从小到大排序;
10
11 这样就能保证拓展的点最多,进而再用最小生成树求最短距离。

最终AC代码:

 1 /*
2 Work By:Suzt_ilymtics
3 */
4 #include<iostream>
5 #include<cstdio>
6 #include<algorithm>
7 #include<queue>
8 using namespace std;
9 const int MAXN=1e5+5;
10 const int MAXM=1e6+6;
11 struct edge{
12 int from,to,nxt;
13 long long w;
14 }e_a[MAXM << 1],e_b[MAXM << 1];
15 int head_a[MAXN],num_edge_a;
16 int num_edge_b;
17 int n,m,cnt;
18 long long ans;
19 int h[MAXN],f[MAXN];
20 bool vis[MAXN];
21 queue<int> q;
22
23 int find(int x) {return f[x] == x ? x : f[x]=find(f[x]);}
24
25 bool cmp(edge x,edge y){return h[x.to] == h[y.to] ? x.w < y.w : h[x.to] > h[y.to];}
26
27 void add_a(int from,int to,int w){
28 e_a[++num_edge_a].from = from;
29 e_a[num_edge_a].to = to;
30 e_a[num_edge_a].w = w;
31 e_a[num_edge_a].nxt = head_a[from];
32 head_a[from] = num_edge_a;
33 }
34
35 void add_b(int from,int to,int w){
36 e_b[++num_edge_b].from = from;
37 e_b[num_edge_b].to = to;
38 e_b[num_edge_b].w = w;
39 }
40
41 void bfs(int x){
42 q.push(x);vis[1]=1;
43 while(!q.empty()){
44 int t=q.front(); q.pop();
45 for(int i=head_a[t];i;i=e_a[i].nxt){
46 add_b(e_a[i].from,e_a[i].to,e_a[i].w);
47 if(!vis[e_a[i].to]){
48 vis[e_a[i].to]=1;
49 q.push(e_a[i].to);
50 cnt++;
51 }
52 }
53 }
54 }
55
56 void kls(){
57 for(int i=1;i<=num_edge_b;++i){
58 int uf=find(e_b[i].from),vf=find(e_b[i].to);
59 if( uf == vf ) continue;
60 else{
61 f[uf] = vf;
62 ans+=e_b[i].w;
63 }
64 }
65 }
66
67 int main()
68 {
69 scanf("%d%d",&n,&m);
70 for(int i=1;i<=n;++i){
71 scanf("%d",&h[i]),f[i]=i;
72 }
73 for(int i=1,u,v,w;i<=m;++i){
74 scanf("%d%d%d",&u,&v,&w);
75 if(h[u]>=h[v]) add_a(u,v,w);
76 if(h[u]<=h[v]) add_a(v,u,w);
77 }
78 bfs(1);
79 sort(e_b+1,e_b+num_edge_b+1,cmp);
80 kls();
81 printf("%d %lld",cnt+1,ans);
82 return 0;
83 }

 The end

(这篇文章终于能看了点)

P2573 [SCOI2012]滑雪 题解的更多相关文章

  1. P2573 [SCOI2012]滑雪

    题目描述 a180285非常喜欢滑雪.他来到一座雪山,这里分布着 M 条供滑行的轨道和 N 个轨道之间的交点(同时也是景点),而且每个景点都有一编号 i ( 1≤i≤N )和一高度 Hi.a18028 ...

  2. BZOJ 2753 [SCOI2012] 滑雪和时间胶囊 最小生成树

    题目链接: 题目 2753: [SCOI2012]滑雪与时间胶囊 Time Limit: 50 Sec Memory Limit: 128 MB 问题描述 a180285非常喜欢滑雪.他来到一座雪山, ...

  3. Bzoj2753 [SCOI2012]滑雪与时间胶囊

    2753: [SCOI2012]滑雪与时间胶囊 Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 2282  Solved: 796 Descriptio ...

  4. Code[VS] 2152 滑雪题解

    Code[VS] 2152 滑雪题解 题目描述 Description trs喜欢滑雪.他来到了一个滑雪场,这个滑雪场是一个矩形,为了简便,我们用r行c列的矩阵来表示每块地形.为了得到更快的速度,滑行 ...

  5. bzoj 2753: [SCOI2012]滑雪与时间胶囊 -- 最小生成树

    2753: [SCOI2012]滑雪与时间胶囊 Time Limit: 50 Sec  Memory Limit: 128 MB Description a180285非常喜欢滑雪.他来到一座雪山,这 ...

  6. 【BZOJ 2753】 2753: [SCOI2012]滑雪与时间胶囊 (分层最小树形图,MST)

    2753: [SCOI2012]滑雪与时间胶囊 Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 2457  Solved: 859 Descriptio ...

  7. BZOJ2753 SCOI2012 滑雪与时间胶囊 【最小生成树】*

    BZOJ2753 SCOI2012 滑雪与时间胶囊 Description a180285非常喜欢滑雪.他来到一座雪山,这里分布着M条供滑行的轨道和N个轨道之间的交点(同时也是景点),而且每个景点都有 ...

  8. 2753: [SCOI2012]滑雪与时间胶囊

    2753: [SCOI2012]滑雪与时间胶囊 Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 2633  Solved: 910 Descriptio ...

  9. bzoj 2753: [SCOI2012] 滑雪与时间胶囊 Label:MST

    题目描述 a180285非常喜欢滑雪.他来到一座雪山,这里分布着M条供滑行的轨道和N个轨道之间的交点(同时也是景点),而且每个景点都有一编号i(1<=i<=N)和一高度Hi.a180285 ...

随机推荐

  1. Linux腾讯云下安装mysql

    百度云盘下载地址https://pan.baidu.com/s/1MqUEdeqZuQbq-veLuVItQQ 将下载好的mysql-5.7.14-linux-glibc2.5-x86_64.tar. ...

  2. 152. Maximum Product Subarray动态规划连乘最大子串

    Find the contiguous subarray within an array (containing at least one number)which has the largest p ...

  3. 虚拟机安装Ubuntu 16.04系统实操教程 详尽步骤 vmware ESXi亲测通过

    1 Ubuntu 16.04系统安装要求 Ubuntu 16.04 LTS下载最新版本的Ubuntu,适用于台式机和笔记本电脑. LTS代表长期支持,这意味着有五年免费安全和维护更新的保证. Ubun ...

  4. [linux]makefile多目录

    在使用makefile多目录编写前需要掌握几个函数及符号 自定义变量 target=edit 引用的时候直接使用 $(target) 有点像C语言中的#define,这里的 $(target)会被替换 ...

  5. JAVA注解基础知识

    定义-摘自JAVA编程思想 注解(元数据)为我们在代码中添加信息提供了一种形式化的方法,使我们在稍后某个时刻能非常方便的使用这些数据. 背景: 注解是JAVA5的一种新特性,是为应对C#等之类对JAV ...

  6. codeforces 1451D,一道有趣的博弈论问题

    大家好,欢迎来到codeforces专题. 今天选择的问题是Contest 1451场的D题,这是一道有趣简单的伪博弈论问题,全场通过的人有3203人.难度不太高,依旧以思维为主,坑不多,非常友好. ...

  7. IndexedDB详解

    目录 简介 IndexedDB简介 IndexedDB的使用 IndexedDB的浏览器支持 创建IndexedDB indexdb中的CURD 使用游标cursor 简介 IndexedDB是一种在 ...

  8. fatal error C1045: 编译器限制 : 链接规范嵌套太深

    前言 我相信你是遇到了同样的问题.通过搜索引擎来到这里的.为了不耽误排查问题的时间,我提前说明一下这篇文章所描述的问题范畴: 我遇到的问题和 c++ 模板相关: 如果我减少传递的参数的话,是有可能避免 ...

  9. 有关Servlet的面试题

    CGI(Common Gateway Interface),通用网管接口 通用网管接口,简称CGI,是一种根据请求信息动态产生回应内容的技术.通过CGI,web服务器可以根据请求的不同,启动不同的外部 ...

  10. Azure 存储简介

    Azure Storage Account(存储账户)包含所有Azure Storage的数据对象,包括Blob.Data Lake Gen2,File.Queue.Disk和Table等服务,该St ...