HDU 5316 Magician (线段树,单值修改,微变形)
题意:给一个初始序列A[1,n],第j个数字代表精灵j的power值,有两种操作:(1)查询区间[L,R] (2)修改某个精灵的power值。
但,查询的是区间[L,R]中一个美丽子序列sub[l,r]的和,美丽子序列是从A[L,R]中挑出的一些数字,这些数字按升序排好序,每两个相邻数字(sub[a],sub[b])在A[L,R]中的下标的的奇偶性不同。sub中至少有一个元素。power值可能为负。
思路:
单值修改的线段树,比较简单的模板。主要的问题在,奇偶性如何满足?
对于一个区间,要从中挑出一些数且满足相邻两数的下标奇偶性不同,那么就有4种选择(奇,奇)(奇,偶)(偶,奇)(偶,偶),“奇偶”代表其开头和结束位置的下标的奇偶性。现在又有问题了,如何合并区间?
以奇结尾的就必须以偶开头,这样全部组合一下,继续维护这4个值即可。比如(奇,奇)+(偶,偶),或(奇,偶)+(偶,奇)这些组合都行。该注意的是,因为power还有负值,又必须选1个,所以还可能(奇,奇)被两个区间(奇,奇)(奇,奇)其中的一个更新,(偶,偶)同理。
问题又来了,查询的时候,怎么返回?
那就绑成一个结构体返回。又要注意的是,可能[L,R]必须拆成两个区间时,合并区间的操作又要再做一次再返回这个结构体。
#include <bits/stdc++.h>
#define LL long long
using namespace std;
const int N=; struct node
{
long long a[][]; //1表示奇数,0表示偶数
bool flag;
node(){a[][]=a[][]=a[][]=a[][]=-1e18;flag=;}
}point[N*]; void get_val(int cur)
{
for(int i=; i<; i++)
{
for(int j=; j<; j++)
{
point[cur].a[i][j]=max( point[cur*].a[i][j], point[cur*+].a[i][j]);
point[cur].a[i][j]=max( point[cur].a[i][j], point[cur*].a[i][]+point[cur*+].a[][j] );
point[cur].a[i][j]=max( point[cur].a[i][j], point[cur*].a[i][]+point[cur*+].a[][j] );
}
}
} void create_tree(int l, int r, int cur) //cur为当前结点下标
{
if(l==r)
{
LL tmp;
scanf("%lld",&tmp);
point[cur]=node();
point[cur].a[l&][l&]=tmp;
return ;
}
int mid=(l+r)/;
create_tree(l, mid, cur*);
create_tree(mid+, r, cur*+);
get_val(cur); //以孩子的值来更新自身的值
} void update(int l,int r,int cur,int a,LL w)
{
if(l==r)
{
point[cur]=node();
point[cur].a[l&][l&]=w;
return ;
}
int mid=(l+r)/;
if(a<=mid) update(l, mid, cur*, a, w); //在左边
else update(mid+, r, cur*+, a, w);
get_val(cur); //回溯时更新自身
} node query(int l,int r, int L,int R,int cur)
{
if(l==L && r==R) return point[cur];
int mid=(L+R)/;
if(r<=mid) return query(l,r, L,mid, cur*);
else if(l>mid) return query(l,r, mid+,R, cur*+);
else
{
node tmp=node(), a=node(), b=node();
a=query(l,mid, L,mid,cur*);
b=query(mid+,r, mid+,R,cur*+);
for(int i=; i<; i++)
{
for(int j=; j<; j++)
{
tmp.a[i][j]=max( a.a[i][j], b.a[i][j]);
tmp.a[i][j]=max( tmp.a[i][j], a.a[i][]+b.a[][j] );
tmp.a[i][j]=max( tmp.a[i][j], a.a[i][]+b.a[][j]);
}
}
return tmp;
}
} int main()
{
//freopen("input.txt", "r", stdin);
int t, n, q, L, R, op;
LL W;
cin>>t;
while(t--)
{
cin>>n>>q;
create_tree(, n, );
for(int i=; i<q; i++)
{
scanf("%d",&op);
if(op==) //修改
{
scanf("%d %lld", &L, &W);
update(, n, , L, W);
}
else //查询
{
scanf("%d %d", &L, &R);
node tmp=query( L, R, , n, );
LL ans1=max(tmp.a[][],tmp.a[][]);
LL ans2=max(tmp.a[][],tmp.a[][]);
printf("%lld\n",max(ans1,ans2) );
}
}
}
return ;
}
AC代码
HDU 5316 Magician (线段树,单值修改,微变形)的更多相关文章
- hdu 5316 Magician 线段树维护最大值
题目链接:Magician 题意: 给你一个长度为n的序列v,你需要对这个序列进行m次操作,操作一共有两种,输入格式为 type a b 1.如果type==0,你就需要输出[a,b]区间内的美丽序列 ...
- hdu 5316 Magician 线段树
链接:http://acm.hdu.edu.cn/showproblem.php? pid=5316 Magician Time Limit: 18000/9000 MS (Java/Others) ...
- HDU 5475(2015 ICPC上海站网络赛)--- An easy problem(线段树点修改)
题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=5475 Problem Description One day, a useless calculato ...
- 题解报告:hdu 1698 Just a Hook(线段树区间修改+lazy懒标记的运用)
Problem Description In the game of DotA, Pudge’s meat hook is actually the most horrible thing for m ...
- HDU - 1754 I Hate It (线段树点修改求最大值)
题意:有N个学生M条操作,0<N<=200000,0<M<5000,要么查询某区间内学生的最高分,要么更改某学生的成绩. 分析:原理和线段树点修改求和类似. #include& ...
- I Hate It(线段树点修改区间查询)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 I Hate It Time Limit: 9000/3000 MS (Java/Others) ...
- lintcode:线段树的修改
线段树的修改 对于一棵 最大线段树, 每个节点包含一个额外的 max 属性,用于存储该节点所代表区间的最大值. 设计一个 modify 的方法,接受三个参数 root. index 和 value.该 ...
- hdu 4031 attack 线段树区间更新
Attack Time Limit: 5000/3000 MS (Java/Others) Memory Limit: 65768/65768 K (Java/Others)Total Subm ...
- hdu 4288 离线线段树+间隔求和
Coder Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Su ...
- hdu 3016 dp+线段树
Man Down Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total S ...
随机推荐
- HDU2295 Radar (DLX)
下面的代码99%参考了这个网站http://www.cnblogs.com/183zyz/archive/2011/08/07/2130193.html 人生的第一道DLX肯定是需要作一些参考的啦. ...
- D&F学数据结构系列——前驱和后继
前驱和后继 本文所述为二叉排序树的前驱和后继,如果想了解二叉排序树的概念,可以参考我的博文http://www.cnblogs.com/sage-blog/p/3864640.html 给定一个二叉查 ...
- 屏蔽wordpress升级提示
根据自己的需要,挑选适合的代码放在主题的functions.php文件中就可以了 /* 去除 WordPress 後台升級提示 */ // 2.8 ~ 2.9: add_filter('pre_tra ...
- 机器学习之单变量线性回归(Linear Regression with One Variable)
1. 模型表达(Model Representation) 我们的第一个学习算法是线性回归算法,让我们通过一个例子来开始.这个例子用来预测住房价格,我们使用一个数据集,该数据集包含俄勒冈州波特兰市的住 ...
- HTTP常用的状态码
一.200状态码: 成功2××:成功处理了请求的状态码. 1.200 :服务器已成功处理了请求并提供了请求的网页. 2.204:服务器成功处理了请求,但没有返回任何内容. 二.300状态码: 重定向3 ...
- JavaPersistenceWithHibernate第二版笔记-第五章-Mapping value types-002使用@Embeddable
一.数据库 二.代码 1. package org.jpwh.model.simple; import javax.persistence.Column; import javax.persisten ...
- Struts2笔记——与ServletAPI解耦
与ServletAPI解耦的访问方式 为了避免与 Servlet API 耦合在一起, 方便 Action 做单元测试, Struts2 对 HttpServletRequest, HttpSessi ...
- os 计算机的启动
零.boot的含义 先问一个问题,”启动”用英语怎么说? 回答是boot.可是,boot原来的意思是靴子,”启动”与靴子有什么关系呢? 原来,这里的boot是bootstrap(鞋带)的缩写,它来自一 ...
- Linux 关机命令详解
在linux下一些常用的关机/重启命令有shutdown.halt.reboot.及init,它们都可以达到重启系统的目的,但每个命令的内部工作过程是不同的,通过本文的介绍,希望你可以更加灵活的运用各 ...
- 使用 DB Fixtures 为 Unit Test 提供基础数据,Sails + Mocha 实现。
使用 DB Fixtures 为 Unit Test 提供基础数据,Sails + Mocha 实现. 问题:Test Fixture 太分散,管理麻烦. 在做单元测试的时候,数据回滚是个比较麻烦的问 ...