Codeforces Round #576 (Div. 1)B. Welfare State(思维/线段树)

该博客介绍了Codeforces Round #576 (Div. 1) B问题"Welfare State"的解题方法,主要聚焦于使用线段树来解决这个问题。题意是处理一个数组,包含两种操作:1. 修改特定位置的数值;2. 将所有小于指定值的数改为该值。博客提出关键在于关注每个元素最后一次1操作后的2操作,并利用线段树实现区间更新和单点更新,通过懒惰标记处理区间修改和单点修改,最终输出符合条件的数组。
摘要由CSDN通过智能技术生成

题目:Welfare State

题意:

一个数组,两种操作,然后输出最终数组,两种操作是:
1 p x: 表示将第 p 个数改为x;
2 x: 表示将所有小于 x 的数改成 x;

题解:

首先用线段树肯定可以做,就是 (区间更新 + 单点更新 )。
但这里我们先换个想法考虑这题:

  1. 题目要求只输出最终序列,那么对于一个数来说,对他有影响的只有他的最后一次 1 操作之后的 2 操作,因为之前不管怎么操作进行 1 操作之后都会变成一个固定的数,之前的 1 操作与 2 操作就不影响它了,影响它的只有之后的 2 操作中的最大那个操作值。
  2. 对于 2 操作,他只更改比它小的数,即如果之前的单点修改的值比他小的话,我们就相当于将该点改成 x。
  3. 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 >= 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值