题意
给定一个长度为n的序列,和m个区间。
对一个区间的操作是:对整个区间的数-1
可以选择任意个区间(可以为0个、每个区间最多被选择一次)进行操作后,要求最大化的序列极差(极差即最大值 - 最小值)。
easy version的范围是(1≤n≤300,0≤m≤300)
hard version的范围是(1≤n≤1e5,0≤m≤300)

分析: 我们可以通过枚举最大或者最小值出现的位置 , 然后把不包括最大值位置或者包括最小值位置的区间选取 , 如果数据小我们是可以通过暴力的 , 但是这里的n有1e5 , 那我们就需要借助数据结构了 , 我们很容易可以看出是线段树的区间修改与区间最值 , 但是有个问题 , 我们枚举的过程中选区出来的区间还要进行回复 , 所有这里有个很精妙的东西:

我们可以对操作区间进行整理,使得在枚举过程中,不需要重复进行区间操作。
比如我们现在枚举的最小值位置是p,则我们需要选择包含了p的所有区间。
那么p前面一个状态p-1,进行的区间操作是包含了p-1的所有区间,这些p-1区间中,有些是包含了p的,有些是没包含p的,那么我们需要先删除掉没包含p的区间的影响(即把这些区间减掉的1给加回来)。同时还需要补上包含了p但是不包含p-1的区间(这样选择,就不会和p-1里面保留了的区间重复)。
包含了p-1,但不包含p的区间,显然是以p-1结尾的区间。我们用add[]数组保存以r[i]结尾的区间的编号。
包含了p,但不包含p-1的区间,显然是以p开始的区间。我们用sub[]数组保存以l[i]开始的区间的编号。
区间修改操作,就要用到线段树了。
最后输出答案时,使用最终确定的最小值的位置P,包含P的区间留下,不包含P的舍弃。
---------------------

#include <bits/stdc++.h>

using namespace std;
const int maxn = 1e5 + ;
const int maxm = ; int n, m;
int a[maxn], l[maxm], r[maxm], ans[maxn];
vector<int> add[maxn], sub[maxn]; struct node
{
int l, r;
int maxx, minn, lazy;
} tree[maxn << ]; void push_up(int k)
{
tree[k].maxx = max(tree[k << ].maxx, tree[k << | ].maxx);
tree[k].minn = min(tree[k << ].minn, tree[k << | ].minn);
} void push_down(int k)
{
if(tree[k].l != tree[k].r)
{
tree[k << ].maxx += tree[k].lazy, tree[k << ].minn += tree[k].lazy;
tree[k << | ].maxx += tree[k].lazy, tree[k << | ].minn += tree[k].lazy;
tree[k << ].lazy += tree[k].lazy;
tree[k << | ].lazy += tree[k].lazy;
}
tree[k].lazy = ;
} void build(int k, int l, int r)
{
tree[k].l = l;
tree[k].r = r;
if(l == r)
{
tree[k].maxx = a[l];
tree[k].minn = a[l];
tree[k].lazy = ;
return ;
}
int mid = (l + r) >> ;
build(k << , l, mid);
build(k << | , mid + , r);
push_up(k);
} void update(int k, int l, int r, int x)
{
if(tree[k].lazy)
push_down(k);
tree[k].maxx += x;
tree[k].minn += x;
if(tree[k].l == l && tree[k].r == r)
{
tree[k].lazy += x;
return ;
}
int mid = (tree[k].l + tree[k].r) >> ;
if(r <= mid)
update(k << , l, r, x);
else if(l > mid)
update(k << | , l, r, x);
else
{
update(k << , l, mid, x);
update(k << | , mid + , r, x);
}
push_up(k);
} int query_max(int k, int l, int r)
{
int maxx;
if(tree[k].lazy)
push_down(k);
if(tree[k].l == l && tree[k].r == r)
return tree[k].maxx;
int mid = (tree[k].l + tree[k].r) >> ;
if(r <= mid)
maxx = query_max(k << , l, r);
else if(l > mid)
maxx = query_max(k << | , l, r);
else
maxx = max(query_max(k << , l, mid), query_max(k << | , mid + , r));
return maxx;
} int query_min(int k, int l, int r)
{
int minn;
if(tree[k].lazy)
push_down(k);
if(tree[k].l == l && tree[k].r == r)
return tree[k].minn;
int mid = (tree[k].l + tree[k].r) >> ;
if(r <= mid)
minn = query_min(k << , l, r);
else if(l > mid)
minn = query_min(k << | , l, r);
else
minn = min(query_min(k << , l, mid), query_min(k << | , mid + , r));
return minn;
} int main()
{
scanf("%d%d", &n, &m);
for(int i = ; i <= n; i++)
scanf("%d", &a[i]); for(int i = ; i <= m; i++)
{
scanf("%d%d", &l[i], &r[i]); //统计区间
sub[ l[i] ].push_back(i);
add[ r[i] ].push_back(i);
} //构造线段树
build(, , n); int d = -, p;//d记录最大极差,p记录最小值的位置
for(int i = ; i <= n; i++)//枚举最小值出现的位置
{
for(int j = ; j < add[i - ].size(); j++)//消除包含i-1,不包含i区间的影响
{
int id = add[i - ][j];//注意这里是i-1
update(, l[id], r[id], );
} for(int j = ; j < sub[i].size(); j++)//添加包含i,不包含i-1区间的影响
{
int id = sub[i][j];
update(, l[id], r[id], -);
} //查询极差 注意这里虽然知道最小值是a[i],但是由于使用了线段树,有lazy标记,标记可能没有更新到最底层,所以不能直接使用a[i]。
int det = query_max(, , n) - query_min(, , n);
if(det > d)
{
d = det;
p = i;
}
} printf("%d\n", d);
int t = ;
for(int i = ; i <= m; i++)//统计区间
{
if(l[i] <= p && p <= r[i])
ans[t++] = i;
}
printf("%d\n", t);
for(int i = ; i < t; i++)
{
if(i != )
printf(" ");
printf("%d", ans[i]);
}
printf("\n");
return ;
}

