题目

本题的贪心算法策略需要深入思考一下

看到题目,最初没有理解题目的要求:看尽量多的完整的节目。尽量多是指数量多,自己理解成观看的时间最长。这样想其实简化了这道题。

正确理解题意后,首先想到的想法是:选择一个节目A,结束后选择另一个节目,如果节目A的时间同时覆盖了节目BC的时间,那么A就不应该看。怎么选择合适的节目?如果都把所有的节目考察一遍,统计看到的节目数量,成了穷举法,不是贪心算法。

遇到第一个关卡:如何在思维上找到突破口/切入点,当思路到达“如何选择合适的节目”这个问题时卡住,应该进一步思考“什么是合适的节目?”

这道题的贪心策略不再显而易见,像是解数学题,某个思路可以吗?需要深入思考、验证,不能蜻蜓点水之后就草率感觉“这样的思路好像不行吧”。

一个判断逻辑没有选择好浪费了大段时间

问题是这样的:

当遇到“当前节目能不能计入总数?”这个问题时,思路就绕了一小会,产生了下面的邪恶的代码。

这段代码在本地测试样例输入后,通过,到OJ上答案错误。

开始分析“样例中没有涉及但可能出现的测试数据”:如果节目A和节目B的时间完全重叠呢?详情见代码中的【错误1】

根据这个新情况修改一次后,造成【错误2】

修改错误2,造成【错误3】

感受:太着急的想法引来的问题多,复杂的想法引来的问题多。很多“直接的办法”往往是简单且可行的。

当一遇到“这个办法怎么这么复杂,要考虑的情况怎么这么多”的感觉时,你就要提高警惕了:思路上很有可能已经误入歧途。

很多时候,复杂不是问题本身困难,而是解决方法有待改进的征兆。

而误入歧途的根源是:着急,懒惰。想到一个办法就立马采用,而不进一步斟酌。

罪魁祸首:

for (former = i - 1; former > 0; former--)
{
if( list[i].tag == 1 && list[former].T_e > list[i].T_s)
{
break;
}
if(former == 0)
{
num++;
list[i].tag = 1;
}
}
/*former是游标,依次指向前面的节目
*num是可以收看的节目的总数,记录的是整个程序的最终输出结果
*tag是标记某个节目是否收看过了
*/

这段代码的背后的想法是:当前节目current能否计入num?取决于current的开始时间是否在前面“收看过的节目”的时间段中。于是拿着current的开始时间去和前面标记为已收看(tag==1;)的节目的时间段比较。

去和前面的时间比较!!!

一个罪恶的想法:去和前面的数据比较

但凡是要回头在前面处理过的数据中再捯饬,都是需要谨慎谨慎再谨慎的!

就目前遇到的程序中,似乎需要回头的时候,总有一个利器可以代替回头看:用一个变量/哨兵记录一下“目前处于什么状态”

太有用了!太好用了!

答案中的做法:

int currentTime = 0;
for (int i = 0; i < n; i++)
{
if (currentTime <= list[i].T_s)
{
currentTime = list[i].T_e;
num++;
}
}

无需回头看,无需tag标记。

写到这里,仔细想想,最初都想到了用tag标记是否看过某个节目,这就是向已经处理过的数据中添加tag的想法。为什么不更直接点,将“当前时间”作为tag来表示数据处理到了何处?!

功力不够,继续修炼!

完整代码

