19.Mex

给定一个长度为 N 的数组。你的程序需要按照输入顺序处理如下两种操作:

修改。修改带有两个参数 p 和 x,表示将数组中下标为 p 的元素修改为 x;
查询。查询操作带有两个参数 L, R,查询的结果为没有在数组中从下标为 L 的元素开始到下标为 R 的元素为止(两端均包含)的这个子数组中出现过的最小正整数。
输入

每个输入文件仅包含一组测试用例。

测试用例第一行为一个正整数 N (1 <= N <= 1’000’000),表示数组的长度。

第二行为 N 个以空格分隔的整数,表示初始数组的每一项。

第三行为一个整数 Q (1 <= Q <= 100’000),表示操作的数量。

从第四行开始的 Q 行,每行包含若干个以空格分隔的整数。其中第一个整数表示操作的类型,1表示修改操作,2表示查询操作;若操作为修改操作,接下来包含两个整数 p 和 x (0 <= p < N, 1 <= x <= 1’000’000’000),含义如前述所示;若操作为查询操作,接下来包含两个整数 L 和 R (0 <= L <= R < N),含义如前述所示。

保证数组中的每个元素在任意时刻均在 [1, 1’000’000’000] 范围内。

保证所有查询中子数组的长度之和不超过40’000’000。

输出

对于输入中的每一个查询操作,你的程序应该在单独的一行上输出查询的结果。

良心题,因为子区间长度不超过n,所以子区间没出现过的最小整数不会超过n+1。所以可以把子区间的数字放在新数组b中,然后用标记数组f将b中的数字出现情况标记一遍,最后输出没标记过的最小整数即可。

#include<stdio.h>
#include<string.h>
#define maxn 1000005
int a[maxn],b[maxn],f[maxn];
int n,q,p,x,L,R,op;
int main()
{
	int i;
	scanf("%d",&n);
	for(i=0;i<n;i++)
		scanf("%d",&a[i]);
	scanf("%d",&q);
	while(q--)
	{
		scanf("%d",&op);
		if(op==1)
		{
			scanf("%d%d",&p,&x);
			a[p]=x;
		}
		else if(op==2)
		{
			scanf("%d%d",&L,&R);
			for(i=L;i<=R;i++)
				b[i-L]=a[i];
			for(i=1;i<=n+1;i++)
				f[i]=0;
			for(i=0;i<=R-L;i++)
				if(b[i]>=1&&b[i]<=n+1)
					f[b[i]]=1;
			for(i=1;i<=n+1;i++)
			{
				if(f[i]==0)
				{
					printf("%d\n",i);
					break;
				}
			}
		}
	}
	return 0;
}

猜你喜欢

转载自blog.csdn.net/ArgentumHook/article/details/83240280
Mex