题目链接: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 线段树维护最大值的更多相关文章

  1. hdu 5316 Magician 线段树

    链接:http://acm.hdu.edu.cn/showproblem.php? pid=5316 Magician Time Limit: 18000/9000 MS (Java/Others)  ...

  2. HDU.5692 Snacks ( DFS序 线段树维护最大值 )

    HDU.5692 Snacks ( DFS序 线段树维护最大值 ) 题意分析 给出一颗树,节点标号为0-n,每个节点有一定权值,并且规定0号为根节点.有两种操作:操作一为询问,给出一个节点x,求从0号 ...

  3. 线段树(维护最大值):HDU Billboard

    Billboard Time Limit: 20000/8000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  4. Codeforces 777E(离散化+dp+树状数组或线段树维护最大值)

    E. Hanoi Factory time limit per test 1 second memory limit per test 256 megabytes input standard inp ...

  5. HDU 6406 Taotao Picks Apples 线段树维护

    题意:给个T,T组数据: 每组给个n,m:n个数,m个操作: (对序列的操作是,一开始假设你手上东西是-INF,到i=1时拿起1,之后遍历,遇到比手头上的数量大的数时替换(拿到手的算拿走),问最后拿走 ...

  6. hdu 5068 线段树维护矩阵乘积

    http://acm.hdu.edu.cn/showproblem.php?pid=5068 题意给的略不清晰 m个询问:从i层去j层的方法数(求连段乘积)或者修改从x层y门和x+1层z门的状态反转( ...

  7. 51nod 1376【线段树维护区间最大值】

    引自:wonter巨巨的博客 定义 dp[i] := 以数字 i(不是下标 i)为结尾的最长上升长度 然后用线段树维护 dp[i]: 每个节点维护 2 个信息,一个是当前区间的最大上升长度,一个是最大 ...

  8. HDU 6155 Subsequence Count 线段树维护矩阵

    Subsequence Count Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 256000/256000 K (Java/Oth ...

  9. hdu 4037 Development Value(线段树维护数学公式)

    Development Value Time Limit: 5000/3000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Others ...

随机推荐

  1. Svm算法原理及实现

    Svm(support Vector Mac)又称为支持向量机,是一种二分类的模型.当然如果进行修改之后也是可以用于多类别问题的分类.支持向量机可以分为线性核非线性两大类.其主要思想为找到空间中的一个 ...

  2. (十九)hashlib模块

    hashlib模块用于加密相关的操作,3.x里代替了md5模块和sha模块,主要提供 SHA1, SHA224, SHA256, SHA384, SHA512 ,MD5 算法 注意:md5和sha25 ...

  3. python optparse模块的用法

    引用原博主文章链接: https://www.cnblogs.com/darkpig/p/5717902.html

  4. Kubernetes CoreDNS 状态是 CrashLoopBackOff 报错

    查看状态的时候,遇见coredns出现crashlookbackoff,首先我们来进行排错,不管是什么原因,查看coredns的详细信息,以及logs [root@k8s-master coredns ...

  5. (四)React Ant Design Pro + .Net5 WebApi:PostgreSQL数据库环境搭建

    一.简介 PostgreSQL,开源数据库(没听过小伙伴自己反思一下自行百度) PgAdmin,官方提供的数据库管理工具. 二.环境 1. 官网下载包,安装数据库 tar xjvf /app/pack ...

  6. Docker 如何动态修改容器端口映射

    Docker端口映射往往是Docker Run命令时通过-p将容器内部端口映射到宿主机的指定端口上,一般来说容器的端口所对应的端口是提前确定需要映射的.但是有些情况下不得不需要临时映射端口,例如Doc ...

  7. java进阶(33)--IO流

    一.IO流概念:1.基本概念2.IO流分类3.java.io流的四大家族4.流的close和flush方法5.java.id下常用的16个流 二.FileInputStream字节输入流1.FileI ...

  8. Django - WebSocket:dwebsocket

    Django - WebSocket:dwebsocket 什么是WebSocket WebSocket是一种在单个TCP连接上进行全双工通信的协议 WebSocket使得客户端和服务器之间的数据交换 ...

  9. 30分钟带你了解「消息中间件」Kafka、RocketMQ

    消息中间件的应用场景 主流 MQ 框架及对比 说明 Kafka 优点 Kafka 缺点 RocketMQ Pulsar 发展趋势 各公司发展 Kafka Kafka 是什么? Kafka 术语 Kaf ...

  10. 一种优化递归算法的方法(javascript)

    看书的时候看到了这个比较酷的方法,分享一下. 一.问题描述:代码如下,我们以计算阶乘(factorial)为例,当重复调用factorial(9),factorial(8),factorial(7)的 ...