J.Different Integers

题意就是给你l,r,问你在区间两侧的[1,l]和[r,n]中,不同数的个数。

两种思路:

1.将数组长度扩大两倍,for(int i=n+1;i<=2*n;i++) a[i]=a[i-n];就可以将两个分开的区间合并成一个区间[r,l+n],然后就可以通过主席树求解,套模板就可以了。

但是用主席树有风险,容易写超时,超内存,只能通过50%,初始化数组memset少写一个就过了,而且while(scanf("%d%d",&n,&m)==2)要写成这样,我赛后试了一下while(~scanf("%d%d",&n,&m)),过不了,只能过50%,所以主席树正好卡过去。

代码:

 //J-主席树刚好卡过
#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
#include<cmath>
#include<cstdlib>
#include<queue>
using namespace std;
const int maxn=*1e5+; struct Node
{
int l,r,sum; }p[maxn*];
int la[maxn],a[maxn],root[maxn];
int n,q,cnt; int build(int l,int r)
{
int nc=++cnt;
p[nc].sum=;
p[nc].l=p[nc].r=;
if(l==r){
return nc;
} int m=l+r>>;
p[nc].l=build(l,m);
p[nc].r=build(m+,r);
return nc;
}
int update(int pos,int c,int v,int l,int r)
{
int nc=++cnt;
p[nc]=p[c];
p[nc].sum+=v;
if(l==r){
return nc;
} int m=l+r>>;
if(m>=pos)
p[nc].l=update(pos,p[c].l,v,l,m);
else
p[nc].r=update(pos,p[c].r,v,m+,r);
return nc;
}
int query(int pos,int c,int l,int r)
{
if(l==r) return p[c].sum;
int m=l+r>>;
if(m>=pos)
return p[p[c].r].sum + query(pos,p[c].l,l,m);
else
return query(pos,p[c].r,m+,r);
}
int main()
{
while(scanf("%d%d",&n,&q)==){
memset(la,-,sizeof(la));
cnt=;
for (int i=;i<=n;i++){
scanf("%d",a+i);
}
for(int i=n+;i<=*n;i++)
a[i]=a[i-n];
n=n*;
root[]=build(,n);
for (int i=;i<=n;i++){
int v=a[i];
if(la[v]==-)
root[i]=update(i,root[i-],,,n);
else{
int t=update(la[v],root[i-],-,,n);
root[i]=update(i,t,,,n);
}
la[v]=i;
}
while(q--){
int x,y;
scanf("%d%d",&x,&y);
printf("%d\n",query(y,root[n/+x],,n));
}
}
}

2.叉姐官方题解思路是树状数组,离线按照 r 从小到大处理询问,考虑没出现的数字个数,假设 r = last[x],那么当 l < first[x] 时,x 没出现,用树状数组维护即可。

唉,真的是长的帅又厉害代码写的还优美,代码写的真的是好看,羡慕(✧◡✧)

代码:(叉姐代码)

 #include <bits/stdc++.h>

 struct Query
{
int l, r, id;
}; bool operator < (const Query& u, const Query& v)
{
return u.r < v.r;
} int main()
{
int n, q;
while (scanf("%d%d", &n, &q) == ) {
std::vector<int> a(n), first(n, -), last(n);
int total = ;
for (int i = ; i < n; ++ i) {
scanf("%d", &a[i]);
a[i] --, last[a[i]] = i;
if (first[a[i]] == -) {
total ++, first[a[i]] = i;
}
}
std::vector<Query> queries;
for (int i = , l, r; i < q; ++ i) {
scanf("%d%d", &l, &r);
queries.push_back(Query {l - , r - , i});
}
std::sort(queries.begin(), queries.end());
std::vector<int> count(n), result(q);
for (int i = , k = ; i < n; ++ i) {
while (k < q && queries[k].r == i) {
int& ref = result[queries[k].id] = total;
for (int j = queries[k].l; j < n; j += ~j & j + ) {
ref -= count[j];
}
k ++;
}
if (last[a[i]] == i) {
for (int j = first[a[i]] - ; ~j; j -= ~j & j + ) {
count[j] ++;
}
}
}
for (int i = ; i < q; ++ i) {
printf("%d\n", result[i]);
}
}
}

