Gorgeous Sequence 题解 (小清新线段树)
这道题被学长称为“科幻题”
事实上,并不是做法科幻,而是“为什么能这么做?”的解释非常科幻
换句话说,复杂度分析灰常诡异以至于吉如一大佬当场吃书
线段树维护的量:区间和sum,区间最大值max1,区间次大值max2,最大值出现次数cnt。
现在假设区间[l,r]对x取min,那么有如下三种情况:
1.max1<=x,不用修改,return ;
2.max2<x<max1,修改只会影响所有最大值,sum+=cnt*(max1-x),更新max1打标记;
3.max2>=x,暴力递归修改左右儿子.
吃书后只能保证复杂度O(nlog2n)
#include<cstdio>
#include<cstring>
#define ls(k) k<<1
#define rs(k) k<<1|1 const int L=<<|;
char buffer[L],*S,*TT;
#define getchar() ((S==TT&&(TT=(S=buffer)+fread(buffer,1,L,stdin),S==TT))?EOF:*S++) const int N=;
typedef long long ll;
int max(int a,int b){return a>b?a:b;}
int min(int a,int b){return a<b?a:b;}
int T,n,m,a[N];
struct segment_tree
{
int max1[N<<],max2[N<<],cnt[N<<];
ll sum[N<<];
void up(int k)
{
sum[k]=sum[ls(k)]+sum[rs(k)];
max1[k]=max(max1[ls(k)],max1[rs(k)]);
max2[k]=max(max2[ls(k)],max2[rs(k)]);
cnt[k]=;
if(max1[ls(k)]!=max1[rs(k)])max2[k]=max(max2[k],min(max1[ls(k)],max1[rs(k)]));
cnt[k]+=(max1[k]==max1[ls(k)]?cnt[ls(k)]:);
cnt[k]+=(max1[k]==max1[rs(k)]?cnt[rs(k)]:);
}
void tag(int k,int val)
{
if(val>=max1[k])return ;
sum[k]+=(ll)(val-max1[k])*cnt[k];
max1[k]=val;
}
void down(int k)
{
tag(ls(k),max1[k]);
tag(rs(k),max1[k]);
}
void build(int k,int l,int r)
{
if(l==r)
{
sum[k]=max1[k]=a[l];
cnt[k]=;max2[k]=-;
return ;
}
int mid=l+r>>;
build(ls(k),l,mid);
build(rs(k),mid+,r);
up(k);
}
void change(int k,int l,int r,int L,int R,int val)
{
if(val>=max1[k])return ;
if(L<=l&&R>=r&&val>max2[k])
{
tag(k,val);
return ;
}
int mid=l+r>>;down(k);
if(L<=mid)change(ls(k),l,mid,L,R,val);
if(R>mid)change(rs(k),mid+,r,L,R,val);
up(k);
}
ll qsum(int k,int l,int r,int L,int R)
{
if(L<=l&&R>=r)return sum[k];
int mid=l+r>>;
down(k);ll res=;
if(L<=mid)res+=qsum(ls(k),l,mid,L,R);
if(R>mid)res+=qsum(rs(k),mid+,r,L,R);
return res;
}
int qmax(int k,int l,int r,int L,int R)
{
if(L<=l&&R>=r)return max1[k];
down(k);
int mid=l+r>>,ans=-;
if(L<=mid)ans=max(ans,qmax(ls(k),l,mid,L,R));
if(R>mid)ans=max(ans,qmax(rs(k),mid+,r,L,R));
return ans;
}
}seg;
inline int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>'')
{if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<='')
{x=(x<<)+(x<<)+ch-'';ch=getchar();}
return x*f;
}
void work()
{
n=read();m=read();
for(int i=;i<=n;i++)a[i]=read();
seg.build(,,n);
while(m--)
{
int op=read(),l=read(),r=read();
if(op==)
{
int val=read();
seg.change(,,n,l,r,val);
}
else if(op==)printf("%d\n",seg.qmax(,,n,l,r));
else if(op==)printf("%lld\n",seg.qsum(,,n,l,r));
}
}
int main()
{
T=read();
while(T--)work();
return ;
}
Gorgeous Sequence 题解 (小清新线段树)的更多相关文章
- [HDU5360]:Gorgeous Sequence(小清新线段树)
题目传送门 题目描述: (原题英文) 操作0:输入l,r,t,线段树区间与t取min. 操作1:输入l,r,区间取最大值. 操作2:输入l,r,区间求和. 输入格式: 第一行一个整数T,表示数据组数: ...
- [BZOJ3211]:花神游历各国(小清新线段树)
题目传送门 题目描述: 花神喜欢步行游历各国,顺便虐爆各地竞赛.花神有一条游览路线,它是线型的,也就是说,所有游历国家呈一条线的形状排列,花神对每个国家都有一个喜欢程度(当然花神并不一定喜欢所有国家) ...
- [CSP-S模拟测试]:联(小清新线段树)
题目描述 由于出题人懒所以没有背景.一个无限长的$01$序列,初始全为$0$,每次选择一个区间$[l,r]$进行操作,有三种操作:$\bullet 1\ l\ r$将$[l,r]$中所有元素变成$1$ ...
- bzoj4355 Play with sequence(吉司机线段树)题解
题意: 已知\(n\)个数字,进行以下操作: \(1.\)区间\([L,R]\) 赋值为\(x\) \(2.\)区间\([L,R]\) 赋值为\(max(a[i] + x, 0)\) \(3.\)区间 ...
- [NOIP10.6模拟赛]2.equation题解--DFS序+线段树
题目链接: 咕 闲扯: 终于在集训中敲出正解(虽然与正解不完全相同),开心QAQ 首先比较巧,这题是\(Ebola\)出的一场模拟赛的一道题的树上强化版,当时还口胡出了那题的题解 然而考场上只得了86 ...
- 一篇文教你使用python Turtle库画出“精美碎花小清新风格树”快来拿代码!
Turtle库手册可以查询查询 python图形绘制库turtle中文开发文档及示例大全,手册中现有示例,不需要自己动手就可以查看演示. 使用Turtle画树,看了一下网上的代码,基本上核心的方法是使 ...
- 洛谷题解P4314CPU监控--线段树
题目链接 https://www.luogu.org/problemnew/show/P4314 https://www.lydsy.com/JudgeOnline/problem.php?id=30 ...
- Almost Regular Bracket Sequence CodeForces - 1095E (线段树,单点更新,区间查询维护括号序列)
Almost Regular Bracket Sequence CodeForces - 1095E You are given a bracket sequence ss consisting of ...
- HDU5306:Gorgeous Sequence——题解
http://acm.hdu.edu.cn/showproblem.php?pid=5306 给一个数组,m次操作: 1:l r x,将a[i](l<=i<=r)=min(a[i],x) ...
随机推荐
- ngTbale真分页实现排序、搜索等功能
一. 真分页表格基础 1. 需求:分页,排序,搜索都是需要发API到服务端. 2. JS实现代码: getStorage是localStorage一个工具方法,可以自己写这个方法. API参数如下: ...
- 第十五周oj刷题——Problem M: C++习题 矩阵求和--重载运算符
Description 有两个矩阵a和b,均为2行3列.求两个矩阵之和.重载运算符"+",使之能用于矩阵相加(如c=a+b). 重载流插入运算符"<<&quo ...
- LeetCode234_PalindromeLinkedList (推断是否为回文链表) Java题解
题目: Given a singly linked list, determine if it is a palindrome. Follow up: Could you do it in O(n) ...
- Cocos2d-x 3.x 图形学渲染系列十五
笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家.特邀编辑,畅销书作者,国家专利发明人;已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D ...
- Arduino程序-光敏电阻
尽管造书去做的.但还是有莫名的成就感 从串口显示出,电压变化, void setup() { // put your setup code here, to run once: Serial. ...
- 专注UI——是alert()打败了你!
在上家公司.常常在页面上写aler()提示代码.没有认为有什么,好寻常.认为提示就本来应该是这种,可是,当我到了这家公司.在測试的时候,由于測试人员看到了一个aler弹出框.结果我的页面被退回重写,后 ...
- LeetCode 648. Replace Words (单词替换)
题目标签:HashMap 题目给了我们一个array 的 root, 让我们把sentence 里面得每一个word 去掉它得 successor. 把每一个root 存入hash set,然后遍历s ...
- Hdu5303 Delicious Apples 贪心
题目链接: HDU5303 题意: 有一条环形的长为L的路,仓库在位置0处, 这条路上有n棵苹果树,给出每棵苹果树的位置和苹果数量, 问用 一次最多能装K个苹果的篮子 把这条路上全部苹果採回仓库最 ...
- Codeforces Round #273 (Div. 2)C. Table Decorations 数学
C. Table Decorations You have r red, g green and b blue balloons. To decorate a single table for t ...
- Ubuntu 查看当前目录使用的总空间大小
查看当前目录使用的总空间大小 du -h --max-depth=0 #du -h --max-depth=0 217M . 查看当前目录使用总空间的大小以及当前目录下一级文件及文件夹各自使用的总空间 ...