题解 \(by\;zj\varphi\)

一道很妙的 \(dp\) 题,方程状态不好设置,细节也不少

看到数据范围,直接想离散化

设 \(f_{i,j}\) 表示处理完前 \(i\) 个水晶,其中摧毁的 \(A_i\) 的最小值为 \(j\) 时最多能摧毁多少。

分类讨论

  1. 当 \(A_i\le B_i\) 时,那么要使 \(A_i\) 为最小值,那么就要摧毁它,所以 \(dp_{i,A_i}=\max(dp_{i-1,B_i+1},...dp_{i-1,\max})+1\)

    但前 \(i-1\) 个数也可能有 \(A_i\) 所以也可以不摧毁它,故 \(dp_{i,A_i}=\max(dp_{i,A_i},dp_{i-1,A_i})\)

  2. 当 \(A_i>B_i\) 时,要使 \(A_i\) 为最小值,则 \(dp_{i,A_i}=\max(dp_{i-1,A_i+1},...dp_{i-1,\max})+1\),

    且 \(j\in (B_i,A_i]\) 时,\(dp_{i,j}=dp_{i-1,j}+1\)。

对于第二种情况,我们要把其中的 \(dp_{i,A_i}\) 取最大值,至于为什么可以在不确定前 \(i\) 个数是否有 \(j\) 时就转移 \(j\)

是因为若没有 \(j\),\(dp_{i,A_i}\) 绝对不会劣于 \(dp_{i,j}\),所以 \(dp_{i,j}\) 的会被 \(dp_{i,A_i}\) 覆盖。

数据范围太大,无法直接转移,考虑线段树优化,转移的式子可以直接当作 单点修改,区间最大值,区间加法。

Code
#include<bits/stdc++.h>
#define ri register signed
#define p(i) ++i
using namespace std;
namespace IO{
char buf[1<<21],*p1=buf,*p2=buf;
#define gc() p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++
template<typename T>inline void read(T &x) {
ri f=1;x=0;register char ch=gc();
while(ch<'0'||ch>'9') {if (ch=='-') f=0;ch=gc();}
while(ch>='0'&&ch<='9') {x=(x<<1)+(x<<3)+(ch^48);ch=gc();}
x=f?x:-x;
}
}
using IO::read;
namespace nanfeng{
#define cmax(x,y) ((x)>(y)?(x):(y))
#define cmin(x,y) ((x)>(y)?(y):(x))
#define FI FILE *IN
#define FO FILE *OUT
static const int N=1e5+7;
int a[N],b[N],tmpab[N<<1],n,ans,cnt;
struct Seg{
#define ls(x) (x<<1)
#define rs(x) (x<<1|1)
#define up(x) T[x].nm=cmax(T[ls(x)].nm,T[rs(x)].nm)
struct segmenttree{int lz,nm;}T[N<<3];
inline void down(int x) {
if (!T[x].lz) return;
int l=ls(x),r=rs(x);
T[l].nm+=T[x].lz,T[l].lz+=T[x].lz;
T[r].nm+=T[x].lz,T[r].lz+=T[x].lz;
T[x].lz=0;
}
void update(int x,int l,int r,int lt,int rt) {
if (l<=lt&&rt<=r) {p(T[x].nm),p(T[x].lz);return;}
int mid(lt+rt>>1);
down(x);
if (l<=mid) update(ls(x),l,r,lt,mid);
if (r>mid) update(rs(x),l,r,mid+1,rt);
up(x);
}
void updates(int x,int p,int k,int l,int r) {
if (l==r) {T[x].nm=cmax(T[x].nm,k);return;}
int mid(l+r>>1);
down(x);
if (p<=mid) updates(ls(x),p,k,l,mid);
else updates(rs(x),p,k,mid+1,r);
up(x);
}
int query(int x,int l,int r,int lt,int rt) {
if (l<=lt&&rt<=r) return T[x].nm;
int mid(lt+rt>>1),res(0);
down(x);
if (l<=mid) res=max(res,query(ls(x),l,r,lt,mid));
if (r>mid) res=max(res,query(rs(x),l,r,mid+1,rt));
return res;
}
}T;
inline int main() {
// FI=freopen("nanfeng.in","r",stdin);
// FO=freopen("nanfeng.out","w",stdout);
read(n);
for (ri i(1);i<=n;p(i)) {
read(a[i]),read(b[i]);
tmpab[p(cnt)]=a[i],tmpab[p(cnt)]=b[i];
}
sort(tmpab+1,tmpab+cnt+1);
ri kab=unique(tmpab+1,tmpab+cnt+1)-tmpab;
for (ri i(1);i<=n;p(i)) {
a[i]=lower_bound(tmpab+1,tmpab+kab,a[i])-tmpab;
b[i]=lower_bound(tmpab+1,tmpab+kab,b[i])-tmpab;
}
kab-=1;
for (ri i(1);i<=n;p(i)) {
int tmp=0;
if (a[i]<=b[i]) {
if (b[i]<kab) tmp=T.query(1,b[i]+1,kab,1,kab);
T.updates(1,a[i],tmp+1,1,kab); } else {
T.update(1,b[i]+1,a[i],1,kab);
if (a[i]<kab) tmp=T.query(1,a[i]+1,kab,1,kab);
T.updates(1,a[i],tmp+1,1,kab);
}
}
printf("%d\n",T.T[1].nm);
return 0;
}
}
int main() {return nanfeng::main();}

