题目大意:

输入n,k ;n次操作 找到覆盖次数在k及以上的段的总长

一开始位置在0 左右活动范围为1-1000000000

接下来n行描述每次操作的步数和方向

Sample Input

6 2
2 R
6 L
1 R
8 L
1 R
2 R

Sample Output

6

下面的方法用了 map 和 区间表示法 http://www.cnblogs.com/zquzjx/p/8321466.html(区间表示法看这里)

但是不同于 题解 https://www.luogu.org/problemnew/solution/P2205 扫描线的方法可以得到整段覆盖次数

(蠢哭)存在一种特殊情况

若指令为      则会出现

———1

2——————

——3

3  2            位置      -6   -1    0   1   5    6

R  5                                    +1            -1

L  11                      +1                         -1

R  5                       +1          -1

最后 map内的值为

-6    2

-1    2

0   2

1   2

5   2

6   0

最后结果为11 而答案应该是10 错在-1到0这一段事实上只有一次覆盖

也就是该法实际上只能得知map中当前端点的被覆盖次数

所以将区间表示法

原本的 末尾端点的下一点(key+1) -1

改成 末尾端点的下半点(key+0.5) -1

#include <algorithm>
#include <iostream>
#include <stdlib.h>
#include <cstring>
#include <stdio.h>
#include <map>
using namespace std;
int main()
{
//freopen("23695all.in","r",stdin);
int n,w;
while(~scanf("%d%d",&n,&w))
{
map <double,int> m; m.clear();
int key=,ed=; m[key]=;
int a; char op;
while(n--)
{
scanf("%d %c",&a,&op);
if(op=='R')
{
m[key]++; key+=a;
m[key+0.5]--;
ed=max(ed,key);
}
else
{
m[key+0.5]--; key-=a;
m[key]++;
}
} map <double,int>::iterator it;
int sum=;
for(it=m.begin();it!=m.end();it++)
{
sum+=it->second;
m[it->first]=sum;
//printf("%.1f %d\n",it->first,it->second);
}
double le,rig,ans=;
for(it=m.begin();it!=m.end();it++)
{
while(it!=m.end()&&it->second<w) it++;
le=it->first;
if(it==m.end()) break;
while(it!=m.end()&&it->second>=w) it++;
rig=it->first; if(it==m.end()) rig=ed+0.5;
if(rig==(int)rig) ans+=rig--le;
else ans+=rig-0.5-le; ///这部分修改下即可 //printf("%.1f %.1f %.1f %d\n",rig,le,ans,ed);
if(it==m.end()) break;
}
printf("%.0f\n",ans);
} return ;
}

若是输出map中所有的值

会发现实际上在头尾之间的端点的覆盖次数会出错

当连续往同一个方向移动时

若指令为      则会出现

—1——2

2 1         位置  0  1  2  3  4

R 1               +1     -1

R 2                   +1         -1

最后 map内的值为

0  1

1  2

2  1

3  1

4  0

虽然这对本题没有什么影响

因为中间端点的覆盖次数虽不准确但绝对会大于头尾端点

且在下一点便会被减去多余的部分 而答案是加右-1-左 所以不会有影响

如上样例若是k=2 那么ans=2-1-1=0

但题目若是要求查看某点的被覆盖次数则会WA

所以我们可以判断一下当前指令与上轮指令是否相同

#include <algorithm>
#include <iostream>
#include <stdlib.h>
#include <cstring>
#include <stdio.h>
#include <map>
#define INF 0x3f3f3f3f
using namespace std;
int main()
{
//freopen("23695all.in","r",stdin);
int n,w;
while(~scanf("%d%d",&n,&w))
{
map <double,int> m; m.clear();
int key=,st=INF,ed=; m[key]=;
int a,cnt=; char op,ops='A';
while(n--)
{
scanf("%d %c",&a,&op);
if(op=='R')
{
if(op!=ops)
{ m[key]++; key+=a;
m[key+0.5]--;}
else
{ m[key+0.5]++; key+=a;
m[key+0.5]--;}
ed=max(ed,key);
}
else
{
if(op!=ops)
{ m[key+0.5]--; key-=a;
m[key]++;}
else
{ m[key]--; key-=a;
m[key]++;}
st=min(st,key);
}
ops=op;
} map <double,int>::iterator it;
int sum=;
for(it=m.begin();it!=m.end();it++)
{
sum+=it->second;
m[it->first]=sum;
//printf("%.1f %d\n",it->first,it->second);
}
double le,rig,ans=;
for(it=m.begin();it!=m.end();it++)
{
while(it!=m.end()&&it->second<w) it++;
le=it->first;
if(it==m.end()) break;
while(it!=m.end()&&it->second>=w) it++;
rig=it->first;
if(it==m.end()) rig=ed+0.5;
if(rig==(int)rig) ans+=rig--le;
else ans+=rig-0.5-le;
//printf("%.1f %.1f %.1f %d\n",rig,le,ans,ed);
if(it==m.end()) break;
}
printf("%.0f\n",ans);
} return ;
} ///

此时 当连续往同一个方向移动时

若指令为      则会出现

—1——2

2 1         位置  0  1   2  3  4

R 1               +1      -1

R 2                        +1     -1

最后 map内的值为

0  1

1  1

2  1

3  1

4  0

