hdu 5316 Magician 线段树维护最大值
题目链接:Magician
题意:
给你一个长度为n的序列v,你需要对这个序列进行m次操作,操作一共有两种,输入格式为
type a b
1、如果type==0,你就需要输出[a,b]区间内的美丽序列中所有元素的和,要使得这个值尽可能大
2、如果type==1,你就需要把a位置的元素值改为b
区间[a,b]的美丽序列就是va,va+1...vb。你需要从中取出任意个元素,这些元素的位置必须是奇偶交替
例如给你一个序列1,2,3,4,5,6,7
你取出来的美丽序列就有可能使1,2,3,4,5,6,7或者1,4,5,6,7或者2,5,6,7
题解:
我们使用线段树,如果type==1的时候就修改就可以了。对于type==0的情况。我们可以维护四个值,分别是区间[a,b]内的美丽序列:
从一个偶数位置开始,到一个奇数位置截至,我们使用ab来代替
从一个奇数位置开始,到一个奇数位置截至,我们使用bb来代替
从一个偶数位置开始,到一个偶数位置截至,我们使用aa来代替
从一个奇数位置开始,到一个偶数位置截至,我们使用ba来代替
对于线段树上一个节点维护的值,我们把这个节点称为a,把它的左右节点称为b,c
a.ab=max(b.aa+c.bb,b.ab+c.ab); 如果左右子树合并
a.ab=max(a.ab,max(b.ab,c.ab)); 如果左右子树不合并
其他四个值的维护也是这样
代码:
#include <map>
#include <set>
#include <list>
#include <queue>
#include <deque>
#include <cmath>
#include <stack>
#include <vector>
#include <bitset>
#include <cstdio>
#include <string>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int maxn = 1e5+10;
const int INF = 0x3f3f3f3f;
const double PI = 3.1415926;
const long long N = 1000006;
const double eps = 1e-10;
typedef long long ll;
#define qmh(x) ask()
#define mt(A, B) memset(A, B, sizeof(A))
#define lson L, mid, rt<<1
#define rson mid + 1, R, rt<<1|1
#define ls rt<<1
#define rs rt<<1|1
#define SIS std::ios::sync_with_stdiget_mod_new(z-x)o(false), cin.tie(0), cout.tie(0)
#define pll pair<long long, long long>
#define lowbit(abcd) (abcd & (-abcd))
#define max(a, b) ((a > b) ? (a) : (b))
#define min(a, b) ((a < b) ? (a) : (b))
struct node
{
ll aa,bb,ab,ba;
void Clear()
{
aa=bb=ab=ba=-INF;
}
} tree[400009],result;
int arr[100009]; void Merge(node &a,node b,node c)
{
a.aa=max(b.ab+c.aa,b.aa+c.ba);
a.aa=max(a.aa,max(b.aa,c.aa));
a.bb=max(b.ba+c.bb,b.bb+c.ab);
a.bb=max(a.bb,max(b.bb,c.bb));
a.ab=max(b.aa+c.bb,b.ab+c.ab);
a.ab=max(a.ab,max(b.ab,c.ab));
a.ba=max(b.bb+c.aa,b.ba+c.ba);
a.ba=max(a.ba,max(b.ba,c.ba));
} void build(int rt,int L,int R)
{
if(L==R)
{
tree[rt].Clear();
if(L&1) tree[rt].aa=arr[L];
else tree[rt].bb=arr[L];
return ;
}
int mid=(L+R)>>1;
build(rt<<1,L,mid),build(rt<<1|1,mid+1,R);
Merge(tree[rt],tree[rt<<1],tree[rt<<1|1]);
} void update(int rt,int L,int R,int pos,int val)
{
if(L==R)
{
tree[rt].Clear();
if(L&1) tree[rt].aa=val;
else tree[rt].bb=val;
return ;
}
int mid=(L+R)>>1;
if(pos<=mid) update(rt<<1,L,mid,pos,val);
else update(rt<<1|1,mid+1,R,pos,val);
Merge(tree[rt],tree[rt<<1],tree[rt<<1|1]);
} void query(int rt,int L,int R,int LL,int RR)
{
if(L>=LL&&R<=RR)
{
Merge(result,result,tree[rt]);
return ;
}
int mid=(L+R)>>1;
if(mid>=LL) query(rt<<1,L,mid,LL,RR);
if(RR>mid) query(rt<<1|1,mid+1,R,LL,RR);
}
int main()
{
int t;
scanf("%d",&t);
while(t--)
{
int n,m;
scanf("%d%d",&n,&m);
for(int i=1; i<=n; i++) scanf("%d",&arr[i]);
build(1,1,n);
while(m--)
{
int type,l,r;
scanf("%d%d%d",&type,&l,&r);
if(type==1) update(1,1,n,l,r);
else
{
result.Clear();
query(1,1,n,l,r);
printf("%lld\n",max(max(result.aa,result.bb),max(result.ab,result.ba)));
}
}
}
return 0;
}
hdu 5316 Magician 线段树维护最大值的更多相关文章
- hdu 5316 Magician 线段树
链接:http://acm.hdu.edu.cn/showproblem.php? pid=5316 Magician Time Limit: 18000/9000 MS (Java/Others) ...
- HDU.5692 Snacks ( DFS序 线段树维护最大值 )
HDU.5692 Snacks ( DFS序 线段树维护最大值 ) 题意分析 给出一颗树,节点标号为0-n,每个节点有一定权值,并且规定0号为根节点.有两种操作:操作一为询问,给出一个节点x,求从0号 ...
- 线段树(维护最大值):HDU Billboard
Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- Codeforces 777E(离散化+dp+树状数组或线段树维护最大值)
E. Hanoi Factory time limit per test 1 second memory limit per test 256 megabytes input standard inp ...
- HDU 6406 Taotao Picks Apples 线段树维护
题意:给个T,T组数据: 每组给个n,m:n个数,m个操作: (对序列的操作是,一开始假设你手上东西是-INF,到i=1时拿起1,之后遍历,遇到比手头上的数量大的数时替换(拿到手的算拿走),问最后拿走 ...
- hdu 5068 线段树维护矩阵乘积
http://acm.hdu.edu.cn/showproblem.php?pid=5068 题意给的略不清晰 m个询问:从i层去j层的方法数(求连段乘积)或者修改从x层y门和x+1层z门的状态反转( ...
- 51nod 1376【线段树维护区间最大值】
引自:wonter巨巨的博客 定义 dp[i] := 以数字 i(不是下标 i)为结尾的最长上升长度 然后用线段树维护 dp[i]: 每个节点维护 2 个信息,一个是当前区间的最大上升长度,一个是最大 ...
- HDU 6155 Subsequence Count 线段树维护矩阵
Subsequence Count Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 256000/256000 K (Java/Oth ...
- hdu 4037 Development Value(线段树维护数学公式)
Development Value Time Limit: 5000/3000 MS (Java/Others) Memory Limit: 65768/65768 K (Java/Others ...
随机推荐
- node中 path.join 和 path.resovle 区别
console.log(__dirname) console.log('----') console.log(path.resolve(__dirname, '/a/b', '../')) conso ...
- Maven学习笔记之第一个Maven项目(Linux)
Maven是Apache旗下的管理Java项目jar包的项目管理工具,有了它可以很方便构建和管理我们的Java项目,你不必在互联网上逐个查找你需要的第三方jar包,你只需在maven reposito ...
- 【Oracle】生成随机数
Oracle生成随机数: dbms_random.string(opt, 6) --括号里的opt要从下面的列表中选择,数字代表要生成几位随机数,如果是1位的话,就改成1 以此类推 opt可取 ...
- ctfhub技能树—RCE—过滤空格
打开靶机 查看页面信息 开始尝试注入 127.0.0.1 || ls 尝试绕过 127.0.0.1||ls 使用cat命令查看flag 127.0.0.11||cat<flag_10872536 ...
- 使用idea插件识别log文件的相关设置
最近要读一些spring boot项目产生的log文件,众所周知,idea拥有强大的插件系统.当我打开log文件时,idea自动帮我推荐了ideolog这个插件. 但是当我安装好之后发现系统并不能完全 ...
- 入门OJ:扫雪
扫雪1 题目描述 大雪履盖了整个城市,市政府要求冬季服务部门尽快将一些街道(列在一份清单中)的积雪清除掉以恢复交通,整个城市由许多交叉路口和街道构成,当然任意两个交叉路口都是直接或间接连通的,清单给出 ...
- 微信登录4-开发回调URL
一.准备 1.引入pom依赖 在要使用HttpClient的项目中加入依赖 <!--httpclient--> <dependency> <groupId>org. ...
- jemeter断言和性能分析
一.添加断言 1.原因:检查是否有该结果,一般一个请求过去除了400和500的只要通过的都会代表请求成功,比如登录页面及时填写了错误密码,虽然会返回密码错误,但这个请求还是成功的,所以我们要添加断言, ...
- 任何Python线程执行前,必须先获得GIL锁,然后,每执行100条字节码,解释器就自动释放GIL锁,让别的线程有机会执行
任何Python线程执行前,必须先获得GIL锁,然后,每执行100条字节码,解释器就自动释放GIL锁,让别的线程有机会执行 多线程 - 廖雪峰的官方网站 https://www.liaoxuefeng ...
- Shell 简单入门教程
大数据开发岗为什么要学习Shell呢?1)需要看懂大数据运维岗人员编写的Shell程序.2)偶尔会编写一些简单Shell程序来管理集群.提高开发效率 艺多不压身 Shell是一个命令行解释器,它接受应 ...