题目来自FZU2163 多米诺骨牌

Time Limit: 1000 mSec    Memory Limit : 32768 KB

 Problem Description

Vasya很喜欢排多米诺骨牌。他已经厌倦了普通的多米诺骨牌,所以他用不同高度的多米诺骨牌。他从左边到右边,把n个多米诺骨牌沿一个轴放在桌子上。每一个多米诺骨牌垂直于该轴,使该轴穿过其底部的中心。第i个多米诺骨牌具有坐标xi与高度hi。现在Vasya想要知道,对于每一个多米诺骨牌如果他推倒的话,右侧会有多少个多米诺骨牌也会倒下。

想想看,一个多米诺倒下,如果它严格的触动右侧的多米诺骨牌,被触碰的也会倒下。换句话说,如果多米诺骨牌(初始坐标x和高度h)倒下,会导致所有在[ X + 1,x + H - 1]范围内的多米诺骨牌倒下。

 Input

输入有多组测试数据,处理到文件结尾。

每组测试数据第一行包含整数n(1≤N≤10^5),这是多米诺骨牌的数量。然后n行,每行包含两个整数xi与hi(-10^8≤xi≤10^8 ,2 ≤hi≤108),xi表示多米诺骨牌的坐标和hi表示多米诺骨牌的高度。没有两个多米诺骨牌在同一个坐标点上。

 Output

对于每组数据输出一行,包含n个空格分隔的数Zi - 表示倒下的多米诺骨牌数量,如果Vasya推第i个多米诺骨牌(包括多米诺骨牌本身)。

 Sample Input

4
16 5
20 5
10 10 18 2
3
6 7
2 9
-6 10

 Sample Output

3 1 4 1
1 2 3

第一次比赛中出现中文题目,真心觉得高兴啊,无需百度翻译了……

这题目在比赛的时候我并没有思路,最后也没有做这题,赛后才思考这到题,一开始的想法就是动态规划,觉得是在可接触范围内的

从后面开始,dp[i]=max{1+dp[x]),x为i~i+h-1中有多米骨牌的坐标。

但是这样一想,dp开的数组太大了(10^8),且这 dp效率明显不高啊!因为这个范围内着到有坐标,明显会超时。

后来换一种思维来看,我们先按照横坐标排序,并重新编号,能推倒的多米诺牌的最后一个,肯定是不能推倒的那个的前一个,所以找到不能推倒的,再回溯回去。

这样很自然,就会使用栈结构了,我们用排序后的第一张牌压入栈,我们按新编号第二张牌开始去遍历,当前牌是栈顶部元素无法接触的时候,那么栈顶部能推倒的是这张牌的新序号减上栈顶部的新序号,并且很有可能栈中其他元素能触碰到这张牌(也按照刚刚的判断方法,是否能触碰到当前牌,能则继续pop)。当能触碰到,则继续压入栈。

即是:只在可以确定到不能触碰的时候才看开始考虑之前的牌!

所以,我们要引入一张无法触碰的,x坐标在无穷远(1<<30),这样,每个牌肯定会找到无法触碰的牌(最坏情况就是找到引入的牌)

我写了两个代码,一个是以类的形式来写,另一个是直接用数组,两者来说,第一个比较好理解。

类的形式:

#include<iostream>
#include<cstdio>
#include<vector>
#include<stack>
#include<cstring>
#include<algorithm>
using namespace std;
int n;
class GP{
public:
int id;
int s_id; // sort id
int x; //横坐标
int h; //高度 GP(){}; bool operator <(const GP &g)const{ return this->x<g.x;
}
void operator =(const GP &g){
this->id=g.id;
this->s_id=g.s_id;
this->x=g.x;
this->h=g.h;
} };
vector<GP> v; //骨牌
stack<GP> stk; //栈结构
int ans[100005];//结果数组
int main(){ int i; while(scanf("%d",&n)!=EOF){
v.clear();
while(!stk.empty()){stk.pop();}
v.resize(n+1);
for(i=0;i<n;i++){
v[i].id=i;
scanf("%d%d",&v[i].x,&v[i].h); }
v[n].x=200000001; //无穷远
v[n].h=0;
v[n].s_id=n;
sort(v.begin(),v.end());
for(i=0;i<n;i++){
v[i].s_id=i; }
stk.push(v[0]);
GP tmp;
for(i=1;i<=n;i++){ while(!stk.empty()&&stk.top().x+stk.top().h-1<v[i].x){ //直到无法触及的多米诺
tmp=stk.top();
stk.pop();
// cout<<tmp.x<<" "<<tmp.s_id<<" "<<v[i].s_id<<endl;
ans[tmp.id]=v[i].s_id-tmp.s_id; }
stk.push(v[i]); } for(i=0;i<n-1;i++){ printf("%d ",ans[i]);
}
printf("%d\n",ans[n-1]); }
return 0; }

数组形式:

#include<iostream>
#include<stack>
#include<algorithm>
#include<cstdio>
using namespace std;
#define N 100005
int id[N],x[N],h[N],ans[N];
bool CMP (int a,int b){ return x[a]<x[b]; }
stack<int> stk;
int main(){
int n,i; while(scanf("%d",&n)!=EOF){ while(!stk.empty()) stk.pop();
for(i=0;i<n;i++){ scanf("%d%d",x+i,h+i);
id[i]=i; }
x[n]=1<<30;
id[n]=n; sort(id,id+n,CMP); stk.push(0);
for(i=1;i<=n;i++){ while(!stk.empty()&&x[id[stk.top()]]+h[id[stk.top()]]-1<x[id[i]]){ //无法触碰的多米诺
int j=stk.top();
stk.pop();
ans[id[j]]=i-j; } stk.push(i); }
for(i=0;i<n-1;i++){
printf("%d ",ans[i]); }
printf("%d\n",ans[n-1]); } return 0;
}

