Noip2011 提高组 选择客栈

P1311 选择客栈

直通

思路:

  ①看题,我们可以发现一个显然的性质,即当最左边的客栈向右移动时,最右边的客栈时单调向右的,并且右端点往右的客栈也符合要求。(因为只要左侧有一个满足的,右边的自然可以选)

  不妨将每种颜色的宾馆分别放到 vector 中。

  然后在每个 vector 中枚举左端点,维护一个单调指针来确定右端点 (vector中的下标)。

  我们接下来就要快速判断一段区间是否合法,我们开一个ok数组,表示从i点开始最近的满足条件的位置(不考虑颜色)。

    这样的话转移就是:

            ok[i]=i;(p[i]<=P),反之:ok[i]=ok[i+1];

            需要注意的是ok[n+1]=n+1;

            表示如果p[n]>P则p[n]=p[n+1]=n+1,即不合法

  ②大模拟,详细看代码

坑点:

  注意输入顺序qwq

上代码:

1)vector版

#include <iostream>
#include <cstdio>
#include <vector>
using namespace std;

const int K = 50;
const int N = 200001;
int n,k,p,ans;
int w[N],ok[N];
vector<int>v[K];

void getnxt() {
    ok[n+1]=n+1;
    for(int i=n; i>=1; i--) {
        if(w[i]<=p) ok[i]=i;
        else ok[i]=ok[i+1];
    }
}

int main() {
    scanf("%d%d%d",&n,&k,&p);
    for(int i=1,x; i<=n; i++) {
        scanf("%d%d",&x,&w[i]);
        v[x].push_back(i);
    }
    getnxt();
    for(int i=0,d,s; i<k; i++) {
        d=0;
        s=v[i].size(); //i客栈连接的s个客栈
        for(int j=0; j<s; j++) {
            d=max(d,j+1); //当前枚举到的最右端的客栈所在色调的编号是多少
            while(ok[v[i][j]]>v[i][d] && d<s) {
                d++;
            } //这里不太懂...
            ans+=s-d;
        }
    }
    printf("%d",ans);
    return 0;
}

2)纯模拟版

#include <iostream>
#include <cstdio>
using namespace std;

const int K = 101;
int n,k,p,ans;
int precnt[K],ok[K],last[K];
//precnt[x] i点前颜色为x的客栈数
//ok[x]     i点前满足能有价格<=p的客栈的颜色为x的客栈数
//last[x]   i点 前一个颜色 为x的点的编号

int main() {
    scanf("%d%d%d",&n,&k,&p);
    for(int i=1,tmp,x,y; i<=n; i++) {
        scanf("%d%d",&x,&y);
        if(y<=p) tmp=i; //记录下来当前色调的可行的最右客栈编号
        if(tmp>=last[x]) ok[x]=precnt[x];
        ans+=ok[x];
        precnt[x]++; //不管是不是可以满足条件
        last[x]=i;
    }
    printf("%d",ans);
    return 0;
}

时间: 2024-08-02 21:07:23

Noip2011 提高组 选择客栈的相关文章

NOIP2011提高组 选择客栈

原题 题目描述 丽江河边有n 家很有特色的客栈,客栈按照其位置顺序从 1 到n 编号.每家客栈都按照某一种色调进行装饰(总共 k 种,用整数 0 ~ k-1 表示),且每家客栈都设有一家咖啡店,每家咖啡店均有各自的最低消费. 两位游客一起去丽江旅游,他们喜欢相同的色调,又想尝试两个不同的客栈,因此决定分别住在色调相同的两家客栈中.晚上,他们打算选择一家咖啡店喝咖啡,要求咖啡店位于两人住的两家客栈之间(包括他们住的客栈),且咖啡店的最低消费不超过 p . 他们想知道总共有多少种选择住宿的方案,保证

洛谷P1003 [NOIP2011提高组Day1T1]铺地毯

P1003 铺地毯 题目描述 为了准备一个独特的颁奖典礼,组织者在会场的一片矩形区域(可看做是平面直角坐标系的第一象限)铺上一些矩形地毯.一共有 n 张地毯,编号从 1 到n .现在将这些地毯按照编号从小到大的顺序平行于坐标轴先后铺设,后铺的地毯覆盖在前面已经铺好的地毯之上. 地毯铺设完成后,组织者想知道覆盖地面某个点的最上面的那张地毯的编号.注意:在矩形地毯边界和四个顶点上的点也算被地毯覆盖. 输入输出格式 输入格式: 输入文件名为carpet.in . 输入共n+2 行. 第一行,一个整数n