NOIP 模拟 $14\; \text{队长快跑}$的更多相关文章

  1. NOIP模拟14「队长快跑·影魔·抛硬币」

    T1:队长快跑 基本思路:   离散化·DP·数据结构优化DP   这三个我都没想到....气死.   定义状态数组:\(c[i][j]\)表示在i时最小的a值是j时可以摧毁的最多的水晶数.   那么 ...

  2. BZOJ4829: [Hnoi2017]队长快跑

    BZOJ4829: [Hnoi2017]队长快跑 Description 众所周知,在P国外不远处盘踞着巨龙大Y. 传说中,在远古时代,巨龙大Y将P国的镇国之宝窃走并藏在了其巢穴中,这吸引着整个P国的 ...

  3. [LOJ 2022]「AHOI / HNOI2017」队长快跑

    [LOJ 2022]「AHOI / HNOI2017」队长快跑 链接 链接 题解 不难看出,除了影响到起点和终点的射线以外,射线的角度没有意义,因为如果一定要从该射线的射出一侧过去,必然会撞到射线 因 ...

  4. NOIP 模拟 $14\; \text{抛硬币}$

    题解 \(by\;\;zj\varphi\) 签到题,自己看题解 Code #include<bits/stdc++.h> #define ri register signed #defi ...

  5. NOIP 模拟 $14\; \text{影魔}$

    题解 \(by\;\;zj\varphi\) 不是原题 一道(对我来说)很需要技巧的题 对于颜色数如何处理 离线,将子树转化为 \(dfs\) 序,但这种做法无法处理深度 我们按照深度加点(可以通过 ...

  6. Noip模拟14 2021.7.13

    T1 队长快跑 本身dp就不强的小马看到这题并未反映过来是个dp(可能是跟题面太过于像那个黑题的队长快跑相似) 总之,基础dp也没搞出来,不过这题倒是启发了小马以后考试要往dp哪里想想 $dp_{i, ...

  7. HZOI20190908模拟40 队长快跑,影魔,抛硬币 题解

    题面:https://www.cnblogs.com/Juve/articles/11487699.html 队长快跑: 权值线段树与dp yy的不错 #include<iostream> ...

  8. ServiceStack.Text 更快的序列化

    Json.net 是以前最经常用的序列化组件,后来又注意到ServiceStack号称最快的,所以我做了以下测试 1)Json.net using System; using System.Colle ...

  9. NOIP模拟 1

    NOIP模拟1,到现在时间已经比较长了.. 那天是6.14,今天7.18了 //然鹅我看着最前边缺失的模拟1,还是终于忍不住把它补上,为了保持顺序2345重新发布了一遍.. #   用  户  名   ...

随机推荐

  1. 二进制方式搭建Kubernetes集群

    环境准备: 演练暂时用单节点一台master和一台node节点来进行部署搭建(kubernetes 1.19版本) 角色 IP 组件 master 10.129.246.114 kube-apiser ...

  2. 如何搭建一个简易的 Web Terminal(一)

    前言 在介绍本篇文章的时候,先说一下本篇文章的一些背景.笔者是基于公司的基础建设哆啦 A 梦(Doraemon)一些功能背景写的这篇文章,不了解.有兴趣的同学可以去 袋鼠云 的 github 下面了解 ...

  3. C++ 标准模板库(STL)——容器(Containers)的用法及理解

    C++ 标准模板库(STL)中定义了通用的模板类和函数,这些模板类和函数可以实现多种流行和常用的算法和数据结构,如向量(vector).队列(queue).栈(stack).set.map等.这次主要 ...

  4. IO流 connect reset

    目录 出现场景 解决思路 出现场景 通过外部OBS下载10文件,然后通过工具将这10个文件打包成一个文件A.zip上传,最后将这个A.zip下载并解压,解压A.zip后发现文件数量不是10个. 解决思 ...

  5. python + mysql 实现表更新数据

    实例如下: import pymysqldef Update_Set(): #打开数据库链接 db = pymysql.connect("localhost","root ...

  6. Leetcode2.两数相加——简洁易懂

    > 简洁易懂讲清原理,讲不清你来打我~ 输入两个链表,相同位置相加,进位给下一个位置,输出相加后的链表![在这里插入图片描述](https://img-blog.csdnimg.cn/f43b7 ...

  7. SQL慢查询排查思路

    前言 平时在工作中每天都会做巡检,将前一天所有超过500ms的慢SQL排查出来 查找原因,是否能进行优化.慢慢中,在形成了一套思路方法论. 我个人认为对于排查慢SQL还是有一定的帮助 (一).是否是S ...

  8. Receiver class com.mchange.v2.c3p0.impl.NewProxyResultSet does not define or inherit an implementation of the resolved method 'abstract boolean isClosed()' of interface java.sql.ResultSet.

    背景: Mayabtis+springboot项目,连接数据库发生异常 报错内容: java.lang.AbstractMethodError: Receiver class com.mchange. ...

  9. ODOO里视图开发案例---定义一个像tree、form一样的视图

    odoo里视图模型MVC模式: 例子:在原来的视图上修改他: var CustomRenderer = KanbanRenderer.extend({ ....});var CustomRendere ...

  10. 第一篇 -- 安装和配置PyQt5

    我的电脑环境是:Win10 + Python3.6.4 + JetBrains PyCharm 2017.3.2 x64 之前用tkinter写界面,现在学习如何用PyQt5写界面. 安装PyQt5: ...