题目

题目大意

有一个\(01\)序列。给你一堆区间,每个区间中有且仅有一个\(1\)点。

问最多的\(1\)点个数。


思考历程

感觉这题特别经典,似乎在哪里见过,又好像没有见过。

一开始朝贪心方面想……想不出来……

后来想DP,还是想不出来……

直到WHH大爷跑过来兴奋地说:不就是个差分约束吗!

于是我心态崩了,去搜题解……

然而搜到的全是DP……

理解了一下DP的方法,虽然理解,但是极度不熟练……

可是这题对代码能力的考验又很强……我本想自己打出极其恶心的方法,但后来还是仔细看了看题解(还有)标程。

几乎是对着标打出来了……

心态崩了……


正解

先说DP。

设\(f_i\)表示选\(i\)的最多的\(1\)点数目。

方程为\(f_i=\max f_j +1\)

现在问题是\(j\)的范围怎么求。

这就很考验人的分析能力了——从两个方面考虑:

  1. 唯一性:所有包括\(i\)的区间内都不能再选别的。这样可以求出右边界\(R_i\),为包含\(i\)的区间的最小左边界\(-1\)
  2. 必要性:所有不包括\(i\)的区间(\(i\)之前的)中都必须要有一个选。这样可以求出右边界\(L_i\),为不包含\(i\)的区间的最大左边界

这样,求出\(L_i\)和\(R_i\),就可以转移了。

如果不刻意追求代码的完美,其实这个时候用线段树来辅助转移就好了,简单粗暴(如果是比赛,我肯定就这么打了)。

但是实际上可以优化。

首先可以证明出一个结论:\(L\)和\(R\)都是递增的。

如果\(L_{i-1}>L_i\),由于包含了\(i\)的区间也会包含\(i-1\)(除非区间左边界为\(i\),但显然这种情况下\(L{i-1}\leq L_i\)),\(L_{i-1}\)为包含\(i-1\)区间的最小左边界\(-1\),所以\(L_{i-1}\)可以被更新,所以不成立。

如果\(R_{i-1}>R_i\),由于不包含\(i-1\)的区间也不包含\(i\),\(R_i\)为不包含\(i\)区间的最大右边界,所以\(R_i\)可以被更新,所以不成立。

既然都是递增的,那就可以很愉快地单调队列DP了……

有了各种单调的性质,代码也可以简单很多,也不需要打线段树或堆了……

然而……如果是在比赛,想这些的时间还不如用来打线段树呢……

还有一种方法是差分约束。

假如我们做一遍前缀和,对于区间\([l,r]\),就会有\(s_l+1=s_r\),相当于\(s_l+1\geq s_r\)且\(s_r-1\geq s_l\)。还有,将每个左右端点排个序记为\(x_i\),那么还有\(x_{i-1}\leq x_i\)

差分约束即可……当然我没有打过,所以纯属瞎哔哔……


代码

using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 200010
int n,m;
struct Section{
int l,r;
} s[N];
inline bool cmps(const Section &a,const Section &b){
return a.l<b.l || a.l==b.l && a.r<b.r;
}
int L[N],R[N];
int q[N],head,tail;
int f[N];
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n+1;++i)
R[i]=i-1;
for (int i=1;i<=m;++i){
scanf("%d%d",&s[i].l,&s[i].r);
L[s[i].r+1]=max(L[s[i].r+1],s[i].l);
R[s[i].r]=min(R[s[i].r],s[i].l-1);
}
for (int i=n;i>=1;--i)
R[i]=min(R[i],R[i+1]);
for (int i=2;i<=n+1;++i)
L[i]=max(L[i],L[i-1]);
memset(f,127,sizeof f);
f[0]=0;
for (int i=1,j=0;i<=n+1;++i){
for (;j<=R[i];++j){
if (f[j]==0x7f7f7f7f)
continue;
while (head<=tail && f[q[tail]]<f[j])
tail--;
q[++tail]=j;
}
while (head<=tail && q[head]<L[i])
head++;
if (head<=tail)
f[i]=f[q[head]]+1;
}
if (f[n+1]==0x7f7f7f7f)
printf("-1\n");
else
printf("%d\n",f[n+1]-1);
return 0;
}

总结

真是一道单调队列的好题。

当然,如果是比赛,我肯定会选择打线段树的。

