题目来自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. 按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n); (2)编写一个类:ClassA来实现接口InterfaceA,实现int method(int n)接口方 法时,要求计算1到n的和; (3)编写另一个类:ClassB来实现接口InterfaceA,实现int method(int n)接口 方法时,要求计算n的阶乘(n!);

    package com.homework2; public class ClassA implements InterfaceA { @Override public int method(int n ...

  2. linux tomcat自启动设置

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  3. 单点登录系统CAS筹建及取得更多用户信息的实现

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  4. android常见错误-The container 'Android Dependencies' references non existing library

    The container 'Android Dependencies' references non existing library

  5. C++容器类对象函数參数问题

    总之中的一个句话:容器类对象作为函数參数,与整数类型作为函数參数的传递特性同样. 验证程序 #include "stdafx.h" #include <iostream> ...

  6. htm explorer

    链接:https://github.com/450640526/HtmExplorer   最低环境 系统安装了.NET 4.0                地址:http://www.baidu. ...

  7. 返回类型和return语句

    return语句终止当前正在执行的函数并将控制权返回到调用该函数的地方.return语句有两种形式: return; return expression; 无返回值函数 没有返回值的return语句只 ...

  8. Linux下各种常见环境变量的配置

      Linux系统下各种环境变量都通过修改/etc/profile文件来实现.由于是系统文件,修改此文件需要root权限.因此实现以下功能都需要用户拥有root权限. 另:不要轻易修改profile文 ...

  9. ping与telnet的区别

    ping 查看某个IP地址是否有效.还可以得出解析IP..评估网络质量.telnet 查看可以PING通IP的机子上的某个端口是否可以进行访问(telnet IP port) ,如果连接失败,可能是防 ...

  10. typeid关键字

    这么看下去太要命了,有太多东西要学了... 而且视频看起来的确费神,费脑,费耳朵. 所以决定由视频驱动转向代码驱动.主攻vs,c++然后先把界面东西做出来,然后在想后面的东西. 所以今天 [先上来看了 ...