http://codeforces.com/problemset/problem/366/D

题意:给出n个点,m条边,a,b,ll,rr分别代表点a,点b相连,点a和点b的区间范围(ll,rr),然后让你选择边相连接使得可以从点1到点n,并且所有被选择的边所共同覆盖的区间范围最大。

4 4
1 2 1 10
2 4 3 5
1 3 1 5
2 4 2 7
6
4个点,4条边,现在可以从1--2--4,存在两条路径,(1,10)--(3,5)或者(1,10)——(2,7),要选择这所共同覆盖的区间范围最大的路径,那么就是(1,10)——(2,7),所覆盖的区间范围为7-2+1

若是最大区间范围没有,输出-1;

思路:仔细思考过后,会发现,其实若存在最大共同覆盖区间(p),那么区间p的左极限必然是某个区间的左端点,区间p的右极限必然是某个区间的右端点,当然这“某个区间”,可能会是一个区间,也可能会是两个区间......

如此的话,我是对所有区间按ll排序,然后枚举rr区间,找ll,用并查集判断是否可以从点1到点n.....可是wa多次......至今没有弄明白,是哪个地方思路问题?

wa代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std; struct node
{
int v1,v2;
int x,y;
//int len;
} s[3*1005];
int father[3000];
int n,m;
int vist[1005],flg1,flg2,ans,xx,yy;
int cmp(const node a,const node b)
{
if(a.x<b.x)
return 1;
else
return 0;
}
int find(int x)
{
int root,i=x;
while(x!=father[x])
x=father[x];
root=x;
x=i;
while(x!=father[x])
{
i=father[x];
father[x]=root;
x=i;
}
return root;
} int main()
{
while(scanf("%d%d",&n,&m)>0)
{
for(int i=0; i<m; i++)
{
scanf("%d%d%d%d",&s[i].v1,&s[i].v2,&s[i].x,&s[i].y);
//s[i].len=s[i].y-s[i].x+1;
}
sort(s,s+m,cmp);
ans=0;
for(int i=0; i<m; i++)
{
for(int j=0; j<=n; j++)
father[j]=j;
for(int j=0;j<m;j++)
{
if(s[i].y<s[j].x)
break;
if(s[j].x>s[i].x)
continue;
if(s[i].x>s[j].y)
continue;
int s1=find(s[j].v1);
int s2=find(s[j].v2);
if(s1!=s2)
{
father[s1]=s2;
}
if(find(1)==find(n))
{
if(ans<s[j].y-s[i].x+1)
ans=s[j].y-s[i].x+1;
//printf("%d %d\n",s[i].x,s[i].y);
}
}
}
if(ans==0)
printf("Nice work, Dima!\n");
else
printf("%d\n",ans);
}
return 0;
}

然后,我依旧是对ll排序,但是这次是枚举rr来找ll,依旧是并查集判断是否联通1到n,额,ac了......有点不可思议,思考ing

ac代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std; struct node
{
int v1,v2;
int x,y;
//int len;
} s[3*1005];
int father[3000];
int n,m;
int vist[1005],flg1,flg2,ans,xx,yy;
int cmp(const node a,const node b)
{
if(a.x<b.x)
return 1;
else
return 0;
}
int find(int x)
{
int root,i=x;
while(x!=father[x])
x=father[x];
root=x;
x=i;
while(x!=father[x])
{
i=father[x];
father[x]=root;
x=i;
}
return root;
} int main()
{
while(scanf("%d%d",&n,&m)>0)
{
for(int i=0; i<m; i++)
{
scanf("%d%d%d%d",&s[i].v1,&s[i].v2,&s[i].x,&s[i].y);
//s[i].len=s[i].y-s[i].x+1;
}
sort(s,s+m,cmp);
ans=0;
for(int i=0; i<m; i++)
{
for(int j=0; j<=n; j++)
father[j]=j; for(int j=0;j<m;j++)
{
if(s[i].y<s[j].x)
break;
if(s[i].y>s[j].y)
continue;
//if(s[j].x>s[i].x)
//continue;
int s1=find(s[j].v1);
int s2=find(s[j].v2);
if(s1!=s2)
{
father[s1]=s2;
}
//printf("%d %d\n",s[j].v1,s[j].v2);
if(find(1)==find(n))
{ if(ans<s[i].y-s[j].x+1)
ans=s[i].y-s[j].x+1;
break;
}
}
}
if(ans==0)
printf("Nice work, Dima!\n");
else
printf("%d\n",ans);
}
return 0;
}