[JZOJ3233] 照片的更多相关文章

  1. Photoshop将普通照片快速制作二次元漫画风格效果

    今天为大家分享Photoshop将普通照片快速制作二次元漫画风格效果,教程很不错,对于喜欢漫画的朋友可以参考本文,希望能对大家有所帮助! 一提到日本动画电影,大家第一印象肯定是宫崎骏,但是日本除了宫崎 ...

  2. mono for android 获取手机照片或拍照并裁剪保存

    axml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android ...

  3. 计算照片的面积(WPF篇)

    昨天,老周突发其想地给大伙伴们说了一下UWP应用中计算照片面积的玩法,而且老周也表示会提供WPF版本的示例.所以,今天就给大伙们补上吧. WPF是集成在.net框架中,属于.net的一部分,千万不要跟 ...

  4. 计算照片的面积(UWP篇)

    今天先说UWP应用程序上计算照片面积的方法,改天有空,再说说WPF篇. 其实计算照片面积的原理真TMD简单,只要你有本事读到照片的像素高度和宽度,以及水平/垂直方向上的分辨率(DPI)就可以了.计算方 ...

  5. 部分Android手机拍照后照片被旋转的解决方案

      在部分Android手机(如MT788.Note2)上,使用Camera拍照以后,得到的照片会被自动旋转(90°.180°.270°),这个情况很不符合预期.仔细分析了一下,因为照片属性中是存储了 ...

  6. Win10系统怎样让打开图片方式为照片查看器

    转载自:百度经验 http://jingyan.baidu.com/article/5d368d1ef0cad13f60c057e3.html 1.首先,我们需要使用注册表编辑器来开启Win10系统照 ...

  7. Smart3D系列教程4之 《案例实战演练1——小物件的照片三维重建》

    一.前言 Wish3D出品的Smart3D系列教程已经推出3讲了,分别是关于倾斜摄影三维建模原理应用.照片采集技巧.Smart3D各个功能模块的作用,它们都是围绕Smart3D建模软件进行的讲解.那么 ...

  8. Smart3D系列教程3之 《论照片三维重建中Smart3D几个工作模块的功能意义》

    [摘要] 近年来,倾斜摄影测量技术是国际测绘遥感领域近年发展起来的一项高新技术,利用照片进行三维重建成为一项关键性的技术.Smart3D软件,是照片三维重建主流软件之一,本文将就Smart3D建模软件 ...

  9. Hawk: 无编程抓取淘女郎的所有高清照片

    1.这是什么鬼? 哦?美女? 最近看了这一篇文章:http://cuiqingcai.com/1001.html 大概说的是用Python和Pyspider(这货好像是我的一位师兄写的,吓尿),抓取淘 ...

随机推荐

  1. 如何删除github里的项目

    1.登录github网站后,就会看到如下画面,在头像下面有个你的项目资源这一栏,假如我要删除名为“hhh”的项目,现在鼠标点进这个项目里面 2.进去后点setting 点进去后,默认是选择了OPtio ...

  2. python 中 random模块的用法

    import random print( random.randint(1,10) ) # 产生 1 到 10 的一个整数型随机数 print( random.random() ) # 产生 0 到 ...

  3. ForEach控制器

    ForEach一般是和用户定义的变量结合使用的 前面前缀一般是和用户定义的变量前缀一致 输出的变量是要使用的变量 这样能保证每次使用的变量不一样 add"_"before numb ...

  4. troff - groff 文档排版系统的 troff 处理器

    总览 SYNOPSIS troff [ -abcivzCERU ] [ -d cs ] [ -f fam ] [ -F dir ] [ -m name ] [ -M dir ] [ -n num ] ...

  5. Center OS 7安装 Apollo

    声明: 每个人的情况都不一样,所以大家在看教程的时候自行斟酌,最好先扫一遍,再来根据自身情况进行操作.同时,遇到的问题也可能不尽相同,要灵活处理. 了解: Apollo是从原始ActiveMQ的基础构 ...

  6. Codeforces 1140E DP

    题意:给你一个数组,如果数组中的某个位置是-1那就可以填1到m的数字中的一个,但是要遵守一个规则:不能出现长度为奇数回文的子串,问合法的填法有多少种? 思路:不出现长度为奇数的回文子串,只需不出现长度 ...

  7. MySQL日期格式化 利用Mysql的DATE_FORMAT()进行日期格式转换

    碰到一个MYSQL的问题,表logstatb中moment字段的内容是"年-月-日 时:分:秒",需要查询匹配“年月日”或“时:分:秒”即可的数据条目,这个时候就可以通过下面的SQ ...

  8. Java——子类对象实例化的全过程

    2.4子类对象实例化的全过程 public class TestDog { public static void main(String[] args) { Dog d = new Dog(); d. ...

  9. redis和redis php扩展安装

    redis的源码安装 wget http://download.redis.io/redis-stable.tar.gz tar -zxvf redis-stable.tar.gz cd redis- ...

  10. GTK+/GNOME编程(一)

    1.开发环境:安装GTK+/GNOME库 #apt-get install gtk+-3.0                   (安装GTK+库文件) #apt-get install gnome- ...