#include <stdio.h>
#include <algorithm>
using namespace std;
struct show {
int T_s;
int T_e;
int tag; //tag==0表示未看,tag==1表示已看 //用于排序sort
bool operator < (const show & B) const
{
return T_e < B.T_e;
}
} list[105];
int totalNum[105], output = 0;
int main()
{
int n;
//int num = 0;
int num;
while (scanf("%d", &n) != EOF)
{
if (n == 0)
break;
num = 0; //清除上一个测试用例的计数结果
for (int i = 0; i < n; i++)
{
scanf("%d%d", &list[i].T_s, &list[i].T_e);
list[i].tag = 0;
}
sort(list, list + n);
//list[i]为当前节目——当前结束最早的节目 for (int i = 0; i < n; i++)
{
int former;
if (list[i].tag == 0)
{
//错误2的版本:for(former = i; former > 0; former--)
for (former = i - 1; former > 0; former--)
{
//如果前面没有冲突则num++
//如果前面有冲突则i++
/*【错误1】
/*if中添加了tag的检查,这个造成了错误:如果list[0]和list[1]完全一样则多计数一次
*因为当考察list[1]时,list[1].tag==0,无法在former==1时break
*当former==0时,该if满足,但是进入if后break与否都不影响former==0
*break:说明时间冲突,但是此时former仍然==0,导致有冲突的情况下节目数仍要加1
*不break:说明时间没有冲突,此时是正确的。
*if (list[former].tag == 1 && list[former].T_e > list[i].T_s)
*尝试改进:去掉list[former].tag==1这个条件,造成错误2
*/
/*【错误2】这个if条件会导致former==i时,即节目A和节目A比较,而节目A.T_e>节目A.T_s恒成立
*导致后面所有的输入都不被正确处理
*改进:former的初始值从i-1开始, 导致错误3
*/
/*【错误3】回到了原点,下面这个if中没有list[former].tag == 1的检查,会造成节目A并没有收看,
*但考察节目B(i=B)时,节目B的时间与A去比较,由于A压根没有看,所以不需要与A比较!
*另外一个错误:当i==0时,former==-1,导致list[0]这个第一个节目永远都无法标记为可以观看的。
*问题的根源:if(former == 0) num++,这个增加可收看的节目个数的“判断条件”选的非常不好!!!
*/
if(list[former].T_e > list[i].T_s)
{
break;
}
}
if (former == 0)
{
num++;
list[i].tag = 1;
}
}
}
totalNum[output++] = num; /*这是答案中的版本,简洁明了!!!!!!!!!!!!!!!!!!
*思想是:添加一个指向当前时间点的哨兵currentTime
*于是,不再需要回顾(关注)前面节目的时间段了!
*直接将currentTime与待考察的节目队列中下一个节目的开始时间T_s比较
*currentTime<=list[i].T_s说明当前时间在下一个节目开始之前,则下一个节目可以收看!
int currentTime = 0;
for (int i = 0; i < n; i++)
{
if (currentTime <= list[i].T_s)
{
currentTime = list[i].T_e;
num++;
}
}
totalNum[output++] = num;
*/
} for (int i = 0; i < output; i++)
{
printf("%d\n", totalNum[i]);
}
return 0;
}