[NOIP2011] 提高组 洛谷P1315 观光公交

题目描述 风景迷人的小城Y 市,拥有n 个美丽的景点.由于慕名而来的游客越来越多,Y 市特意安排了一辆观光公交车,为游客提供更便捷的交通服务.观光公交车在第 0 分钟出现在 1号景点,随后依次前往 2.3 .4 ……n 号景点.从第 i 号景点开到第 i+1 号景点需要 Di 分钟.任意时刻,公交车只能往前开,或在景点处等待. 设共有m 个游客,每位游客需要乘车1 次从一个景点到达另一个景点,第i 位游客在Ti 分钟来到景点 Ai ,希望乘车前往景点Bi (Ai<B i ).为了使所有乘客都能顺

刷题总结——mayan游戏(NOIP2011提高组day2T3)

题目: 题目背景 NOIP2011提高组 DAY1 试题. 题目描述 Mayan puzzle 是最近流行起来的一个游戏.游戏界面是一个 7 行 5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定的步数内消除所有的方块,消除方块的规则如下: 1.每步移动可以且仅可以沿横向(即向左或向右)拖动某一方块一格:当拖动这一方块时,如果拖动后到达的位置(以下称目标位置)也有方块,那么这两个方块将交换位置(参见输入输出样例说明中的图6 到图

Noip2011提高组总结

这套题思考的难度比较大,应该说是有四题基础题,一题比较复杂的搜索加模拟,还有一题需要深度思考一下.自己的代码漏洞还是很大,而且思考的时候会遗漏一些情况,这些错误都是致命的,去年Noip的惨败也证实了这一点,许多时候,我并没有败在算法上,而是细节与心态上.记住犯过的错误,尽力不在同一个地方摔倒,那么才可能不断进步,否则一直都在原地踏步. (未完) Noip2011提高组总结

洛谷P1311 [NOIP2011提高组Day1T2] 选择客栈

P1311 选择客栈 题目描述 丽江河边有n 家很有特色的客栈,客栈按照其位置顺序从 1 到n 编号.每家客栈都按照某一种色调进行装饰(总共 k 种,用整数 0 ~ k-1 表示),且每家客栈都设有一家咖啡店,每家咖啡店均有各自的最低消费. 两位游客一起去丽江旅游,他们喜欢相同的色调,又想尝试两个不同的客栈,因此决定分别住在色调相同的两家客栈中.晚上,他们打算选择一家咖啡店喝咖啡,要求咖啡店位于两人住的两家客栈之间(包括他们住的客栈),且咖啡店的最低消费不超过 p . 他们想知道总共有多少种选择

noip2011提高组day1+day2解题报告

Day1 T1铺地毯https://www.luogu.org/problem/show?pid=1003 [题目分析] 全部读入以后从最后一个往前找,找到一个矩形的范围覆盖了这个点,那这个矩形就是最上面的地毯,输出即可 #include<iostream> #include<cstdio> #include<cstring> using namespace std; const int maxn=10010; int a[maxn],b[maxn],g[maxn],k

NOIP2011提高组 聪明的质监员 -SilverN

题目描述 小T 是一名质量监督员,最近负责检验一批矿产的质量.这批矿产共有 n 个矿石,从 1到n 逐一编号,每个矿石都有自己的重量 wi 以及价值vi .检验矿产的流程是: 1 .给定m 个区间[Li,Ri]: 2 .选出一个参数 W: 3 .对于一个区间[Li,Ri],计算矿石在这个区间上的检验值Yi: 这批矿产的检验结果Y 为各个区间的检验值之和.即:Y1+Y2...+Ym 若这批矿产的检验结果与所给标准值S 相差太多,就需要再去检验另一批矿产.小T 不想费时间去检验另一批矿产,所以他想通

NOIP2011提高组(选择客栈)

题目链接:http://codevs.cn/problem/1135/ 题目大意:中文题...就不解释了 题目思路:看了其他巨巨的blog写的,dp思路 #include <iostream> #include <cstdio> #include <cstdlib> #include <cmath> #include <algorithm> #include <cstring> #include <stack> #incl