[置顶] hdu4747 Mex 线段树
题意:给你一个序列,让你求出对于所有区间<i, j>的mex和,mex表示该区间没有出现过的最小的整数。
思路:从时限和点数就可以看出是线段树,并且我们可以枚举左端点i, 然后求出所有左端点为i的区间内mex值的和。
先把数插满,然后先询问后删除当前最左边的断点i。而且显然线段树里面保存的是mex值,而且这个序列是非递减的。
分析:我们先预处理出对于右端点为i的所有<1,i>的mex,分别插入线段树的i位置。然后每次删除最左边的左端点i
,假如当前我们要删除a[i] ,我们找到它之后第一个位置j满足a[i] == a[j], 那么区间i------j-1里面的所有mex都要更新,取线段树内的值和a[i]的最小值。 实际操作我们只要找到第一个比a[i]的位置l, r = j-1, 更新<l,r>之间的mex为a[i]即可。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
#define lson l, m, rt<<1
#define rson m+1, r, rt<<1|1
#define ls rt<<1
#define rs rt<<1|1
#define Mid int m = l+r>>1
const int maxn = 2000006;
int next[maxn], pre[maxn], n;
int a[maxn], mex;
bool vis[maxn];
ll sum[maxn<<2];
int mx[maxn<<2], col[maxn<<2];
void build(int l=1, int r=n, int rt=1) {
col[rt] = -1;
sum[rt] = 0;
mx[rt] = 0;
if(l == r) return;
Mid;
build(lson);
build(rson);
}
inline void down(int l, int r, int rt) {
if(~col[rt]) {
col[ls] = col[rs] = col[rt];
Mid;
sum[ls] = (ll)(m-l+1)*col[rt];
mx[ls] = mx[rs] = col[rt];
sum[rs] = (ll)(r-m)*col[rt];
col[rt] = -1;
}
}
inline void up(int rt) {
sum[rt] = sum[ls] + sum[rs];
mx[rt] = max(mx[ls], mx[rs]);
}
void update(int L, int R, int v, int l=1, int r=n, int rt=1) {
if(L <= l && r <= R) {
col[rt] = mx[rt] = v;
sum[rt] = (ll)(r-l+1)*v;
return;
}
Mid; down(l, r, rt);
if(L <= m) update(L, R, v, lson);
if(R > m) update(L, R, v, rson);
up(rt);
}
ll query(int L, int R, int l=1, int r=n, int rt=1) {
if(L <= l && r <= R)
return sum[rt];
Mid; down(l, r, rt);
ll ret = 0;
if(L <= m) ret += query(L, R, lson);
if(R > m) ret += query(L, R, rson);
up(rt);
return ret;
}
int find(int v, int l=1, int r=n, int rt=1) {
if(mx[rt] <= v) return n+1;
if(l == r) return l;
Mid; down(l, r, rt);
if(mx[ls] > v) return find(v, lson);
else return find(v, rson);
}
int main() {
int i, j;
while(~scanf("%d", &n) && n) {
for(i = 1; i <= n; i++) {
scanf("%d", &a[i]);
pre[i] = vis[i] = 0;
next[i] = n+1;
}
pre[0] = vis[0] = 0;
for(i = 1; i <= n; i++)
if(a[i] <= n) {
if(pre[a[i]])
next[pre[a[i]]] = i;
pre[a[i]] = i;
}
build();
mex = 0;
for(i = 1; i <= n; i++) {
if(a[i] <= n){
vis[a[i]] = 1;
while(vis[mex]) mex++;
}
update(i, i, mex);
}
ll ans = 0;
for(i = 1; i <= n; i++) {
ans += query(i, n);
if(a[i] <= mex) {
int l = max(find(a[i]), i);
int r = next[i]-1;
if(l <= r) update(l, r, a[i]);
}
}
printf("%I64d\n", ans);
}
return 0;
}
/*
3
0 10000 20000
*/
[置顶] hdu4747 Mex 线段树的更多相关文章
- HDU-4747 Mex 线段树
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4747 题意:求一个数列中,所有mex(L,R)的和. 注意到mex是单调不降的,那么首先预处理出mex ...
- hdu 4747 mex 线段树+思维
http://acm.hdu.edu.cn/showproblem.php?pid=4747 题意: 我们定义mex(l,r)表示一个序列a[l]....a[r]中没有出现过得最小的非负整数, 然后我 ...
- BZOJ.3585.mex(线段树)
题目链接 题意:多次求区间\(mex\). 考虑\([1,i]\)的\(mex[i]\),显然是单调的 而对于\([l,r]\)与\([l+1,r]\),如果\(nxt[a[l]]>r\),那么 ...
- bzoj 3585 mex - 线段树 - 分块 - 莫队算法
Description 有一个长度为n的数组{a1,a2,...,an}.m次询问,每次询问一个区间内最小没有出现过的自然数. Input 第一行n,m. 第二行为n个数. 从第三行开始,每行一个询问 ...
- HDU-4747 二分+线段树
题意:给出长度为n的序列,问任两个区间的mex运算结果的总和. 解法:直接讲线段树做法:我们注意到mex(1,1),mex(1,2),mex(1,3)...mex(1,i)的结果是单调不减的,那么我们 ...
- Codeforces 1083C Max Mex [线段树]
洛谷 Codeforces 思路 很容易发现答案满足单调性,可以二分答案. 接下来询问就转换成判断前缀点集是否能组成一条链. 我最初的想法:找到点集的直径,判断直径是否覆盖了所有点,需要用到树套树,复 ...
- 【bzoj3585】mex 线段树 mex,sg
Description 有一个长度为n的数组{a1,a2,…,an}.m次询问,每次询问一个区间内最小没有出现过的自然数. Input 第一行n,m. 第二行为n个数. 从第三行开始,每行一个询问l, ...
- CF1083C Max Mex 线段树
题面 CF1083C Max Mex 题解 首先我们考虑,如果一个数x是某条路径上的mex,那么这个数要满足什么条件? 1 ~ x - 1的数都必须出现过. x必须没出现过. 现在我们要最大化x,那么 ...
- [置顶] hdu 1890 伸展树区间翻转
题意: 给你n个数,每次先输出第i大的数的位置(如果有多个,选下标小的那个),然后每次将第i个位置到第i大的数所在位置之间的数进行翻转. 思路:输入的数组可能有多个相同的值,我们可以进行两次排序把数组 ...
随机推荐
- python学习day9
目录 一.队列 二.生产者消费者模型 三.协程 四.select\poll\epoll 五.paramiko 六.mysql API调用 一.队列(queue) 队列分以下三种: class queu ...
- PM产品经理练级攻略(1-5等级)
大家都叫“PM”,但做的事情却完全不同? “PM”这个词到底是什么意思? 这个话题恐怕也是各位同行都一直在想,也一直想不清楚的吧,我也是. 每次看到各种“产品经理的能力模型”,我都觉得有点扯淡,总觉得 ...
- OSCHina技术导向:Java轻量web开发框架——JFinal
JFinal 是基于 Java 语言的极速 WEB + ORM 框架,其核心设计目标是开发迅速.代码量少.学习简单.功能强大.轻量级.易扩展.Restful.在拥有Java语言所有优势的同时再拥有ru ...
- python模块目录文件后续
1,新增PythonModule加载path Ruiy tip(关于python list[]数据库类型特殊你懂的!append(""),extend([""] ...
- 学用 ASP.Net 之 System.Math 类
本文来自:http://www.cnblogs.com/del/archive/2011/01/03/1924746.html 成员: /* 字段 */ Math.E; //2.7182 ...
- ORACLE 主要后台进程1
Oracle Database Background Processes: 1.Database writer (DBWn)The database writer writes modified bl ...
- golang之pkg(包)
一.概述 Golang拥有超过100个标准包(可用go list std |wc -l查看) 任何包系统设计的目的都是简化大型程序的设计和维护工作,通过将一组相关的特性放进一个独立的模块以便于理解和更 ...
- 贪吃蛇 WPF
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- json输出用法+jquery validate
<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="WebForm1.aspx. ...
- 【IOS学习基础】NSObject.h学习
一.<NSObject>协议和代理模式 1.在NSObject.h头文件中,我们可以看到 // NSObject类是默认遵守<NSObject>协议的 @interface N ...