一眼线段树...显然,我们可以考虑最后所留下的区间,那显然这个区间中应当不能存在任何与区间外相同的颜色。这里的转化也是很常用的,我们用 \(nxt[i]\) 表示与 \(i\) 颜色相同的下一个位置在哪里, \(last[i]\) 表示与 \(i\) 颜色相同的上一个位置在哪里;那么一个区间 \(i, j\) 是满足要求的当且仅当 \(min(last[k]) >= i, max(nxt[k]) <= j (i <= k  <= j)\) 。我们可以用单调栈处理出 \(lim[i]\) 记录下第一个 \(last[k] < i (k >= i)\) 的 \(k\)。那么我们可以发现以 \(i\) 为区间左端点的区间右端点一定在 \([i, lim[i] - 1]\) 之间。我们考虑如何满足有关 \(j\) 的限制。

  我们由于左端点是从右往左的扫描线,所以考虑答案的时候也只需要考虑当前 \(>= i\) 的 \(j\) (满足当前考虑的限制均是在 \([i, n]\) 范围内的限制)。由于 \(max(nxt[k]) <= j\) ,所以对于所有的 \(k\) 与 \(nxt[k]\) 而言,它所影响到的右端点就是在 \([k, nxt[k] - 1]\) 范围内的节点,让他们都无法与之后所有的左端点匹配成为合法的区间。我们只需要维护一棵每碰到限制就区间赋值为 \(0\)的线段树 ,并查询区间内的总和就可以计算出合法的区间总数啦。

#include <bits/stdc++.h>
using namespace std;
#define maxn 1000000
#define LL long long
#define INF 99999999999LL
int n, a[maxn], rec[maxn], mark[maxn];
int top, S[maxn], nxt[maxn], last[maxn], lim[maxn];
int sum[maxn]; LL ans; int read()
{
int x = , k = ;
char c; c = getchar();
while(c < '' || c > '') { if(c == '-') k = -; c = getchar(); }
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x * k;
} void Push_down(int p)
{
if(!mark[p]) return;
int ls = p << , rs = p << | ;
mark[ls] = mark[rs] = ; sum[ls] = sum[rs] = ;
mark[p] = ;
} int Query(int p, int l, int r, int L, int R)
{
if(L > R) return ;
if(L > r || R < l) return ;
if(L <= l && R >= r) return sum[p];
int mid = (l + r) >> ;
Push_down(p);
return Query(p << , l, mid, L, R) + Query(p << | , mid + , r, L, R);
} void Update(int p, int l, int r, int L, int R)
{
if(L > r || R < l) return;
if(L <= l && R >= r) { mark[p] = , sum[p] = ; return; }
int mid = (l + r) >> ;
Push_down(p);
Update(p << , l, mid, L, R), Update(p << | , mid + , r, L, R);
sum[p] = sum[p << ] + sum[p << | ];
} void Build(int p, int l, int r)
{
if(l == r) { sum[p] = ; return; }
int mid = (l + r) >> ;
Build(p << , l, mid), Build(p << | , mid + , r);
sum[p] = sum[p << ] + sum[p << | ];
} void init()
{
memset(mark, , sizeof(mark));
memset(rec, , sizeof(rec));
ans = ;
} signed main()
{
int T = read();
while(T --)
{
init(); n = read();
for(int i = ; i <= n; i ++) a[i] = read();
for(int i = ; i <= n; i ++)
{
last[i] = nxt[i] = ;
last[i] = rec[a[i]], nxt[rec[a[i]]] = i;
rec[a[i]] = i;
}
for(int i = ; i <= n; i ++)
if(!last[i]) last[i] = n + ;
for(int i = n; i >= ; i --)
{
while(top >= && last[S[top]] > last[i]) top --;
S[++ top] = i;
while(top >= && last[S[top]] >= i) top --;
if(!top) lim[i] = n + ;
else lim[i] = S[top];
}
Build(, , n);
for(int i = n; i >= ; i --)
{
if(nxt[i]) Update(, , n, i, nxt[i] - );
ans += (LL) Query(, , n, i, lim[i] - );
}
printf("%lld\n", ans);
}
return ;
}

