2019CCPC网络选拔赛 array(权值线段树)

2019CCPC网络选拔赛1002 HDU6703

题目大意:

T个样例。给你一个长度为n的数组a,1≤a[i]≤n,a[i]各不相同。m个操作。ans初始为0。有两种操作:

操作1:给你t1。pos=t1^ans。把数组下标为pos的数,数值+1e7;

操作2:给你t2,t3。r=t2^ans,k=t3^ans。输出**与数组下标1~r的数不同**且**不小于k**的最小数。更新ans。

数据范围:

1≤T≤10,1≤n≤1e5,1≤m≤1e5,1≤a[i]≤n,a[i]各不相同,1≤pos≤n,1≤r≤n,1≤k≤n。

∑n≤510,000,∑m≤510,000。

赛后补题。

此处照搬标准题解:

因为数组中的值唯一,且在1到n的范围内,而询问的r和k也在1到n的范围内。 所以对于任意一个被操 作1修改过的值都不会成为询问的答案,而询问的结果也必然在k到n+1的范围内。 因为没有被修改过 值是唯一的,所以可以建立权值线段树,维护权值区间内的值所在下标的最大值。而询问则转化为不小 于k的值里面,下标超过r的最小权值是多少。 如何处理询问呢,一种较为暴力的解法是直接在线段树上 询问权值在k到n+1的范围内第一个下标超过r的权值是多少。但复杂度可能会被卡,需要减枝。 再加上 一个额外的判断就可以了,就是在递归查询完左子树内存不存在大于r的下标之后,如果不存在,则先 看一下右子树内的下标的最大值是否大于r。如果不大于r,则不必再进入右子树内查询,否则答案一定 在右子树内。在进左子树之前也利用同样的判断条件来判断是否有必要进入左子树,这样做可以保证单 次查询的复杂度是O(log n) 的。 而对于操作1,则等价于修改某个权值的下标为无穷大。操作复杂度也 是O(log n )的。 综上所述,得到了一个复杂度为O( m * log n )的在线算法,可以较快地通过此题。

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+50;
const int inf=0x3f3f3f3f;
struct P{
    int l,r,id;
}tree[maxn<<2];
void build(int l,int r,int k)
{
    tree[k].l=l;tree[k].r=r;
    if(tree[k].l==tree[k].r)
    {
        tree[k].id=0;
        return ;
    }
    int mid=(tree[k].l+tree[k].r)/2;
    build(l,mid,k*2);
    build(mid+1,r,k*2+1);
    tree[k].id=0;
}
void add(int x,int id,int k)
{
    if(tree[k].l==tree[k].r)
    {
        tree[k].id=id;return;
    }
    int mid=(tree[k].l+tree[k].r)/2;
    if(x<=mid)add(x,id,k*2);
    else add(x,id,k*2+1);
    tree[k].id=max(tree[k*2].id,tree[k*2+1].id);
}
int find(int l,int r,int k,int minid)
{
    if(tree[k].l==tree[k].r)
    {
        if(tree[k].id>minid)return l;
        return inf;
    }
    if(tree[k].id<=minid)return inf;
    int mid=(tree[k].l+tree[k].r)/2;
    int id1=tree[k*2].id,id2=tree[k*2+1].id;
    if(r<=mid)
    {
        if(id1<=minid)return inf;
        return find(l,r,k*2,minid);
    }
    if(l>mid)
    {
        if(id2<=minid)return inf;
        return find(l,r,k*2+1,minid);
    }
    int ans1=inf,ans2=inf,ans=inf;
    if(id1>minid)ans1=find(l,mid,k*2,minid);
    if(id2>minid&&ans1==inf)ans2=find(mid+1,r,k*2+1,minid);
    ans=min(ans1,ans2);
    return ans;
}
/*void f(int l,int r,int k)
{
    printf("##l:%d ##r:%d##id:%d\n",tree[k].l,tree[k].r,tree[k].id);
    if(tree[k].l==tree[k].r)return;
    int mid=(tree[k].l+tree[k].r)/2;
    f(l,mid,k*2);
    f(l,mid,k*2+1);
}*/
int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
        int n,m,op,t1,t2,t3,i,a[maxn],ans=0;
        scanf("%d%d",&n,&m);
        build(1,n,1);
        for(i=1;i<=n;i++)
        {
            scanf("%d",&a[i]);
            add(a[i],i,1);
        }
        //f(1,n,1);
        for(i=1;i<=m;i++)
        {
            scanf("%d",&op);
            if(op==1)
            {
                scanf("%d",&t1);
                add(a[t1^ans],inf,1);
                //f(1,n,1);
                continue;
            }
            scanf("%d%d",&t2,&t3);
            t2^=ans;t3^=ans;
            ans=find(t3,n,1,t2);
            //printf("@@r:%d k:%[email protected]@ans:",t2,t3);
            if(ans!=inf)printf("%d\n",ans);
            else{
                ans=n+1;
                printf("%d\n",ans);
            }
        }
        //puts("############");
    }
}

原文地址:https://www.cnblogs.com/kkkek/p/11441193.html

时间: 2024-08-30 02:33:01

2019CCPC网络选拔赛 array(权值线段树)的相关文章

[bzoj3932][CQOI2015]任务查询系统-题解[主席树][权值线段树]

