3973: seq
3973: seq
题目描述
小y 的男朋友送给小y 一个数列{ai}{ai},并且刁难小y 要她维护这个序列。
具体而言,小y 的男朋友要求小y 完成两个操作:
1. 修改数列中的一个数
2. 设pipi表示maxij=1ajmaxj=1iaj,求出n∑i=1pi∑i=1npi。
小y 不会做,于是向你求助。
输入
第一行一个数nn表示数列长度。
第二行nn个由空格隔开的数表示数列aa。
第三行一个数mm表示修改数。
接下来mm行,每行两个数pos,valuepos,value,表示把aposapos改成valuevalue。
输出
mm行,每行一个数,表示对于每次修改后的n∑i=1pi∑i=1npi。
样例输入
<span style="color:#333333"><span style="color:#333333">10
114 357 904 407 100 624 449 897 115 846
20
5 357
6 350
2 939
9 1182
7 1062
2 3300
4 6867
4 2076
3 8458
9 6575
10 5737
10 338
9 10446
4 7615
2 5686
4 10091
1 6466
6 15551
3 10914
7 3234</span></span>
样例输出
<span style="color:#333333"><span style="color:#333333">7703
7703
8565
9051
9297
29814
54783
29814
71078
71078
71078
71078
75054
75054
77440
85605
92737
119327
123429
123429</span></span>
提示
对于前30%30%的数据,n,m≤5000n,m≤5000;
对于前60%60%的数据,n,m≤50000n,m≤50000;
对于100%100%的数据,n≤3×105,ai≤109n≤3×105,ai≤109。
来源
solution
传说中的套路题,可是我不会
理解了好久
首先最终每个数的答案一定是递增的,也就是每个数会被之前的某个数“盖住”
我们用线段树维护l~r的答案和最大值
这里的答案只考虑l~r,不受其他影响
那么修改很好实现,主要是怎么维护区间
分类
1.tree[k*2].max>=tree[k*2+1].max
也就是左边完全盖住了右边
那答案就是tree[k*2].max*tree[k*2+1].len+tree[k*2].sum
2.tree[k*2].max<tree[k*2+1].max
左边盖住右边的一小部分
再分类,设M为tree[k*2].max,ls为右儿子的左儿子,rs为右儿子的右儿子
(1)M<tree[ls].max
那么rs的答案是正确的,而左儿子未知
所以递归算左儿子,右儿子用减的
(2)M>tree[ls].max
左儿子被完全盖住,右儿子递归算
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define ll long long
#define maxn 300005
using namespace std;
int n,m,a[maxn],p,val;
struct node{
int l,r,len;ll Max,sum;
}tree[maxn*4];
ll pushup(int k,ll v){
if(v>tree[k].Max)return tree[k].len*v;
if(tree[k].l==tree[k].r)return max(tree[k].Max,v);
if(tree[k*2].Max>v){
return pushup(k*2,v)+tree[k].sum-tree[k*2].sum;
}//right remain
if(tree[k*2].Max<=v){
return tree[k*2].len*v+pushup(k*2+1,v);
}//left covered
}
void wh(int k){
tree[k].Max=max(tree[k*2].Max,tree[k*2+1].Max);
tree[k].sum=tree[k*2].sum+pushup(k*2+1,tree[k*2].Max);
}
void build(int k,int L,int R){
tree[k].l=L;tree[k].r=R;tree[k].len=R-L+1;
if(L==R){
tree[k].Max=tree[k].sum=a[L];return;
}
int mid=L+R>>1;
build(k*2,L,mid);build(k*2+1,mid+1,R);
wh(k);
}
void change(int k){
if(tree[k].l==tree[k].r){
tree[k].sum=tree[k].Max=val;
return ;
}
int mid=tree[k].l+tree[k].r>>1;
if(p<=mid)change(k*2);
else change(k*2+1);
wh(k);
}
int main()
{
cin>>n;
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
build(1,1,n);
cin>>m;
for(int i=1;i<=m;i++){
scanf("%d%d",&p,&val);
change(1);
printf("%lld\n",tree[1].sum);
}
return 0;
}
3973: seq的更多相关文章
- BZOJ1798: [Ahoi2009]Seq 维护序列seq[线段树]
1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec Memory Limit: 64 MBSubmit: 5504 Solved: 1937[Submit ...
- Linux 设备驱动程序 proc seq
不能再简化 #include<linux/module.h> #include<linux/init.h> #include<linux/seq_file.h> # ...
- seq
Linux 中seq 命令的用法 用于产生从某个数到另外一个数之间的所有整数 用法: seq [选项]... 尾数 或:seq [选项]... 首数 尾数 或:seq [选项]... 首数 增量 尾数 ...
- 20 seq 某个数到另外一个数之间的所有整数
seq命令Shell内建命令 seq命令用于产生从某个数到另外一个数之间的所有整数. 语法 : seq [选项]... 尾数 seq [选项]... 首数 尾数 seq [选项]... 首数 增量 尾 ...
- 【双标记线段树】bzoj1798维护序列seq
一.题目 描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,-,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列 ...
- seq 显示00 01的格式
for i in `seq -w 00 20` ; do echo $i ;done 00 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 ...
- RNA seq 两种计算基因表达量方法
两种RNA seq的基因表达量计算方法: 1. RPKM:http://www.plob.org/2011/10/24/294.html 2. RSEM:这个是TCGAdata中使用的.RSEM据说比 ...
- Linux常用命令之seq
标题:seq命令的使用 作用:seq命令用于以指定增量从首数开始打印数字到尾数,即产生从某个数到另外一个数之间的所有整数,并且可以对整数的格式.宽度.分割符号进行控制 语法: [1] seq [选项] ...
- CDOJ 1157 数列(seq) 分块+线段树
数列(seq) Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.uestc.edu.cn/#/problem/show/1157 Desc ...
随机推荐
- linux用命令行运行matlab的.mat文件
入m文件所在目录后,运行 $ matlab -nodesktop -nosplash -r matlabfile 只用文件名matlabfile,不能添加.m
- THinkPHP 5.0 域名路由
ThinkPHP支持完整域名.子域名和IP部署的路由和绑定功能 要启用域名部署路由功能,首先需要开启: 'url_domain_deploy' => true 定义域名部署规则支持两种方式:动态 ...
- windows下安装php依赖关系管理工具composer
1.安装Composer Composer是PHP的依赖管理工具之一,官方网站 http://getcomposer.org/ .它支持多种安装方式,对于在win下做开发的草来说,最便捷的方式就是下载 ...
- 图片url转base64
var xhr = new XMLHttpRequest() // 配置的代理,解决跨域问题 xhr.open('GET', url.replace('http://xxx.com', '/img') ...
- 第30题:LeetCode155. Min Stack最小栈
设计一个支持 push,pop,top 操作,并能在O(1)时间内检索到最小元素的栈. push(x) -- 将元素 x 推入栈中. pop() -- 删除栈顶的元素. top() -- 获取栈顶元素 ...
- STL笔记(に)--vector容器
Vector 1.可变长的动态数组 2.需包含头文件#include<vector> (当然,如果用了万能头文件#include<bits/stdc++.h>则可忽略) 3.支 ...
- C++ 学习笔记 (六) 继承- 子类与父类有同名函数,变量
学习了类的继承,今天说一下当父类与子类中有同名函数和变量时那么程序将怎么执行.首先明确当基类和子类有同名函数或者变量时,子类依然从父类继承. 举例说明: 例程说明: 父类和子类有同名的成员 data: ...
- Log错误日志级别
日志记录器(Logger)的级别顺序: 分为OFF.FATAL.ERROR.WARN.INFO.DEBUG.ALL或者您定义的级别.Log4j建议只使用四个级别,优先级 从高到低分别是 ERR ...
- 读取properties的简单方法,使用@Configuration
配置类代码如下 import org.springframework.beans.factory.annotation.Value; import org.springframework.contex ...
- jsp页面:一个form,不同请求提交form
需求:一个表单中有一个请求 action="url"发送数据地址: 在表单外有一个请求,请求form表单提交的数据 我们用js来写:通过每次请求传不同的action=url; 例如 ...