题目:Welfare State
题意:
一个数组,两种操作,然后输出最终数组,两种操作是:
1 p x: 表示将第 p 个数改为x;
2 x: 表示将所有小于 x 的数改成 x;
题解:
首先用线段树肯定可以做,就是 (区间更新 + 单点更新 )。
但这里我们先换个想法考虑这题:
- 题目要求只输出最终序列,那么对于一个数来说,对他有影响的只有他的最后一次 1 操作之后的 2 操作,因为之前不管怎么操作进行 1 操作之后都会变成一个固定的数,之前的 1 操作与 2 操作就不影响它了,影响它的只有之后的 2 操作中的最大那个操作值。
- 对于 2 操作,他只更改比它小的数,即如果之前的单点修改的值比他小的话,我们就相当于将该点改成 x。
- 2 操作只对之前的单点操作有影响,对在他之后的操作无影响。那么我们可以把所有单点操作后,记录操作的序号,最后从头判断每个位置的最后一次1操作之后的2操作的最大值比不比他本身的值大,大就更改,不大就不更改。
代码:
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2E5 + 10;
int a[N], b[N], last[N];
int main() {
int n, q;
scanf("%d", &n);
for ( int i = 1; i <= n; i ++ ) scanf("%d",&a[i]);
scanf("%d", &q);
for( int i = 1; i <= q; i ++ ) {
int t, p, x;
scanf("%d", &t);
if( t==1 ) {
scanf("%d%d", &p, &x);
a[p] = x;
last[p] = i; //存储最后一个对进行1操作的序号
}
else {
scanf("%d", &b[i]);
}
}
for ( int i = q-1; i >=