首页 > 其他分享 >浅谈区间MEX问题

浅谈区间MEX问题

时间:2023-01-02 18:11:07浏览次数:64  
标签:浅谈 int pos seg 区间 query MEX

区间MEX问题

MEX是什么

​ MEX:指一个序列中最小没有出现过的自然数。

​ 例如 \(mex\left\{1, 2, 0, 3\right\} = 4\),\(mex\left\{5, 1, 2, 3\right\} = 0\),\(mex\left\{0, 2, 1, 5\right\} = 3\)

​ 在谈到区间MEX之前,我们先实现一个维护MEX的数据结构。

​ 这个数据结构需要支持这样的操作:

  1. 对集合进行插入或删除
  2. 查询集合的MEX

​ 我们可以开一个 \(vis\) 数组,表示某个数是否出现过。那么往集合中插入一个数的时候,就把 \(vis_i\) 标记为 1 ,删除就标记为 0 。查询的时候,就从0开始暴力枚举,第一个 \(vis_i=0\) 的就是 MEX。显然,这样时间复杂度是非常高的,我们考虑优化一下这个暴力枚举的过程。我们可以维护一个 \(set\) ,表示未出现的数的集合,当我们添加一个新的数的时候,就在 \(set\) 中将其删去,如果删除操作使得一个数没有出现过,就把他加入\(set\) 。那么查询 MEX 的时候直接查询 \(set\) 中的最小值即可。

int cnt[N];
set<int> st;
int m, op, x;

void solve()
{
    for(int i = 0; i < N; i ++)
        st.insert(i);
    cin >> m;
    while(m --)
    {
        cin >> op;
        if(op == 0) //add
        {
            cin >> x;
           	if(cnt[x] == 0)
                st.erase(x);
            cnt[x] ++;
        }
        else if(op == 1) //query
        {
            cout << *st.begin() << '\n';
        }
        else //delete
        {
            cin >> x;
            if(-- cnt[x] == 0)
                st.insert(x);
	}
    }
}

复杂度为 \(O(nlogn)\)

注:在长度为 n 的集合中,MEX的最大值为 n,所以可以把大于 n 的都改为 \(n + 1\)。

区间MEX

在了解MEX的概念之后,我们引入一道题目来说明一下什么是区间MEX

Rmq Problem / mex

区间MEX,就是查询一个序列中指定区间的MEX,序列不带修。

方法1:离线 + 线段树

​ 这是一个区间问题,对于区间问题我们常常可以考虑是否能够通过离线来处理问题。线段树维护的是一个叫 \(last\_pos\) 的东西,表示某个数最后一次在数组中出现的位置,维护他的区间最小值。我们先对询问区间按右端点排序,遍历一遍 1 到 n ,把 \(a_i\) 更新到线段树中的同时,若 \(i = query[j].R\) ,这个时候就可以对 \(query[j]\) 的答案进行计算了。怎么计算呢?我们直接找到最后一次位置小于 \(query[j].l\) 的最小的数。那我们直接在线段树上二分就可以了。

#include <bits/stdc++.h>

using namespace std;
const int N = 300005;
struct ASK {
	int l, r, id, mex;
};
int n, m;
int a[N];
ASK Q[N];

struct Seg_tree1  {
    struct Tree {
    	int l, r;
    	int v;
    } seg[N << 2];
    void pushup(int k)
    {
        seg[k].v = min(seg[k << 1].v, seg[k << 1 | 1].v);
    }
    void build(int k, int l, int r)
    {
        seg[k] = {l, r, 0};
        if(l == r) {
            return;
        }   
        int mid = (l + r) >> 1;
        build(k << 1, l, mid);
        build(k << 1 | 1, mid + 1, r);
    }
    void modify(int k, int pos, int x)
    {
        int l = seg[k].l, r = seg[k].r;
        if(l == r) {
            seg[k].v = x;
            return;
        }
        int mid = (l + r) >> 1;
        if(pos <= mid)
            modify(k << 1, pos, x);
        else
            modify(k << 1 | 1, pos, x);
        pushup(k);
    }
    int query(int k, int pos)
    {
        int l = seg[k].l, r = seg[k].r;
        if(l == r) {
            return seg[k].l;
        }
        if(seg[k << 1].v < pos)
            return query(k << 1, pos);
        else
            return query(k << 1 | 1, pos);
    }
};

Seg_tree1 T1;

int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; i ++)
    {
    	scanf("%d", &a[i]), a[i] ++;
        a[i] = min(a[i], n + 2);
    }
    T1.build(1, 1, n + 2);

    for(int i = 1; i <= m; i ++)
    {
    	int l, r;
    	scanf("%d%d", &l, &r);
    	Q[i] = {l, r, i, 0};
    }

    sort(Q + 1, Q + m + 1, [](ASK a, ASK b) {return a.r < b.r;});

    for(int i = 1, j = 1; i <= n; i ++)
    {
    	T1.modify(1, a[i], i);
    	while(j <= m && Q[j].r == i)
    	{
    	    Q[j].mex = T1.query(1, Q[j].l);
       	    j ++;
    	}
    }

    sort(Q + 1, Q + m + 1, [](ASK a, ASK b) {return a.id < b.id;});
    for(int i = 1; i <= m; i ++)
    	printf("%d\n", Q[i].mex - 1);
}