这样的话每个端点的覆盖次数就都是准确的了

USACO 2013 January Silver Painting the Fence /// oj23695的更多相关文章

  1. USACO翻译:USACO 2013 NOV Silver三题

    USACO 2013 NOV SILVER 一.题目概览 中文题目名称 未有的奶牛 拥挤的奶牛 弹簧牛 英文题目名称 nocow crowded pogocow 可执行文件名 nocow crowde ...

  2. USACO翻译:USACO 2013 DEC Silver三题

    USACO 2013 DEC SILVER 一.题目概览 中文题目名称 挤奶调度 农场航线 贝西洗牌 英文题目名称 msched vacation shuffle 可执行文件名 msched vaca ...

  3. USACO 2013 Nov Silver Pogo-Cow

    最近因为闲的蛋疼(停课了),所以开始做一些 USACO 的银组题.被完虐啊 TAT 貌似 Pogo-Cow 这题是 2013 Nov Silver 唯一一道可说的题目? Pogo-Cow Descri ...

  4. USACO 2008 January Silver Telephone Lines /// 二分最短路 邻接表dijkstra oj22924

    题目大意: 一共有N (1 ≤ N ≤ 1,000)个电线杆,有P P (1 ≤ P ≤ 10,000)对电线杆是可以连接的, 用几条线连接在一起的电线杆之间都可相互通信,现在想要使得电线杆1和电线杆 ...

  5. [luogu P2205] [USACO13JAN]画栅栏Painting the Fence

    [luogu P2205] [USACO13JAN]画栅栏Painting the Fence 题目描述 Farmer John has devised a brilliant method to p ...

  6. USACO翻译:USACO 2013 JAN三题(1)

    USACO 2013 JAN 一.题目概览 中文题目名称 镜子 栅栏油漆 奶牛排队 英文题目名称 mirrors paint lineup 可执行文件名 mirrors paint lineup 输入 ...

  7. USACO翻译:USACO 2014 DEC Silver三题

    USACO 2014 DEC SILVER 一.题目概览 中文题目名称 回程 马拉松 奶牛慢跑 英文题目名称 piggyback marathon cowjog 可执行文件名 piggyback ma ...

  8. USACO翻译:USACO 2012 FEB Silver三题

    USACO 2012 FEB SILVER 一.题目概览 中文题目名称 矩形草地 奶牛IDs 搬家 英文题目名称 planting cowids relocate 可执行文件名 planting co ...

  9. USACO翻译:USACO 2014 FEB SILVER 三题

    USACO 2014 FEB SILVER 一.题目概览 中文题目名称 自动打字 路障 神秘代码 英文题目名称 auto rblock scode 可执行文件名 auto rblock scode 输 ...

随机推荐

  1. Struts2配置详情

    Struts2的核心功能是action,对于开发人员来说,使用Struts2主要就是编写action,action类通常都要实现com.opensymphony.xwork2.Action接口,并实现 ...

  2. 2017-3-7html基础

    一:<html><!--html文档的开始--> <head><!--头标签,html文档的开头,来描述html文档的信息,head之间的内容不会在浏览器中显 ...

  3. 暑假集训test-8-31(am)

    1.字符串匹配 看到题目以为真是字符串题结果是数学题..70分做法很傻逼然而我更傻逼只有30... 正解是发现两个位置会匹配当且仅当mod gcd(lena,lenb)同余,在一个lcm(lena,l ...

  4. NX二次开发-UFUN获取系统相关信息UF_ask_system_info

    NX9+VS2012 #include <uf.h> UF_initialize(); UF_system_info_t Info; UF_ask_system_info(&Inf ...

  5. NX二次开发-UFUN CSYS坐标系转换UF_CSYS_map_point

    1 NX9+VS2012 2 3 #include <uf.h> 4 #include <uf_curve.h> 5 #include <uf_csys.h> 6 ...

  6. NX二次开发-UFUN获取NX系统默认导出CGM的选项设置UF_CGM_ask_default_export_options

    文章转载自唐康林NX二次开发论坛,原文出处: http://www.nxopen.cn/thread-126-1-1.html 刚才有同学问到这个问题,如果是用NXOpen来做,直接录制一下就可以了: ...

  7. 大数据-KNN算法

    KNN是通过测量不同特征值之间的距离进行分类.它的思路是:如果一个样本在特征空间中的k个最相似(即特征空间中最邻近)的样本中的大多数属于某一个类别,则该样本也属于这个类别,其中K通常是不大于20的整数 ...

  8. scrapy爬取cnblogs文章列表

    scrapy爬取cnblogs文章 目标任务 安装爬虫 创建爬虫 编写 items.py 编写 spiders/cnblogs.py 编写 pipelines.py 编写 settings.py 运行 ...

  9. 全面解读PHP面试

    php面试考察点    1.PHP基础知识 引用变量 常量及数据类型 运算符及流程控制 自定义函数及内部函数 正则表达式 文件及目录处理 会话控制 面向对象 网络协议 开发环境相关考点 2.JavaS ...

  10. 剑指offer——07用两个栈实现队列

    题目描述 用两个栈来实现一个队列,完成队列的Push和Pop操作. 队列中的元素为int类型.   题解: 有两个栈,stack1和stack2 push只在stack1上操作,pop只在stack2 ...