自己改了一下叉姐代码加了一点注释的代码:

 //J-离线树状数组处理
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
#include<map>
#include<vector>
using namespace std;
const int maxn=1e5+;
const int inf=0x3f3f3f3f;
const double eps=1e-;
struct node
{
int l,r,id;
bool operator< (const node &a) const{ //结构体内嵌比较函数
return r<a.r;
}
};
int main()
{
int n,q;
while(~scanf("%d%d",&n,&q)){
vector<int>a(n),first(n, -),last(n);
int total=;
for (int i=;i<n;i++){
scanf("%d",&a[i]);
a[i]--,last[a[i]]=i;//记录元素最后出现的位置
if(first[a[i]]==-){//如果元素之前未出现,则记录第一次出现的位置
total++;
first[a[i]]=i;
}
}
vector<node> queries;
for (int i=;i<q;i++) {
int l,r;
scanf("%d%d",&l,&r);
queries.push_back(node {l-,r-,i});//将区间和id存到容器内
}
sort(queries.begin(),queries.end());//区间右边界升序排序(结构体内内嵌了比较函数)
vector<int>count(n),result(q);
for (int i=,k=;i<n;i++){//树状数组维护
while(k<q&&queries[k].r==i){
int &ref=result[queries[k].id]=total;
for(int j=queries[k].l;j<n;j+=~j&j+){//~j&j+1 +的优先级比&高
ref-=count[j];
}
k++;
}
if(last[a[i]]==i){
for(int j=first[a[i]]-;~j;j-=~j&j+){
count[j]++;//统计没有出现过的元素个数
}
}
}
for (int i=;i<q;i++) {
printf("%d\n",result[i]);
}
}
}

讲道理,我是真的菜,难受。

