题目

在平面上有 \(n\) 个点,现在有一个人要从某个点出发,

每次只能到达横纵坐标都超过原坐标的点,也就是 \(x_j<x_i,y_j<y_i\)

如果他要经过最多的点,那么哪些点是可能到达的,哪些点是必须到达的。


分析

按横坐标排序,实际上只需要管纵坐标,离散化之后直接用树状数组维护 \(dp[i]=dp[j]+1(y_j<y_i)\)

然后将序列翻转再做一遍,可能到达也就是 \(dp[i]+dp'[i]=ans-1\)

一定到达就是在可能到达的基础上,要保证 \(dp[i]\) 唯一,再统计一下 \(dp[i]\) 的个数即可。


代码

#include <cstdio>
#include <cctype>
#include <algorithm>
using namespace std;
const int N=100011;
struct rec{int x,y,rk;}a[N];
int c[N],n,ans[N],mx,f[N],g[N],m,b[N];
int iut(){
int ans=0,f=1; char c=getchar();
while (!isdigit(c)) f=(c=='-')?-f:f,c=getchar();
while (isdigit(c)) ans=ans*10+c-48,c=getchar();
return ans*f;
}
bool cmp(rec x,rec y){
return x.x<y.x||(x.x==y.x&&x.y<y.y);
}
int max(int a,int b){return a>b?a:b;}
void update(int x,int y){
for (;x<=m;x+=-x&x) c[x]=max(c[x],y);
}
int query(int x){
int ans=0;
for (;x;x-=-x&x) ans=max(ans,c[x]);
return ans;
}
int main(){
n=iut();
for (int i=1;i<=n;++i) a[i]=(rec){iut(),b[i]=iut(),i};
sort(a+1,a+1+n,cmp),sort(b+1,b+1+n),m=unique(b+1,b+1+n)-b-1;
for (int i=1;i<=n;++i) a[i].y=lower_bound(b+1,b+1+m,a[i].y)-b;
for (int l=1,r;l<=n;l=r+1){
for (r=l;r<=n&&a[r].x==a[l].x;++r); --r;
for (int i=l;i<=r;++i) f[i]=query(a[i].y-1)+1;
for (int i=l;i<=r;++i) update(a[i].y,f[i]);
}
for (int i=1;i<=m;++i) c[i]=0;
for (int r=n,l;r;r=l-1){
for (l=r;l&&a[l].x==a[r].x;--l); ++l;
for (int i=l;i<=r;++i) g[i]=query(m-a[i].y)+1;
for (int i=l;i<=r;++i) update(m-a[i].y+1,g[i]);
}
for (int i=1;i<=m;++i) c[i]=0;
for (int i=1;i<=n;++i) mx=max(mx,f[i]);
for (int i=1;i<=n;++i)
if (f[i]+g[i]-1==mx) ans[++ans[0]]=a[i].rk,++c[f[i]];
sort(ans+1,ans+1+ans[0]);
printf("%d ",ans[0]);
for (int i=1;i<=ans[0];++i) printf("%d ",ans[i]);
putchar(10),ans[0]=0;
for (int i=1;i<=n;++i)
if (f[i]+g[i]-1==mx&&c[f[i]]==1) ans[++ans[0]]=a[i].rk;
sort(ans+1,ans+1+ans[0]),printf("%d ",ans[0]);
for (int i=1;i<=ans[0];++i) printf("%d ",ans[i]);
return 0;
}

#树状数组,dp#SGU 521 North-East的更多相关文章

  1. codeforces 597C (树状数组+DP)

    题目链接:http://codeforces.com/contest/597/problem/C 思路:dp[i][j]表示长度为i,以j结尾的上升子序列,则有dp[i][j]= ∑dp[i-1][k ...

  2. hdu 4622 Reincarnation trie树+树状数组/dp

    题意:给你一个字符串和m个询问,问你l,r这个区间内出现过多少字串. 连接:http://acm.hdu.edu.cn/showproblem.php?pid=4622 网上也有用后缀数组搞得. 思路 ...

  3. Codeforces 597C. Subsequences (树状数组+dp)

    题目链接:http://codeforces.com/contest/597/problem/C 给你n和数(1~n各不同),问你长为k+1的上升自序列有多少. dp[i][j] 表示末尾数字为i 长 ...

  4. HDU2227Find the nondecreasing subsequences(树状数组+DP)

    题目大意就是说帮你给出一个序列a,让你求出它的非递减序列有多少个. 设dp[i]表示以a[i]结尾的非递减子序列的个数,由题意我们可以写出状态转移方程: dp[i] = sum{dp[j] | 1&l ...

  5. CodeForces - 314C Sereja and Subsequences (树状数组+dp)

    Sereja has a sequence that consists of n positive integers, a1, a2, ..., an. First Sereja took a pie ...

  6. HDU 6348 序列计数 (树状数组 + DP)

    序列计数 Time Limit: 4500/4000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)Total Subm ...

  7. [Codeforces261D]Maxim and Increasing Subsequence——树状数组+DP

    题目链接: Codeforces261D 题目大意:$k$次询问,每次给出一个长度为$n$的序列$b$及$b$中的最大值$maxb$,构造出序列$a$为$t$个序列$b$连接而成,求$a$的最长上升子 ...

  8. 【XSY2727】Remove Dilworth定理 堆 树状数组 DP

    题目描述 一个二维平面上有\(n\)个梯形,满足: 所有梯形的下底边在直线\(y=0\)上. 所有梯形的上底边在直线\(y=1\)上. 没有两个点的坐标相同. 你一次可以选择任意多个梯形,必须满足这些 ...

  9. hdu5489 树状数组+dp

    2015-10-06 21:49:54 这题说的是个给了一个数组,然后删除任意起点的一个连续的L个数,然后求最长递增子序列<是递增,不是非递减>,用一个树状数组维护一下就ok了 #incl ...

  10. hdu5125 树状数组+dp

     hdu5125 他说的是n个人每个人都有两个气球a,b,气球各自都有相应的体积,现在让他们按照序号排列好来,对他们的a气球体积值计算最长上升子序列,对于这整个排列来说有m次机会让你将a气球替换成b气 ...

随机推荐

  1. 【Python OO其一】构造函数__init__()

    Python对象包括三个部分:id(identity识别码).type(对象类型).value(对象的值) __ init __()构造函数 __ init __()方法应用定义构造函数,作用是在实例 ...

  2. 第一百一十三篇: JS数组Array(二)数组方法 栈、队列、排序

    好家伙,    在上一篇中,我们知道了, JS的数组中每个槽位可以存储任意类型的数据 那么,我们能通过数组去模仿某些数据结构吗? 答案是肯定的 1.栈方法 ECMAScript 给数组提供几个方法,让 ...

  3. mysql常用语句(持续更新)

    查询数据库中各表数量 select table_name,table_rows from information_schema.tables where TABLE_SCHEMA = 'miot' o ...

  4. 【Azure 环境】当本地网络通过ER专线与Azure云上多个虚拟网络打通,如何通过特定的网络策略来限制本地部分网段访问云上虚拟机22端口?

    问题描述 当本地网络通过ER专线与Azure云上多个虚拟网络打通,如何通过特定的网络策略来限制本地部分网段访问云上虚拟机22端口? 问题回答 根据文档调研,在ER线路服务的层面,是无法做网络策略来限制 ...

  5. Dockerfile编写(备份)

    1-使用#注释 2-Dockerfile 主体内容分为四部分:基础镜像信息. 维护者信息. 镜像操作指令和容器启动时执行指令 简单示例: FROM   xxx:latest   #基于xxx:late ...

  6. 在.Net中使用Java代码?

    前言 你没有看错,我确实在.Net6的项目中在编写java,我都using java了,算不算在写java那? using com.microsoft.sqlserver.jdbc; using ja ...

  7. Redis单线程为什么如此之快

    一.概述 Redis的高并发和快简单可以归结为一下几点: 1.Redis是基于内存的: 2.Redis是单线程的: 3.Redis使用多路复用技术. 4.高效的数据结构 但具体怎么做的呢,下面来详细看 ...

  8. python使用selenium适配谷歌浏览器插件, chromedriver与chrome各版本及下载地址

    python selenium使用,需要谷歌chromedriver.exe插件 chromedriver.exe插件是放在python的安装目录下(亲测,其它的都不对) 以下是chromedrive ...

  9. 用java实现书城项目(简单增删改查2)

    书城项目 登录 dao 接口:UserDao Users login(String username,String password); 实现:UserDaoImpl QueryRunner quer ...

  10. FreeRTOS教程2 任务管理

    1.准备材料 正点原子stm32f407探索者开发板V2.4 STM32CubeMX软件(Version 6.10.0) Keil µVision5 IDE(MDK-Arm) 野火DAP仿真器 XCO ...