链接:

https://www.nowcoder.com/acm/contest/139/J

题意:

给出n个整数的序列a(1≤ai≤n)和q个询问(1≤n,q≤1e5),每个询问包含两个整数L和R(1≤L,R≤n)。
对于每个询问,输出a[1...L]和a[R...n]的不同数字的个数。

分析:

将原数组复制一份拼接到末尾,把询问a[1...L]和a[R...n]转换为询问a[R...L+n]。
设kind[i]为a[1...i]出现的数字种类,
则询问a[L...R]的答案为 kind[R] - kind[L-1] +(a[1...L-1]和a[L...R]同时出现的数字种类)。
可以用树状数组维护a[1...L-1]和a[L...R]同时出现的数字种类。
如果a[i]在区间1...L-1出现过,则对应的树状数组位置的值应为1。查询时只需查询区间L...R。
可以对询问区间按左端点排序,左端点每次右移时把对应的数的下一个位置在树状数组里加1即可。

代码:

 #include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int MAXS = 1e6 + ;
int sf[MAXS];
void add(int p, int v, int n) {
while(p <= n) sf[p] += v, p += p&-p;
}
int sum(int p) {
int res = ;
while(p) res += sf[p], p -= p&-p;
return res;
} struct REGION {
int L, R, id;
bool operator < (const REGION& that) const {
return L < that.L;
}
} r[MAXS];
int a[MAXS], kind[MAXS], nxt[MAXS], rec[MAXS], ans[MAXS]; int main() {
int n, q;
while(~scanf("%d%d", &n, &q)) {
for(int i = ; i <= n; i++) scanf("%d", &a[i]);
for(int i = ; i < q; i++) {
scanf("%d%d", &r[i].R, &r[i].L);
r[i].R += n;
r[i].id = i;
}
memcpy(a+n+, a+, n*sizeof(int));
n *= ;
memset(rec, false, sizeof(rec));
for(int i = ; i <= n; i++) {
if(rec[a[i]]) kind[i] = kind[i-];
else kind[i] = kind[i-] + , rec[a[i]] = true;
}
memset(rec, false, sizeof(rec));
for(int i = n; i > ; i--) {
nxt[i] = rec[a[i]];
rec[a[i]] = i;
}
sort(r, r + q);
memset(sf, , sizeof(sf));
for(int p = , i = ; i < q; i++) {
while(p < r[i].L) add(nxt[p], , n), p++;
ans[r[i].id] = kind[r[i].R]-kind[r[i].L-] + sum(r[i].R)-sum(r[i].L-);
}
for(int i = ; i < q; i++) printf("%d\n", ans[i]);
}
return ;
}

牛客网多校训练第一场 J - Different Integers(树状数组 + 问题转换)的更多相关文章

  1. 牛客网多校第5场 H subseq 【树状数组+离散化】

    题目:戳这里 学习博客:戳这里 题意:给n个数为a1~an,找到字典序第k小的序列,输出该序列所有数所在位置. 解题思路:先把所有序列预处理出来,方法是设一个数组为dp,dp[i]表示以i为开头的序列 ...

  2. 牛客网多校第5场 I vcd 【树状数组+离散化处理】【非原创】

    题目:戳这里 学习博客:戳这里 作者:阿狸是狐狸啦 n个点,一个点集S是好的,当且仅当对于他的每个子集T,存在一个右边无限延长的矩形,使的这个矩形包含了T,但是和S-T没有交集. 求有多少个这种集合. ...

  3. 牛客网多校训练第一场 I - Substring(后缀数组 + 重复处理)

    链接: https://www.nowcoder.com/acm/contest/139/I 题意: 给出一个n(1≤n≤5e4)个字符的字符串s(si ∈ {a,b,c}),求最多可以从n*(n+1 ...

  4. 牛客网多校训练第一场 F - Sum of Maximum(容斥原理 + 拉格朗日插值法)

    链接: https://www.nowcoder.com/acm/contest/139/F 题意: 分析: 转载自:http://tokitsukaze.live/2018/07/19/2018ni ...

  5. 牛客网多校训练第一场 E - Removal(线性DP + 重复处理)

    链接: https://www.nowcoder.com/acm/contest/139/E 题意: 给出一个n(1≤n≤1e5)个整数(范围是1至10)的序列,求从中移除m(1≤m≤min(n-1, ...

  6. 牛客网多校训练第一场 D - Two Graphs

    链接: https://www.nowcoder.com/acm/contest/139/D 题意: 两个无向简单图都有n(1≤n≤8)个顶点,图G1有m1条边,图G2有m2条边,问G2有多少个子图与 ...

  7. 牛客网多校训练第一场 B - Symmetric Matrix(dp)

    链接: https://www.nowcoder.com/acm/contest/139/B 题意: 求满足以下条件的n*n矩阵A的数量模m:A(i,j) ∈ {0,1,2}, 1≤i,j≤n.A(i ...

  8. 牛客网多校训练第一场 A - Monotonic Matrix(Lindström–Gessel–Viennot lemma)

    链接: https://www.nowcoder.com/acm/contest/139/A 题意: 求满足以下条件的n*m矩阵A的数量模(1e9+7):A(i,j) ∈ {0,1,2}, 1≤i≤n ...

  9. 牛客网多校训练第二场D Kth Minimum Clique

    链接:https://ac.nowcoder.com/acm/contest/882/D来源:牛客网 Given a vertex-weighted graph with N vertices, fi ...

随机推荐

  1. Windows开启telnet命令

    1.点击开始 → 运行 → 输入telnet,回车. 2.点击启用或关闭Windows功能 3.找到Telnet客户端,勾选,点击确认 4.搞定,测试一下 打开CMD,在出来的DOS界面里,输入tel ...

  2. wrqer

  3. JqueryEasyUI $.Parser

    Parser(解析器) 对象的属性和方法: 使用: <link href="~/jquery-easyui-1.5.2/themes/bootstrap/easyui.css" ...

  4. javaEE Design Patter(1)初步了解23种常用设计模式

    设计模式分为三大类: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式. 结构型模式,共七种:适配器模式.装饰器模式.代理模式.外观模式.桥接模式.组合模式.享元模式. ...

  5. 深入浅出ConcurrentHashMap1.8

    转载:https://www.jianshu.com/p/c0642afe03e0 好文 关于文章中的疑问:为什么要构造一个反序链表,放在nextTable的i+n的位置上呢,在<深入分析Con ...

  6. 如何开发一个Servlet

    1 如何开发一个Servlet 1.1 步骤: 1)编写java类,继承HttpServlet类 2)重新doGet和doPost方法 3)Servlet程序交给tomcat服务器运行!! 3.1 s ...

  7. 微信小程序开发2-第一个小程序开发准备

    1.首先在官网上注册一个账号( https://mp.weixin.qq.com/ )申请一个AppID(类似于人的身份证,小程序也需要身份证) 注册过程不多说 2.安装开发工具( https://m ...

  8. 千里之堤毁于蚁穴(慎用HD Wallets)

    转自:http://blog.sina.com.cn/s/blog_12ce70a430102vbu9.html 千里之堤毁于蚁穴(慎用HD Wallets) -- 随机系列谈之四 现在我们都该明白, ...

  9. c++实现对输入数组进行快速排序

    #include "stdafx.h" #include <iostream> #include <string> #include <vector& ...

  10. Oracle中用户和方案的区别

    从定义中我们可以看出方案(Schema)为数据库对象的集合,为了区分各个集合,我们需要给这个集合起个名字,这些名字就是我们在企业管理器的方案下看到的许多类似用户名的节点,这些类似用户名的节点其实就是一 ...