Description 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第Ei秒后结束(第Si秒和Ei秒任务也在运行 ),其优先级为Pi.同一时间可能有多个任务同时执行,它们的优先级可能相同,也可能不同.调度系统会经常向 查询系统询问,第Xi秒正在运行的任务中,优先级最小的Ki个任务(即将任务按照优先级从小到大排序后取前Ki个 )的优先级之和是多少.特别的,如

P2234 [HNOI2002]营业额统计 (权值线段树)

P2234 [HNOI2002]营业额统计 题目描述 Tiger最近被公司升任为营业部经理,他上任后接受公司交给的第一项任务便是统计并分析公司成立以来的营业情况. Tiger拿出了公司的账本,账本上记录了公司成立以来每天的营业额.分析营业情况是一项相当复杂的工作.由于节假日,大减价或者是其他情况的时候,营业额会出现一定的波动,当然一定的波动是能够接受的,但是在某些时候营业额突变得很高或是很低,这就证明公司此时的经营状况出现了问题.经济管理学上定义了一种最小波动值来衡量这种情况: 当最小波动值越大

【BZOJ4605】崂山白花蛇草水 权值线段树+kd-tree

[BZOJ4605]崂山白花蛇草水 Description 神犇Aleph在SDOI Round2前立了一个flag:如果进了省队,就现场直播喝崂山白花蛇草水.凭借着神犇Aleph的实力,他轻松地进了山东省省队,现在便是他履行诺言的时候了.蒟蒻Bob特地为他准备了999,999,999,999,999,999瓶崂山白花蛇草水,想要灌神犇Aleph.神犇Aleph求(跪着的)蒟蒻Bob不要灌他,由于神犇Aleph是神犇,蒟蒻Bob最终答应了他的请求,但蒟蒻Bob决定将计就计,也让神犇Aleph回答

【bzoj2161】布娃娃 权值线段树

题目描述 小时候的雨荨非常听话,是父母眼中的好孩子.在学校是老师的左右手,同学的好榜样.后来她成为艾利斯顿第二代考神,这和小时候培养的良好素质是分不开的.雨荨的妈妈也为有这么一个懂事的女儿感到高兴.一次期末考试,雨荨不知道第多少次,再次考了全年级第一名.雨荨的妈妈看到女儿100分的成绩单时,脸上又泛起了幸福的笑容,作为奖励,她给雨荨买了n个布娃娃.细心的雨荨发现,第i个布娃娃有一个耐心值P[i]以及一个魅力值C[i],并且还有能够忍受的耐心值的上限R[i]以及下限L[i].当一个布娃娃j满足L[

模板——权值线段树(逆序对)

Ultra-QuickSort Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 62455   Accepted: 23259 Description In this problem, you have to analyze a particular sorting algorithm. The algorithm processes a sequence of n distinct integers by swappin

【bzoj4605】崂山白花蛇草水 权值线段树套KD-tree

题目描述 神犇Aleph在SDOI Round2前立了一个flag:如果进了省队,就现场直播喝崂山白花蛇草水.凭借着神犇Aleph的实力,他轻松地进了山东省省队,现在便是他履行诺言的时候了.蒟蒻Bob特地为他准备了999,999,999,999,999,999瓶崂山白花蛇草水,想要灌神犇Aleph.神犇Aleph求(跪着的)蒟蒻Bob不要灌他,由于神犇Aleph是神犇,蒟蒻Bob最终答应了他的请求,但蒟蒻Bob决定将计就计,也让神犇Aleph回答一些问题.具体说来,蒟蒻Bob会在一个宽敞的广场上

【bzoj3065】带插入区间K小值 替罪羊树套权值线段树

题目描述 从前有n只跳蚤排成一行做早操,每只跳蚤都有自己的一个弹跳力a[i].跳蚤国王看着这些跳蚤国欣欣向荣的情景,感到非常高兴.这时跳蚤国王决定理性愉悦一下,查询区间k小值.他每次向它的随从伏特提出这样的问题: 从左往右第x个到第y个跳蚤中,a[i]第k小的值是多少.这可难不倒伏特,他在脑袋里使用函数式线段树前缀和的方法水掉了跳蚤国王的询问.这时伏特发现有些跳蚤跳久了弹跳力会有变化,有的会增大,有的会减少.这可难不倒伏特,他在脑袋里使用树状数组套线段树的方法水掉了跳蚤国王的询问.(orz 主席

【bzoj2770】YY的Treap 权值线段树

题目描述 志向远大的YY小朋友在学完快速排序之后决定学习平衡树,左思右想再加上SY的教唆,YY决定学习Treap.友爱教教父SY如砍瓜切菜般教会了YY小朋友Treap(一种平衡树,通过对每个节点随机分配一个priority,同时保证这棵平衡树关于priority是一个小根堆以保证效率).这时候不怎么友爱的510跑了出来,他问了YY小朋友一个极不和谐的问题:怎么求Treap中两个点之间的路径长度.YY秒了之后决定把这个问题交给你来做,但只要求出树中两点的LCA. 输入 第一行两个整数n,m 第二行

权值线段树求逆序对问题

我们都知道,求逆序对数量可以用归并排序解决.但是用归并排序只能解决静态的序列问题,没有扩展的区间.因此就有了用权值线段树求逆序对的方法. 1 #include<iostream> 2 #include<iomanip> 3 #include<ctime> 4 #include<climits> 5 #include<algorithm> 6 #include<queue> 7 #include<vector> 8 #inc