[置顶] 解成电OJ1003真实的谎言的记录
原题目
Description
N个人做一个游戏,游戏中每个人说了一句话(可能是真的也可能是假的)
第i个人说:“N个人中有至少有ai个,至多有bi个人说的是真话!”(i = 1, 2, 3…..N)你能推断出最多能有多少个人说的是真话吗?
1 <= N <= 100000;
0 <= ai<=bi<=1000000000;
Input
第一行为一个整数T,代表测试数据的组数;
每组数据以n开头,接下来有n行,每行两个整数ai,bi(代表第i个人说的);
Output
输出占一行。如果原问题有解,输出最多能有多少个人说的是真话;否则输出-1.
Sample Input
2
3
0 0
1 1
2 2
3
2 5
3 5
0 3
Sample Output
1
3
分析
题目什么意思呢?
总共N个人,那说真话最多只有N个人,为什么ai和bi的值可以远大于N呢?结合示例不难得知,原来ai和bi的值只是给个说真话的人数的范围,说真话的人数自然是在N以内的。假如某人可能说真话,则其给出的区间必然包含1至N的某值,因此说出区间[0,0]的那家伙肯定说假话;如果给出的区间左端值ai>N,那么他也肯定说假话。因此,判断可能说真话的最多人数时,可以不考虑这些必然说假话的人。假如最多X个人全部说真话(当然其余N-X说假话了),那么X必然在他们给出的交集区间[ax,bx]中。
于是,本题的意思就是寻找最大的X满足:X属于[ax,bx],其中[ax,bx]表示X个人给出的交集区间。
那如何才能得到呢?对任意一个给出的区间[ai,bi],他能结合的可能其他人数是有限的,不妨设为Xi,Xi的最大值取决于区间的右端值。N个人中,可能有些部分之间没有交集,应该单独考虑这些部分,最后再比较。对于有交集的部分,可以证明只有按照Xi有序进行结合,才能得到最大的X。结合所能给出的可能全部说真话的人数取决于他们当中最小的Xi,不妨设X1<X2<X3(逆序等效),则X1和X2结合、X1和X3结合均取决于X1。如果X1、X2、X3能全部结合,即X1+X2+X3在他们的交集区间中,那么与顺序无关,但是倘若只有部分能结合,则显然能结合的部分的最小Xi同较小Xk结合,使得剩下的Xj较大,则可能使Xj同其他结合得到更大值。因此,可以按照给出区间的右端值进行排序,然后再逐个结合判断。
判断的大致过程:若有交集并且结合后的X值不大于交集右端值(可能也小于左端值,因为结合尚未完成),则结合;否则表示这部分结合完毕,可以判断是否可能是符合题意的X,即X是否在交集区间中,若是则记录以便同其他部分比较,否则继续另一部分的结合。最后比较各部分得到最大的X值即为所求。
解答
我提交了好几次,推测出成电OJ的测试数据已经是排好序的了,因此解题代码中就不考虑排序了。实际上,我写了排序后,测了几组数据没问题,而AC的代码(即未排序的)测未排序的数据会出错。提交排序的后,结果run time error,原来不允许调用qsort快排例程。
附录
下面是我的AC代码。这个代码的时间39ms、空间852KB,在该题Best Solution中排第九,而且所占空间是最小的。但是代码长度略长,后来去掉join函数简化到约50行,可是结果却没这个好。最后本人能力实在有限,不对之处恳请指正,不胜感激。
真实的谎言 (1003 )
User: xuebao2013 Result: AcceptedDate: 2013-10-10 03:07:55
- #include<stdio.h>
- bool join(int *AB,int A,int B)
- {
- int a=AB[0],b=AB[1];
- if(B<a||A>b)return false;
- if(B<b)b=B;
- if(A>a)a=A;
- if(AB[2]+1>b)return false;
- AB[0]=a;
- AB[1]=b;
- AB[2]++;
- return true;
- }
- int main()
- {
- int T;
- scanf("%d",&T);
- int AB[3],N,A,B,maxcount;
- int i,j;
- for(i=0;i<T;i++)
- {
- AB[0]=AB[1]=0;
- AB[2]=-1;
- scanf("%d",&N);
- maxcount=0;
- for(j=0;j<N;j++)
- {
- scanf("%d%d",&A,&B);
- if(B<1||A>N)continue;
- if(AB[2]==-1)
- {
- AB[0]=A;AB[1]=B;
- AB[2]=1;
- continue;
- }
- if(join(AB,A,B))continue;
- if(AB[2]>=AB[0])
- if(AB[2]>maxcount)
- maxcount=AB[2];
- AB[0]=A;AB[1]=B;
- AB[2]=1;
- }
- if(AB[2]>=AB[0])
- if(AB[2]>maxcount)
- maxcount=AB[2];
- if(maxcount>0)printf("%d\n",maxcount);
- else printf("-1\n");
- }
- return 0;
- }
[置顶] 解成电OJ1003真实的谎言的记录的更多相关文章
- ahk之路:利用ahk在window7下实现窗口置顶
操作系统:win7 64位 ahk版本:autohotkey_L1.1.24.03 今天安装了AutoHotkey_1.1.24.03.SciTE.PuloversMacroCreator,重新开始我 ...
- DataGridView控件绑定数据之后,置顶操作
一个小小的置顶,就搞了半个小时,还是记录一下吧. 1.第一个问题就是datatable的插入只能是Insert DataRow,但是获取选中的行,都是DataGridViewRow,不能直接转换. 找 ...
- [置顶] Android开发笔记(成长轨迹)
分类: 开发学习笔记2013-06-21 09:44 26043人阅读 评论(5) 收藏 Android开发笔记 1.控制台输出:called unimplemented OpenGL ES API ...
- C#或者WPF中让某个窗体置顶
原文:C#或者WPF中让某个窗体置顶 前记:在工作中有个需求,要求不管到那个界面,我必须让一个浮动条(其实是个窗体)置顶. 我用wpf,因为有之前有好几个界面已经设置成topmost了,所以在这几个界 ...
- js之滚动置顶效果
0.js获取高度 ? 1 2 3 4 5 6 document.all // 只有ie认识 document.body.clientHeight // 文档的高,屏幕 ...
- 【搬运工】NOIP吧置顶贴
目的是存置顶贴里的链接.. 原帖:http://tieba.baidu.com/p/1753284199 资源站:*C++资源:http://tieba.baidu.com/p/1239792581* ...
- android linearlayout imageview置顶摆放
在练习android时,想在Linearlayout内放一图片,使其图片置顶,预期效果是这样的: 但xml代码imageview写成这样后, <ImageView android:layout_ ...
- QT窗口置顶/真透明/背景模糊/非矩形/跳过任务栏分页器/无边框/无焦点点击/焦点穿透
qt 窗口置顶/真透明/背景模糊/非矩形/跳过任务栏分页器/无边框/无焦点点击/焦点穿透 窗口置顶qt 里是 setWindowFlags(Qt::WindowStaysOnTopHint)kde 里 ...
- [知了堂学习笔记]_css3特效第二篇--行走的线条&&置顶导航栏
一.行走的线条. 效果图(加载可能会慢一点儿,请稍等...): html代码: <div class="movingLines"> <img src=" ...
随机推荐
- ANDROID_MARS学习笔记_S01_011ProgressBar
文档是这样来设置样式 <ProgressBar android:layout_width="wrap_content" android:layout_height=" ...
- P147、面试题26:复杂链表的复制
题目:请实现ComplexListNode* Clone(ComplexListNode* pHead),复制一个复杂链表.在复杂链表中,每个结点除了有一个m_pNext指针指向下一个结点外,还有一个 ...
- P94、面试题12:打印1到最大的n位数
题目:输入数字n,按顺序打印出从1最大的n位十进制数.比如输入3,则打印出1,2,3一直到最大的3位数999. 思路:先把字符串中的每一个数字都初始化为‘0’,然后每一次为字符串表示的数字加1,再打印 ...
- Linux信号列表
我们运行如下命令,可看到Linux支持的信号列表: ~$ kill -l1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL5) SIGTRAP 6) SIGABRT 7) ...
- 比nerdtree更好的文件浏览器:vimfiler
通过:VimFilerExplorer来打开一个文件浏览器 h:收起 t:展开 -:close 回车:进入或展开 空格:收起
- php 对象调用方法
static union _zend_function *zend_std_get_method(zval **object_ptr, char *method_name, int method_le ...
- UVa 1451 (数形结合 单调栈) Average
题意: 给出一个01串,选一个长度至少为L的连续子串,使得串中数字的平均值最大. 分析: 能把这道题想到用数形结合,用斜率表示平均值,我觉得这个想法太“天马行空”了 首先预处理子串的前缀和sum,如果 ...
- Python脚本控制的WebDriver 常用操作 <五> 访问链接
下面将使用webdriver来访问一个web链接 测试用例场景 测试中,经常会点击几个链接来进行操作,所以访问链接是基本的常见操作 Python脚本 from selenium import webd ...
- FZU 2129 子序列个数
Problem Description 子序列的定义:对于一个序列a=a[1],a[2],......a[n].则非空序列a'=a[p1],a[p2]......a[pm]为a的一个子序列,其中1& ...
- IE兼容性问题解决方案1--ajax请求不发送到后台
相信很多小伙伴会遇到这种问题,用ajax做异步请求的时候,在IE浏览器下,并没有发送出去.但是相关程序确实执行了.为什么呢? 原来这是IE缓存方式的原因,所以呢,用下边的解决方案吧. 1.在请求的UR ...