传送门

这道题是主席树好题啊……

####### 题目大意:给定一个序列,每次给定一段区间,区间内所有不同的数第一次出现的位置排成一个序列,求这个序列的中位数。

其实求中位数并不是很难,只要我们能把这个序列中不同的数出现的位置都求出,就能很直接的转化为用主席树求静态第k小的问题。首先我们考虑一下如何计算区间内有多少个不同的数,如果离线的话这个问题可以用莫队或者树状数组处理,但是本题因为强制在线,所以需要用主席树处理。具体的方法很巧妙,因为我们要维护的是最左边的第一次出现的值,所以我们倒着插入。因为维护的是位置,所以我们用第\(i\)棵主席树维护区间\(i~r\)内所有不同的数出现的次数(也就是主席树以位置为值域),以每个数的位置为权值在树上进行二分。当一个数在一起出现过一次的时候,我们就先找到它原来所在的位置,把它从树上删除,之后再在它的新位置加入这个数,更改这个数原来的位置即可。

至于查询的时候,我们就可以只查询第l棵主席树上的情况了,我们像线段树一样\(query\)一下这个树上有多少不同的数,之后仿照模板 ,在树上二分求静态第k小即可。

然后还有一道题是它的弱化版here 此题不强制在线不用求第k大,不过数的范围大了一点,所以需要把记录位置的数组稍微开大一点才行。

本题千万要注意一下输出格式!修改的时候一定要按代码中顺序修改,如果倒过来的话,一旦这个数是第一次出现就会出错(没继承上一个根的信息)。

看一下代码。

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cmath>
#include<set>
#include<vector>
#include<map>
#include<queue>
#define rep(i,a,n) for(int i = a;i <= n;i++)
#define per(i,n,a) for(int i = n;i >= a;i--)
#define enter putchar('\n')
#define fr friend inline
#define y1 poj
#define mp make_pair
#define pr pair<int,int>
#define fi first
#define sc second
#define pb push_back using namespace std;
typedef long long ll;
const int M = 200005;
const int N = 2000005;
const int INF = 1000000009;
const double eps = 1e-7; int read()
{
int ans = 0,op = 1;
char ch = getchar();
while(ch < '0' || ch > '9')
{
if(ch == '-') op = -1;
ch = getchar();
}
while(ch >= '0' && ch <= '9')
{
ans *= 10;
ans += ch - '0';
ch = getchar();
}
return ans * op;
} struct node
{
int lson,rson,v;
}t[N<<2]; int T,n,m,l,r,a[M],pos[M],idx,root[M],la; void modify(int old,int &p,int l,int r,int pos,int val)
{
p = ++idx;
t[p].lson = t[old].lson,t[p].rson = t[old].rson,t[p].v = t[old].v + val;
if(l == r) return;
int mid = (l+r) >> 1;
if(pos <= mid) modify(t[old].lson,t[p].lson,l,mid,pos,val);
else modify(t[old].rson,t[p].rson,mid+1,r,pos,val);
} int query(int p,int l,int r,int kl,int kr)
{
if(l == kl && r == kr) return t[p].v;
int mid = (l+r) >> 1;
if(kr <= mid) return query(t[p].lson,l,mid,kl,kr);
else if(kl > mid) return query(t[p].rson,mid+1,r,kl,kr);
else return query(t[p].lson,l,mid,kl,mid) + query(t[p].rson,mid+1,r,mid+1,kr);
} int ask(int p,int l,int r,int k)
{
if(l == r) return l;
int mid = (l+r) >> 1;
if(t[t[p].lson].v >= k) return ask(t[p].lson,l,mid,k);
else return ask(t[p].rson,mid+1,r,k - t[t[p].lson].v);
} void clear()
{
memset(root,0,sizeof(root));
memset(pos,0,sizeof(pos));
idx = la = 0;
} int main()
{
//freopen("1.out","w",stdout);
T = read();
rep(i,1,T)
{
printf("Case #%d:",i);
n = read(),m = read();
rep(j,1,n) a[j] = read();
per(j,n,1)
{
modify(root[j+1],root[j],1,n,j,1);
if(pos[a[j]]) modify(root[j],root[j],1,n,pos[a[j]],-1); //Accepted
pos[a[j]] = j;
}
rep(j,1,m)
{
l = read(),r = read();
l = (l + la) % n + 1,r = (r + la) % n + 1;
if(l > r) swap(l,r);
//printf("#%d %d\n",l,r);
int cur = query(root[l],1,n,l,r);
la = ask(root[l],1,n,(cur+1)>>1);
printf(" %d",la);
}
enter;
clear();
}
return 0;
}

