我是靠谱客的博主 冷酷小白菜,最近开发中收集的这篇文章主要介绍【bzoj4636】【蒟蒻的数列】【线段树】,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

Description

蒟蒻DCrusher不仅喜欢玩扑克,还喜欢研究数列
题目描述
DCrusher有一个数列,初始值均为0,他进行N次操作,每次将数列[a,b)这个区间中所有比k小的数改为k,他想知
道N次操作后数列中所有元素的和。他还要玩其他游戏,所以这个问题留给你解决。

Input

第一行一个整数N,然后有N行,每行三个正整数a、b、k。
N<=40000 , a、b、k<=10^9

Output

一个数,数列中所有元素的和

Sample Input

4
2 5 1
9 10 4
6 8 2
4 6 3

Sample Output

16

题解: 直接在线段树上打标记,最后dfs一遍整棵线段树即可.

因为区间长度比较大,所以需要动态开点.

注意一些没有建立的节点也是有贡献的,要把这些加进去.

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#define N 40010
#define inf 1000000000
using namespace std;
int n,a,b,k,cnt,root,ls[N*40],rs[N*40],v[N*40];
long long ans;
int read(){
  int x(0);char ch=getchar();
  while (ch<'0'||ch>'9') ch=getchar();
  while (ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
  return x;
}
void insert(int &k,int l,int r,int ll,int rr,int x){
  if (!k) k=++cnt;
  int mid=(l+r)>>1;
  if (l==ll&&r==rr){
    v[k]=max(v[k],x);
    //cout<<l<<' '<<r<<' '<<v[k]<<endl;
    return;
  }
  if (rr<=mid) insert(ls[k],l,mid,ll,rr,x);
  else if (mid<ll) insert(rs[k],mid+1,r,ll,rr,x);
  else{
    insert(ls[k],l,mid,ll,mid,x);
    insert(rs[k],mid+1,r,mid+1,rr,x);
  }
}
void query(int k,int l,int r,int x){
  int mid=(l+r)>>1;
  if (!k) return;
  v[k]=max(v[k],x);
  if (!ls[k]&&!rs[k]){
    //cout<<l<<' '<<r<<' '<<v[k]<<endl;
    ans+=(long long)(r-l+1)*v[k];
    return;
  }
  query(ls[k],l,mid,v[k]);
  query(rs[k],mid+1,r,v[k]);
  if (!ls[k]) ans+=(long long)(mid-l+1)*v[k];
  if (!rs[k]) ans+=(long long)(r-mid)*v[k]; 
}
int main(){
  //freopen("a.in","r",stdin);
  n=read(); 
  for (int i=1;i<=n;i++){
    a=read();b=read();k=read();
    if (a==b) continue;
    insert(root,1,inf,a,b-1,k);
  }
  query(root,1,inf,0);
  cout<<ans<<endl;
}


最后

以上就是冷酷小白菜为你收集整理的【bzoj4636】【蒟蒻的数列】【线段树】的全部内容,希望文章能够帮你解决【bzoj4636】【蒟蒻的数列】【线段树】所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(55)

评论列表共有 0 条评论

立即
投稿
返回
顶部