经过思考,我觉得wa代码是因为rr无序,也就是说,我只对ll排序了,枚举ll所找到的rr是无规律的,导致本来有些区间是应该会先连通点1到点n的,而因为rr无序,扰乱了.....所以wa,额,为了验证是不是这样的,便有了接下来的代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std; struct node
{
int v1,v2;
int x,y;
//int len;
} s[3*1005];
int father[3000];
int n,m;
int vist[1005],flg1,flg2,ans,xx,yy;
int cmp(const node a,const node b)
{
if(a.y>b.y)
return 1;
else
return 0;
}
int find(int x)
{
int root,i=x;
while(x!=father[x])
x=father[x];
root=x;
x=i;
while(x!=father[x])
{
i=father[x];
father[x]=root;
x=i;
}
return root;
} int main()
{
while(scanf("%d%d",&n,&m)>0)
{
for(int i=0; i<m; i++)
{
scanf("%d%d%d%d",&s[i].v1,&s[i].v2,&s[i].x,&s[i].y);
//s[i].len=s[i].y-s[i].x+1;
}
sort(s,s+m,cmp);
ans=0;
for(int i=0; i<m; i++)
{
for(int j=0; j<=n; j++)
father[j]=j; for(int j=0;j<m;j++)
{
if(s[i].x>s[j].y)
continue;
if(s[i].y<s[j].x)
continue;
//if(s[i].x)
if(s[j].x>s[i].x)
continue;
int s1=find(s[j].v1);
int s2=find(s[j].v2);
if(s1!=s2)
{
father[s1]=s2;
}
//printf("%d %d\n",s[j].v1,s[j].v2);
if(find(1)==find(n))
{ if(ans<s[j].y-s[i].x+1)
ans=s[j].y-s[i].x+1;
break;
}
}
}
if(ans==0)
printf("Nice work, Dima!\n");
else
printf("%d\n",ans);
}
return 0;
}

ac了......真是这样的哈......

具体的来说,我觉得解一道题目,不仅仅只是ac了就可以的........

这道题目,对于区间的排序问题,有讲究的。比如说只有四个点的情况,1 2  1  10与2  4  2  7;
表示点1和点2相连,区间值为1--10,点2与点4相连,区间值为2--7;

现在按区间的ll从小到大排序:

s[0].v1=1   s[0].v2=2   s[0].ll=1   s[0].rr=10;

s[1].v1=2   s[1].v2=4   s[1].ll=2   s[1].rr=7;

若枚举ll来确定rr,那么就会出现10-2+1的输出,其实为什么会这样?自己在纸上动手画画就可以明白;

但枚举rr来确定ll,却不会产生这样的情况......

同理,若是按照区间rr从大到小排序,然后枚举ll来确定rr,也会是正确的.......至于为什么一种要从小到大,而另一种要从大到小,找两个这样的区间,一个区间被另一个区间包含,然后动手画画,就可以明白........