方法2:在线 + 主席树

​ 如果学习过主席树的话,不难想到,跟上面离线的思想是一模一样的,对于一个询问 \([L, R]\) ,我们可以直接在 \(root[R]\) 上进行查询第一个最后一次出现的位置小于 \(L\) 的权值即可。

#include <bits/stdc++.h>

using namespace std;
const int N = 200005;

int tot = 0;
struct node {
    int ls, rs;
    int v;
} seg[N * 24];
int rt[N], a[N];
int n, m;

void pushup(int k)
{
    seg[k].v = min(seg[seg[k].ls].v, seg[seg[k].rs].v);
}
int modify(int old, int l, int r, int pos, int v)
{
    int p = ++tot;
    seg[p] = seg[old];
    if (l == r)
    {
        seg[p].v = v;
        return p;
    }
    int mid = (l + r) >> 1;
    if (pos <= mid)
        seg[p].ls = modify(seg[old].ls, l, mid, pos, v);
    else
        seg[p].rs = modify(seg[old].rs, mid + 1, r, pos, v);
    pushup(p);
    return p;
}
int query(int p, int l, int r, int pos)
{
    if (l == r)
    {
        return l;
    }
    int mid = (l + r) >> 1;
    if (seg[seg[p].ls].v < pos)
        return query(seg[p].ls, l, mid, pos);
    else
        return query(seg[p].rs, mid + 1, r, pos);
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);
    cin >> n >> m;
    for (int i = 1; i <= n; i++)
    {
        cin >> a[i];
        rt[i] = modify(rt[i - 1], 0, n, a[i], i);
    }
    while (m--)
    {
        int l, r;
        cin >> l >> r;
        cout << query(rt[r], 0, n, l) << '\n';
    }
}

题单:

ittle w and Discretization

C. The Third Problem

C. Meximum Array

D1. Balance (Easy version)

D2. Balance (Hard version)

D.MEX Tree

标签:浅谈,int,pos,seg,区间,query,MEX
From: https://www.cnblogs.com/DM11/p/17020308.html

相关文章

  • 浅谈Java并发
    Java并发是比较难的知识点,难于对并发的理解。并发要从操作系统和硬件层面去理解,才会比较深入,而不单单是从编程语言的逻辑去理解。首先对于并发要清楚的几点:线程可能在任......
  • 区间dp
    例题:石子合并设有\(N\)堆石子排成一排,其编号为\(1,2,3,…,N\)。每堆石子有一定的质量,可以用一个整数来描述,现在要将这\(N\)堆石子合并成为一堆。每次只能合并相邻的两......
  • 按编号区间拆分成行
    问题:编号区间为001-010,拆分成10行,如图所示。let源=Excel.CurrentWorkbook(){[Name="表1"]}[Content],最大编号=Table.AddColumn(源,"最大编号",eachif......
  • AcWing362. 区间
    题目描述给定\(n\)个区间\([a_i,b_i]\)和\(n\)个整数\(c_i\)。你需要构造一个整数集合\(Z\),使得\(\foralli\in[1,n]\),\(Z\)中满足\(a_i\lex\leb_i\)......
  • “Good-Exchanging” 软件开发经验浅谈
    需求分析本次开发将实现一个物资交换软件,具有以下基本功能:该程序允许添加物品的信息,删除物品的信息,显示物品列表,也允许查找物品的信息物品有公共的信息(物品名称,物品......
  • cf-1767C-Count Binary Strings(区间dp)
    题面https://codeforces.com/problemset/problem/1767/C下面展示带注释的ac代码在代码里解释思路Ac代码#include<bits/stdc++.h>#defineioios::sync_with_stdio(f......
  • 重叠区间、合并区间、插入区间问题详解
    判断区间是否重叠题目链接252.合并区间给定一个会议时间安排的数组intervals,每个会议时间都会包括开始和结束的时间intervals[i]=[starti,endi],请你判断一个人......
  • 浅谈C语言编译原理
    C语言我们在学习计算机学科时,往往最先接触到的编程语言是C,它是所有语言中,最接近底层的高级语言之一,因而它具有执行速度快的优点。但它又具有开发周期长和对于经验不足......
  • 浅谈如何召回流失用户
    对于负责用户运营的人员,用户流失是一个必须要关注的问题。如果没有及时发现用户流失的原因,及时采取相对应的策略进行干预和挽留,最终到了流失的末期,那么整个产品可能会宣告死......
  • 新年第一篇---算法浅谈
    一、前述2020是不平凡的一年。展望2021,希望大家都能有所收获。在此谈下算法方面的工作。二、工作类别目前算法工作的话,第一类是数据挖掘,它包含的知识,跟机器学习相关度会更大......