牛客网 暑期ACM多校训练营(第一场)J.Different Integers-区间两侧不同数字的个数-离线树状数组 or 可持久化线段树(主席树)的更多相关文章

  1. 牛客网暑期ACM多校训练营 第九场

    HPrefix Sum study from : https://blog.csdn.net/mitsuha_/article/details/81774727 k较小.分离x和k. 另外的可能:求a ...

  2. 牛客网暑期ACM多校训练营(第四场):A Ternary String(欧拉降幂)

    链接:牛客网暑期ACM多校训练营(第四场):A Ternary String 题意:给出一段数列 s,只包含 0.1.2 三种数.每秒在每个 2 后面会插入一个 1 ,每个 1 后面会插入一个 0,之 ...

  3. 牛客网暑期ACM多校训练营(第五场):F - take

    链接:牛客网暑期ACM多校训练营(第五场):F - take 题意: Kanade有n个盒子,第i个盒子有p [i]概率有一个d [i]大小的钻石. 起初,Kanade有一颗0号钻石.她将从第1到第n ...

  4. 牛客网 暑期ACM多校训练营(第二场)A.run-动态规划 or 递推?

    牛客网暑期ACM多校训练营(第二场) 水博客. A.run 题意就是一个人一秒可以走1步或者跑K步,不能连续跑2秒,他从0开始移动,移动到[L,R]的某一点就可以结束.问一共有多少种移动的方式. 个人 ...

  5. 牛客网 暑期ACM多校训练营(第一场)A.Monotonic Matrix-矩阵转化为格子路径的非降路径计数,Lindström-Gessel-Viennot引理-组合数学

    牛客网暑期ACM多校训练营(第一场) A.Monotonic Matrix 这个题就是给你一个n*m的矩阵,往里面填{0,1,2}这三种数,要求是Ai,j⩽Ai+1,j,Ai,j⩽Ai,j+1 ,问你 ...

  6. 牛客网暑期ACM多校训练营(第三场)H Diff-prime Pairs (贡献)

    牛客网暑期ACM多校训练营(第三场)H Diff-prime Pairs (贡献) 链接:https://ac.nowcoder.com/acm/contest/141/H来源:牛客网 Eddy ha ...

  7. 2018牛客网暑期ACM多校训练营(第二场)I- car ( 思维)

    2018牛客网暑期ACM多校训练营(第二场)I- car 链接:https://ac.nowcoder.com/acm/contest/140/I来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 ...

  8. 牛客网 暑期ACM多校训练营(第二场)J.farm-STL(vector)+二维树状数组区间更新、单点查询 or 大暴力?

    开心.jpg J.farm 先解释一下题意,题意就是一个n*m的矩形区域,每个点代表一个植物,然后不同的植物对应不同的适合的肥料k,如果植物被撒上不适合的肥料就会死掉.然后题目将每个点适合的肥料种类( ...

  9. 牛客网暑期ACM多校训练营(第七场)Bit Compression

    链接:https://www.nowcoder.com/acm/contest/145/C 来源:牛客网 题目描述 A binary string s of length N = 2n is give ...

  10. 牛客网暑期ACM多校训练营(第一场) - J Different Integers(线段数组or莫队)

    链接:https://www.nowcoder.com/acm/contest/139/J来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 524288K,其他语言1048 ...

随机推荐

  1. 《算法》C++代码 快速排序

    快速排序,简称快排,常称QuickSort.QSort.在排序算法中非常常用,其编程复杂度低,时间复杂度O(NlogN),空间复杂度O(N),执行效率稳定,而且常数很低. 基本思想就是二分,例如你要将 ...

  2. jenkins 连接服务器并运行脚本

    1.登录,在系统管理——节点管理——新增节点——配置从节点,添加远程工作目录,选择启动方式:通过JAVA WEB启动代理,添加JDK 2.在列表点节点,点launch下载插件,放到D:\JENKINS ...

  3. Python网络编程(子进程的创建与处理、简单群聊工具)

    前言: 昨天我们已经了解了多进程的原理以及它的实际使用 Unix/Linux操作系统提供了一个fork()系统调用,它非常特殊.普通的函数调用,调用一次,返回一次, 但是fork()调用一次,返回两次 ...

  4. 孤荷凌寒自学python第四十三天python 的线程同步之Queue对象

     孤荷凌寒自学python第四十三天python的线程同步之Queue对象 (完整学习过程屏幕记录视频地址在文末,手写笔记在文末) Queue对象是直接操作队列池的对象,队列中可以存放多种对象,当然也 ...

  5. HDU 4665 Unshuffle DFS找一个可行解

    每层找一对相等的整数,分别放在两个不同的串中. 参考了学弟的解法,果断觉得自己老了…… #include <cstdio> #include <cstring> #includ ...

  6. web自动化测试:watir+minitest(四)

    脚本连跑: rake是ruby中的一个构建工具,和make很像.允许用ruby来写rakefile. 我们使用rake以任务的方式来运行我们的脚本集. 新建Rakefile文件,写入如下内容: req ...

  7. Sockt编程(多线程)

    服务器端: package com.zeph.serverclient; import java.io.BufferedReader; import java.io.IOException; impo ...

  8. EXTJS4.0 显示图片 动态图片

    在网上找了好久没有好的解决方法 都是用 'box' 什么的组件 改写的 autoEl 好麻烦,修改 好的属性都不能用.弄了好久没弄成 最后: 用panel  显示 html 文本 追加 ‘'<i ...

  9. JZOJ 5279 香港记者

    一句话题意:一个带点权边权的无向图,输出从1号点到n号点的最短路中字典序最小的一条路径 样例输入: 8 9 1 2 3 4 5 6 7 8 1 2 2 2 3 3 3 8 3 1 4 3 4 5 2 ...

  10. POJ 2749 Building roads 2-sat+二分答案

    把爱恨和最大距离视为限制条件,可以知道,最大距离和限制条件多少具有单调性 所以可以二分最大距离,加边+check #include<cstdio> #include<algorith ...