cf 366D D. Dima and Trap Graph (计算所有线段共同覆盖的某段区间)的更多相关文章

  1. Codeforces 336D Dima and Trap Graph 并查集

    Dima and Trap Graph 枚举区间的左端点, 然后那些左端点比枚举的左端点小的都按右端点排序然后并查集去check #include<bits/stdc++.h> #defi ...

  2. Codefroces 366 D Dima and Trap Graph (最短路)

    Dima and Trap Graph 题意:Dima和Inna越来越喜欢对方,但是Dima的室友缺很没有热情出去玩,然后Dima就把他室友Seryozha骗进了陷阱里.现在Seryozha想要从陷阱 ...

  3. cf D. Dima and Trap Graph

    http://codeforces.com/contest/366/problem/D 遍历下界,然后用二分求上界,然后用dfs去判断是否可以. #include <cstdio> #in ...

  4. 【BZOJ5334】数学计算(线段树)

    [BZOJ5334]数学计算(线段树) 题面 BZOJ 洛谷 题解 简单的线段树模板题??? 咕咕咕. #include<iostream> #include<cstdio> ...

  5. CF#214 C. Dima and Salad 01背包变形

    C. Dima and Salad 题意 有n种水果,第i个水果有一个美味度ai和能量值bi,现在要选择部分水果做沙拉,假如此时选择了m个水果,要保证\(\frac{\sum_{i=1}^ma_i}{ ...

  6. cf B. Little Dima and Equation

    http://codeforces.com/contest/460/problem/B import java.util.*; import java.math.*; public class Mai ...

  7. cf 366C C. Dima and Salad(01背包)

    http://codeforces.com/contest/366/problem/C 题意:给出n个水果的两种属性a属性和b属性,然后挑选苹果,选择的苹果必须要满足这样一个条件:,现在给出n,k,要 ...

  8. POJ 2125 Destroying the Graph 二分图最小点权覆盖

    Destroying The Graph Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 8198   Accepted: 2 ...

  9. CF 316E3 Summer Homework(斐波那契矩阵+线段树)

    题目链接:http://codeforces.com/problemset/problem/316/E3 题意:一个数列A三种操作:(1)1 x y将x位置的数字修改为y:(2)2 x y求[x,y] ...

随机推荐

  1. rsync配置及优化

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...

  2. /proc/net/sockstat 里的信息是什么意思?

    cat /proc/net/sockstat sockets: used 294 TCP: inuse 35 orphan 0 tw 0 alloc 45 mem 1 UDP: inuse 13 me ...

  3. Kickstart无人值守安装[转载]

    导言 作为中小公司的运维,经常会遇到一些机械式的重复工作,例如:有时公司同时上线几十甚至上百台服务器,而且需要我们在短时间内完成系统安装. 常规的办法有什么? 光盘安装系统===>一个服务器DV ...

  4. (转)Groupon前传:从10个月的失败作品修改,1个月找到成功 并不挶泥在这个点子上面,它反而往后站一步,看看他们已经做好的这个网站,可以再怎么包装成另一个完完全全不同的网站?所有的人所做的每件失败的事情中, 一定有碰到或含有成功的答案」在里面,只是他们不知道而已。 人不怕失败」,只怕宣布失败」

    (转)Groupon前传:从10个月的失败作品修改,1个月找到成功 今天读到 一个非常励志人心的故事 ,就像现在「叶问」有「前传」,最近很火红的团集购网站Groupon 也出现了「Groupon前传」 ...

  5. TensorFlow Google大会总结

    一.概述 介绍TPU,需要使用XLA编译,否则没有做内部优化,无法达到加速的效果: TPU相关的性能分析器: 二.新版本的输入库 之前TensorFlow的输入方式: feed_dict: 太过于低效 ...

  6. Android平台的音乐资源管理与播放

    Android平台基于Linux和开放手机联盟(OHA)系统,经过中国移动的创新研发,设计出拥有新颖独特的用户操作界面,增强了浏览器能力和WAP 兼容性,优化了多媒体领域的OpenCORE.浏览器领域 ...

  7. 图片标注工具LabelImg使用教程

    1.进入labelImg-master文件夹,在空白处使用 “Shift+鼠标右键” ,选择在此处打开命令窗口,依次输入下面语句即可打开软件. pyrcc4 -o resources.py resou ...

  8. maven配置(安装&使用&私服)文档

    1.Maven 环境搭建 Maven 是一个基于 Java 的工具,所以要做的第一件事情就是安装 JDK. 系统要求 项目 要求 JDK Maven 3.3 要求 JDK 1.7 或以上 Maven ...

  9. 在windows下安装Jupyter Notebook的安装和使用

    1 认识jupyter jupyter /ˈdʒuːpɪtə(r)/可以提供适合捕捉整个计算过程的基于web的应用程序:开发.记录和执行代码,以及结果输出. jupyter Notebook提供了两个 ...

  10. 【转】Lisp 已死,Lisp 万岁!

    Lisp 已死,Lisp 万岁! 有一句古话,叫做“国王已死,国王万岁!”它的意思是,老国王已经死去,国王的儿子现在继位.这句话的幽默,就在于这两个“国王”其实指的不是同一个人,而你咋一看还以为它自相 ...