参考与原文:https://blog.csdn.net/Floraqiu/article/details/86632558

CF E2 - Array and Segments (Hard version) (线段树)的更多相关文章

  1. Codeforces Round #535 (Div. 3) E2. Array and Segments (Hard version) 【区间更新 线段树】

    传送门:http://codeforces.com/contest/1108/problem/E2 E2. Array and Segments (Hard version) time limit p ...

  2. CF #535 (Div. 3) E2 Array and Segments (Hard version) 利用线段树进行区间转移

    传送门 题意:    有m个区间,n个a[ i ] , 选择若干个区间,使得整个数组中的最大值和最小值的差值最小.n<=1e5,m<=300; 思路: 可以知道每个i,如果一个区间包含这个 ...

  3. E1. Array and Segments (Easy version)(暴力) && E2. Array and Segments (Hard version)(线段树维护)

    题目链接: E1:http://codeforces.com/contest/1108/problem/E1 E2:http://codeforces.com/contest/1108/problem ...

  4. Codeforces 1108E2 Array and Segments (Hard version) 差分, 暴力

    Codeforces 1108E2 E2. Array and Segments (Hard version) Description: The only difference between eas ...

  5. Codeforces 1108E2 Array and Segments (Hard version)(差分+思维)

    题目链接:Array and Segments (Hard version) 题意:给定一个长度为n的序列,m个区间,从m个区间内选择一些区间内的数都减一,使得整个序列的最大值减最小值最大. 题解:利 ...

  6. 1304F2 - Animal Observation (hard version) 线段树or单调队列 +DP

    1304F2 - Animal Observation (hard version) 线段树or单调队列 +DP 题意 用摄像机观察动物,有两个摄像机,一个可以放在奇数天,一个可以放在偶数天.摄像机在 ...

  7. CF1108E2 Array and Segments (Hard version)

    线段树 对于$Easy$ $version$可以枚举极大值和极小值的位置,然后判断即可 但对于$Hard$ $version$明显暴力同时枚举极大值和极小值会超时 那么,考虑只枚举极小值 对于数轴上每 ...

  8. CF 552(div 3) E Two Teams 线段树,模拟链表

    题目链接:http://codeforces.com/contest/1154/problem/E 题意:两个人轮流取最大值与旁边k个数,问最后这所有的数分别被谁给取走了 分析:看这道题一点思路都没有 ...

  9. CF 666E Forensic Examination 【SAM 倍增 线段树合并】

    CF 666E Forensic Examination 题意: 给出一个串\(s\)和\(n\)个串\(t_i\),\(q\)次询问,每次询问串\(s\)的子串\(s[p_l:p_r]\)在串\(t ...

随机推荐

  1. Angular问题04 模块导入错误???、BrowserModule模块重复加载???、material模块引入后报错

    1 模块导入错误 1.1 问题描述 项目启动时报错:元数据错误,错误截图如下: 1.2 问题原因 利用VsCode开发angular项目时利用自动导入出现错误 坑01:VsCode 的自动导入功能比较 ...

  2. 基于PCL的屏幕选点、框选点云、单点选取

    1. 单点选取 #include <pcl/io/pcd_io.h> #include <pcl/point_cloud.h> #include <pcl/point_t ...

  3. Apache apxs命令

    一.简介 apxs是一个为Apache HTTP服务器编译和安装扩展模块的工具,用于编译一个或多个源程序或目标代码文件为动态共享对象,使之可以用由mod_so提供的LoadModule指令在运行时加载 ...

  4. R list frame, matrix

    list是个筐,什么都可以往里装,包括它自身.数据框是个二维数组,同列必须同类型,行随意.

  5. db2 中 SQL判断物理表是否存在、修改表名

    1.db2 中 SQL判断物理表是否存在 SELECT * FROM SYSIBM.SYSTABLES WHERE TID <> 0 AND Name = 'TABLE_NAME' AND ...

  6. lda:变分的推导

    lda,latent diriclet allocation,是一个最基本的bayesian模型.本文要研究lda基于变分的推导方法.意义是重大的. 一.符号的定义 : the number of t ...

  7. React + Python 七月小说网 启程(一)

    一.为啥要做这个网站 很久没有写技术相关的博客了,最近几个月忙飞,各种工作,技术根本学不完,很难受. 趁着春节期间,终于有空闲时间做自己爱做的事情了,美滋滋. 热爱技术,热爱小说,于是诞生了个这么玩意 ...

  8. GetPixelAddress()函数Alpha通道会丢失

    CImage类中GetPixelAddress()函数来设置获取对应的颜色值是发现Alpha无效. void CGBImage::Load(){ CImage sourceImage; sourceI ...

  9. 扩展卢卡斯定理(Exlucas)

    题目链接 戳我 前置知识 中国剩余定理(crt)或扩展中国剩余定理(excrt) 乘法逆元 组合数的基本运用 扩展欧几里得(exgcd) 说实话Lucas真的和这个没有什么太大的关系,但是Lucas还 ...

  10. 多态的作用-游戏编程展示------新标准c++程序设计

    游戏软件的开发最能体现面向对象设计方法的优势.游戏中的人物.道具.建筑物.场景等都是很直观的对象,游戏运行的过程就是这些对象相互作用的过程.每个对象都有自己的属性和方法,不同对象也可能有共同的属性和方 ...