【九度OJ】题目1434贪心算法的更多相关文章

  1. 九度OJ 1433 FatMouse -- 贪心算法

    题目地址:http://ac.jobdu.com/problem.php?pid=1433 题目描述: FatMouse prepared M pounds of cat food, ready to ...

  2. 九度OJ 1082 代理服务器 -- 贪心算法

    题目地址:http://ac.jobdu.com/problem.php?pid=1082 题目描述: 使用代理服务器能够在一定程度上隐藏客户端信息,从而保护用户在互联网上的隐私.我们知道n个代理服务 ...

  3. 九度OJ 题目1384:二维数组中的查找

    /********************************* * 日期:2013-10-11 * 作者:SJF0115 * 题号: 九度OJ 题目1384:二维数组中的查找 * 来源:http ...

  4. hdu 1284 关于钱币兑换的一系列问题 九度oj 题目1408:吃豆机器人

    钱币兑换问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Sub ...

  5. 九度oj题目&amp;吉大考研11年机试题全解

    九度oj题目(吉大考研11年机试题全解) 吉大考研机试2011年题目: 题目一(jobdu1105:字符串的反码).    http://ac.jobdu.com/problem.php?pid=11 ...

  6. 九度oj 题目1007:奥运排序问题

    九度oj 题目1007:奥运排序问题   恢复 题目描述: 按要求,给国家进行排名. 输入:                        有多组数据. 第一行给出国家数N,要求排名的国家数M,国家号 ...

  7. 九度oj 题目1087:约数的个数

    题目链接:http://ac.jobdu.com/problem.php?pid=1087 题目描述: 输入n个整数,依次输出每个数的约数的个数 输入: 输入的第一行为N,即数组的个数(N<=1 ...

  8. 九度OJ题目1105:字符串的反码

    tips:scanf,cin输入字符串遇到空格就停止,所以想输入一行字符并保留最后的"\0"还是用gets()函数比较好,九度OJ真操蛋,true?没有这个关键字,还是用1吧,还是 ...

  9. 九度oj题目1009:二叉搜索树

    题目描述: 判断两序列是否为同一二叉搜索树序列 输入:                        开始一个数n,(1<=n<=20) 表示有n个需要判断,n= 0 的时候输入结束. 接 ...

  10. 九度oj题目1002:Grading

    //不是说C语言就是C++的子集么,为毛printf在九度OJ上不能通过编译,abs还不支持参数为整型的abs()重载 //C++比较正确的做法是#include<cmath.h>,cou ...

随机推荐

  1. 【tyvj】P1049 最长不下降子序列

    时间: 1000ms / 空间: 131072KiB / Java类名: Main 描述 求最长不下降子序列的长度 输入格式 第一行为n,表示n个数 第二行n个数 输出格式 最长不下降子序列的长度 测 ...

  2. lintcode :数组剔除元素后的乘积

    题目: 数组剔除元素后的乘积 给定一个整数数组A. 定义B[i] = A[0] * ... * A[i-1] * A[i+1] * ... * A[n-1], 计算B的时候请不要使用除法. 样例 给出 ...

  3. Hibernate中createCriteria即QBC查询的详细用法 .Hibernate中createCriteria即QBC查询的详细用法 .

    现在假设有一个Student类,内有id,name,age属性String hql = "from Student s";按照以前的做法,我们通常是Query query = se ...

  4. ServletRequest中getReader()和getInputStream()只能调用一次的解决办法

    转载:http://blog.sina.com.cn/s/blog_870cd7b90101fg58.html 最近使用spring mvc做项目,数据格式是json,有一个功能是实现记录请求的参数, ...

  5. iOS开发--数组

    1.sortedArrayUsingSelector (按Key值大小对NSDictionary排序) NSMutableArray *array = [NSMutableArray arrayWit ...

  6. 8、双向一对多的关联关系(等同于双向多对一。1的一方有对n的一方的集合的引用,同时n的一方有对1的一方的引用)

    双向一对多关联关系 “双向一对多关联关系”等同于“双向多对一关联关系”:1的一方有对n的一方的集合的引用,同时n的一方有对1的一方的引用. 还是用客户Customer和订单Order来解释: “一对多 ...

  7. javascript正则表达式控制input只能输入数字

    不能输入中文 <input type="text" name="textfield"  onkeyup="this.value=this.val ...

  8. ubuntu下安装Ming的教程

    Ming是一个操纵swf(flash movice)的C库,支持php. ruby. python等语言. 重要提示: 在安装Ming之前,应该准备好你的系统,特别是Linux/Unix系统,如果你对 ...

  9. maven小项目注册服务(一)--email和persist模块

    跟着书里的讲解,跟着做了一遍该项目: 首先明白注册账户的需求: 账号的lD和Email地址都可以用来唯一地标识某个用户,而显示名称则用来显示在页面下,方便浏览.注册的时候用户还需要输入两次密码,以确保 ...

  10. Tomcat原理 分类: 原理 2015-06-28 19:26 5人阅读 评论(0) 收藏

    Tomcat的模块结构设计的相当好,而且其Web 容器的性能相当出色.JBoss直接就使用了Tomcat的web容器,WebLogic的早期版本也是使用了Tomcat的代码. Web容器的工作过程在下 ...