【题解】JXOI2017颜色的更多相关文章

  1. [JXOI2017]颜色 线段树求点对贡献

    [JXOI2017]颜色 题目链接 https://www.luogu.org/problemnew/show/P4065 题目描述 可怜有一个长度为 n 的正整数序列 Ai,其中相同的正整数代表着相 ...

  2. JXOI2017颜色 解题报告

    JXOI2017颜色 首先记录每个位置上颜色在序列中上次出现的位置 开两颗线段树,第一棵维护区间最大值,实际上是维护当前必须被删去的颜色的位置的最大值,第二棵则是维护区间和 首先倒着扫一遍,对于当前颜 ...

  3. BZOJ5011 & 洛谷4065 & LOJ2275:[JXOI2017]颜色——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=5011 https://www.luogu.org/problemnew/show/P4065 ht ...

  4. [JXOI2017]颜色 线段树扫描线 + 单调栈

    ---题面--- 题解: 首先题目要求删除一些颜色,换个说法就是要求保留一些颜色,那么观察到,如果我们设ll[i]和rr[i]分别表示颜色i出现的最左边的那个点和最右边的那个点,那么题目就是在要求我们 ...

  5. [BZOJ5011][JXOI2017]颜色

    5011: [Jx2017]颜色 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 84  Solved: 46[Submit][Status][Disc ...

  6. BZOJ5011 [JXOI2017]颜色 【线段树 + 主席树】

    题目链接 BZOJ5011 题解 一定只有我这种智障会用这么奇怪的方法做这道题.. 由题我们知道最后剩余的一定是一个区间,而且区间内的颜色不存在于区间外 所以我们的目的就是为了找到这样的区间的数量 区 ...

  7. [JXOI2017]颜色

    \(Orz\) 各位题解大佬,我来膜拜一发 还有单调栈实在没弄懂 法一:线段树+堆 首先,讨论区间的个数的题目,我们可以想到枚举一个端点\(r\),找到所有的\(l\) 我们不妨设:\(ml[i]\) ...

  8. 洛谷P4065 [JXOI2017]颜色(线段树)

    题意 题目链接 Sol 线段树板子题都做不出来,真是越来越菜了.. 根据题目描述,一个合法区间等价于在区间内的颜色没有在区间外出现过. 所以我们可以对于每个右端点,统计最长的左端点在哪里,刚开始以为这 ...

  9. JXOI2017颜色

    题面 loj 分析 这道题非常妙啊 对于可保留区间[l, r] 枚举右端点r 考虑l的取值范围有两重约数 记颜色i出现的最右侧位置是\(max_i\) 最左侧位置是\(min_i\) r前最后一次出现 ...

随机推荐

  1. C++三元操作符

    c++的三元操作符形式: //条件表达式 ? 表达式1 : 表达式2; 语义:如果“条件表达式”为true,则整个表达式的值就是表达式1,忽略表达式2:如果“条件表达式”为false,则整个表达式的值 ...

  2. MyBatis-自定义结果映射规则

    1.自定义结果集映射规则 ①查询 <!-- public Employee getEmpById(Integer id); --> <select id="getEmpBy ...

  3. Omad群组部署、依赖部署一键解决

    本文来自网易云社区 作者:李培斌 前言 基于omad部署平台实现一键部署的实践已有很多成功的经验,杭研QA的技术先锋们也在ks圈里有很多不同的文章去阐述关于这类需求的实现和思路,当然包括我们金融事业部 ...

  4. MySQL 取得字段子串修改

    MySQL 中, GeneID 为 GRMZM2G549533_P01,1123,45 , 需要修改为 GRMZM2G549533_P01 update test set GeneID=SUBSTRI ...

  5. 【转】Oracle 如何找回已经删除了的表记录

    有的时候我们不小心把数据库表(emp)中重要的记录给删除了,怎么给找回来了,看下面这个例子你就会明白. 某一天,10点钟的时候,张三一不小心给数据库表emp的一条重要记录给删除了并且还提交了,此时也没 ...

  6. PLSQL集合类型

    PLSQL集合类型   --联合数组(索引表) /* 用于存储某个数据类型的数据集合类型 .通过索引获得联合数组中得值 如下例子: */ DECLARE CURSOR cur_chars IS SEL ...

  7. 关于css,js显示不出来

    有时候取消了对js,css文件的拦截,但是在网页上还是显示不了,可以试一下下面的方法: 在jsp页面添加下面内容 css的调用路径用path来代替

  8. 结合BeautifulSoup和hackhttp的爬虫实例

    网页页数的改变 headers头不添加

  9. Selenium自动化测试第一天(下)

    如有任何学习问题,可以添加作者微信:lockingfree 目录 Selenium自动化测试基础 Selenium自动化测试第一天(上) Selenium自动化测试第一天(下) Selenium自动化 ...

  10. Spring Boot下的lombok安装 (日志) 不能识别log变量问题

    参考地址:http://blog.csdn.net/blueheart20/article/details/52909775 ps:除了要加载依赖之外 还要安装lombok插件