省选训练赛第4场D题(多米诺骨牌)的更多相关文章

  1. HDU 4902 Nice boat 2014杭电多校训练赛第四场F题(线段树区间更新)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4902 解题报告:输入一个序列,然后有q次操作,操作有两种,第一种是把区间 (l,r) 变成x,第二种是 ...

  2. 训练赛第三场A题 zoj 559

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2559 解题报告:比赛的时候的想法是可以确定至少有两对相切的圆,所 ...

  3. 计蒜客 ACM竞赛高校联盟训练赛 第8场 煎牛排

    水一水. https://nanti.jisuanke.com/t/24205 煎牛排 题目描述 又是一个难得的周六,是时候远离食堂和外卖出去大吃一顿了.圈内知名吃货AA正想着中午去吃汉堡炸鸡烤肉火锅 ...

  4. 2018牛客网暑假ACM多校训练赛(第二场)E tree 动态规划

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round2-E.html 题目传送门 - 2018牛客多校赛第二场 E ...

  5. NOI2019省选模拟赛 第五场

    爆炸了QAQ 传送门 \(A\) \(Mas\)的童年 这题我怎么感觉好像做过--我记得那个时候还因为没有取\(min\)结果\(100\to 0\)-- 因为是个异或我们肯定得按位考虑贡献了 把\( ...

  6. NOI2019省选模拟赛 第六场

    传送门 又炸了-- \(A\) 唐时月夜 不知道改了什么东西之后就\(A\)掉了\(.jpg\) 首先,题目保证"如果一片子水域曾经被操作过,那么在之后的施法中,这片子水域也一定会被操作&q ...

  7. 2019牛客训练赛第七场 C Governing sand 权值线段树+贪心

    Governing sand 题意 森林里有m种树木,每种树木有一定高度,并且砍掉他要消耗一定的代价,问消耗最少多少代价可以使得森林中最高的树木大于所有树的一半 分析 复杂度分析:n 1e5种树木,并 ...

  8. HDU 4941 Magical Forest(map映射+二分查找)杭电多校训练赛第七场1007

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4941 解题报告:给你一个n*m的矩阵,矩阵的一些方格中有水果,每个水果有一个能量值,现在有三种操作,第 ...

  9. Contest1593 - 2018-2019赛季多校联合新生训练赛第三场(部分题解)

    H 10255 自然数无序拆分 H 传送门 题干: 题目描述 美羊羊给喜羊羊和沸羊羊出了一道难题,说谁能先做出来,我就奖励给他我自己做的一样礼物.沸羊羊这下可乐了,于是马上答应立刻做出来,喜羊羊见状, ...

随机推荐

  1. SqlLite ---.net连接数据库

    初识SqlLite ---.net连接数据库   SqlLite以小巧和嵌入式闻名,以前只是听说,现在终于忍不住要尝试下. 先下载ADO.NET2.0 Provider for SQLite,下载完后 ...

  2. WinFrom 安装包制作

    1.添加安装向导项目打开文件系统界面,选择应用程序文件夹.在右侧右击->添加->文件,把程序需要的文件都添加进来. 2.右击程序集->创建快捷方式.右击快捷方式->属性窗口-& ...

  3. 博客转移到lyso.me

    博客转移到http://lyso.me :)

  4. navicat for mysql (10038)如何解决,远程无法连接问题

    ubuntu server下安装了MySQL 5.5数据库,然后在windows下通过Navicat for MySQL连接时,出现 Can't connect to mysql server on ...

  5. Sublime Text 超好用的侧栏插件SideBarEnhancements

    SideBarEnhancements插件有效地改进了Sublime Text的侧边栏.安装插件后在侧边栏上点击右键,可以找到一下新功能:在资源管理器中打开.新建文件.新建文件夹.以…打开.在浏览器中 ...

  6. 一个方便的shell命令,查看软件安装目录

    查看软件安装路径:whereis phpfind / -name nginx.configfind 查找 / 从根目录 -name 文件查找

  7. Asp.Net Mvc5 之Controller

    经过前面介绍了路由系统之后,我们知道任何一个请求在经过asp.net url路由系统的拦截之后,会生成以controller/action 名称为核心的路由数据.asp.net mvc 根据此解析出目 ...

  8. cocos 3.0 一键打包android平台应该注意的细节

    cocos2d-x 移植越来越便捷,走到cocos2d-x-3.0rc2,能够说移植已经非常完好了,我们仅仅要进行适当的适配,cocos能够直接帮助我们生成apk 我网络不好无法上传图片:(无图无捷豹 ...

  9. android学习日记25--ANR和Hander消息机制

    1.ANR(Application Not Responding)定义 在Android上,如果你的应用程序有一段时间响应不够灵敏,系统会向用户显示一个对话框,这个对话框称作应用程序无响应(ANR:A ...

  10. [原创]javascript prototype 对象 函数 <精简的美丽......>

    精简的美丽...... javascript prototype 对象 函数 在javascript中我们都知道创建一个对象使用如下代码var x = {}对象可以拥有属性和方法var x = {   ...