HDU5919 Sequence2的更多相关文章

  1. HDU5568/BestCoder Round #63 (div.2) B.sequence2 dp+高精度

    sequence2 Problem Description Given an integer array bi with a length of n, please tell me how many ...

  2. Hdu 5568 sequence2 高精度 dp

    sequence2 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=556 ...

  3. sequence2(高精度dp)

    sequence2 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total ...

  4. 【HDU5919】SequenceII(主席树)

    [HDU5919]SequenceII(主席树) 题面 Vjudge 翻译(by ppl) 给一个长度为N的数列A,有m个询问,每次问 数列[l,r]区间中所有数的第一次出现的位置的中位 数是多少 题 ...

  5. [HDU5919]Sequence II

    [HDU5919]Sequence II 试题描述 Mr. Frog has an integer sequence of length n, which can be denoted as a1,a ...

  6. HDU 5568:sequence2 大数+DP

    sequence2  Accepts: 93  Submissions: 358  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: 6553 ...

  7. HDU 5568 sequence2 区间dp+大数

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5568 题意: 求所有长度为k的严格升序子序列的个数. 题解: 令dp[i][k]表示以i结尾的长度为 ...

  8. HDU 5568 - BestCoder Round #63 - sequence2

    题目链接 : http://acm.hdu.edu.cn/showproblem.php?pid=5568 题意 : 给一个长度已知的序列, 给一个值k, 问该序列中有多少种长度为k的上升子序列 思路 ...

  9. HDU5919:Sequence II

    题面 Vjudge Sol 给一个数列,有m个询问,每次问数列[l,r]区间中所有数的第一次出现的位置的中位数是多少,强制在线 主席树 询问区间内不同的数的个数 树上二分找到那个中位数 # inclu ...

随机推荐

  1. 以面试官的角度看strcpy函数

    一:笔试或者面试的总结 之 一 (1)在笔试或者面试中常常会被问道,strcpy  memmove memcpy 函数的实现.有时也会问你STL 中string的 split 和 trim的实现.有的 ...

  2. 使用正則表達式对URL进行解析

    对URL进行解析,一般用到的參数有: 1.协议 如http,https 2.域名或IP 3.port号,如7001,8080 4.Web上下文 5.URI.请求资源地址 6.请求參数 一个URL演示样 ...

  3. Odoo configuration items

    [options] addons_path = /opt/odoo/server/openerp/addons,/opt/odoo/server/addons admin_passwd = admin ...

  4. Linux基础(4)-硬盘分区、格式化及文件系统的管理、软件包的管理、yum管理RPM包和python的源码安装

    一: 1)  开启Linux系统前添加一块大小为15G的SCSI硬盘 2)  开启系统,右击桌面,打开终端 3)  为新加的硬盘分区,一个主分区大小为5G,剩余空间给扩展分区,在扩展分区上划分1个逻辑 ...

  5. Zoj2421 广搜

    <span style="color:#330099;">/* M - 广搜 加强 Time Limit:2000MS Memory Limit:65536KB 64b ...

  6. Intel平台map

  7. 随便写一点最近开发遇到的问题和解决方法 大部分关于laravel和php

    laravel里要想对对象进行自己设计的排序(usort()), 得用匿名方法,  原声php就不用 php里面可以随便写html代码,  比如可以把html直接后缀名改成.php, 然后在任何地方& ...

  8. 轻松搞定RabbitMQ(二)——工作队列之消息分发机制

    转自 http://blog.csdn.net/xiaoxian8023/article/details/48681987 上一篇博文中简单介绍了一下RabbitMQ的基础知识,并写了一个经典语言入门 ...

  9. (八):构建WineLib DLL

    (一):介绍 出于某些原因,你可能会发现你想要和使用Windows DLL一样使用你的Linux库.对于这有一些原因例如以下: 你正在支持一个使用多个第三方库的大应用.该项目在Linux中是可用的,可 ...

  10. 用算法求N(N&gt;=3)之内素数的个数

    首先.我们谈一下素数的定义.什么是素数?除了1和它本身外,不能被其它自然数整除(除0以外)的数 称之为素数(质数):否则称为合数. 依据素数的定义,在解决问题上,一開始我想到的方法是从3到N之间每一个 ...