HDU 4893 2014多校三 线段树
给定一个初始都为0的序列,有三种操作,前两种比较正常,一个是对某个位置的数add k,另一个是query区间和。然后比较麻烦的是第三个操作,把某个区间里面的每个值改成离它最近的Fibonacci数,如果存在左右两个离它近的,优先取左边数值小的
一看到前两个操作马上就想上手敲树状数组,后来看到第三个就有点傻眼了,思维当时一直停留在怎么快速改值。。但忽略了题目本身要求什么,只有操作2才是输出,也就是只要求区间和值而且,其他两个都是操作而已,在聪哥的提醒下,知道对线段树开两个值记录+一个懒惰标记,一个值d记录正常的区间和值,另一个sf记录当前的fibanacci数值和
懒惰标记记录当前区间是否被3操作过,若被标记了,下次query就是返回sf,否则返回d。
不过一开始思路还是有点不清晰,还WA了几发。明显按上面的转化就是一个普通的单点更新+懒惰标记+区间查询的线段树了嘛
主要错在pushdown函数和3操作处理上。没分清模块
首先3操作命名为fix,fix找到对应的区间后,就进行懒惰标记,顺便把当前区间d改为sf值,然后向上传递。
add操作,首先,找到最后单点即进行更新操作,否则pushdown,把懒惰标记传递下去,在标记里面也是要对d=sf操作的,最后向上传递结果
query就更明显了,找区间return sf 或者 d,没找到就先pushdown
一开始糊里糊涂,pushdown又没push好,还没修改当前的值
比赛的时候还是有点糊里糊涂,线段树功底还不够,还要继续练
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
#define LL __int64
using namespace std;
const int N = ;
LL f[];
LL d[N<<],flag[N<<],sf[N<<];
LL pref[N];
int n,q;
void init()
{
LL tmp=1LL<<;
f[]=f[]=;
for (int i=;i<=;i++){
f[i]=f[i-]+f[i-]; }
} void build(int rt ,int l,int r)
{
flag[rt]=d[rt]=;
if (l>=r){
sf[rt]=;
return;
}
int mid=(l+r)>>;
build(lson);
build(rson);
sf[rt]=sf[rt<<]+sf[rt<<|];
}
void change(int rt)
{
d[rt]=sf[rt];
}
void up(int rt,int l,int r)
{
d[rt]=d[rt<<]+d[rt<<|];
sf[rt]=sf[rt<<]+sf[rt<<|]; }
void pushdown(int rt,int l,int r)
{
int mid=(l+r)>>;
if (flag[rt]== || l>=r) return;
flag[rt<<]=flag[rt<<|]=flag[rt];
d[rt<<]=sf[rt<<];
d[rt<<|]=sf[rt<<|];
flag[rt]=;
}
void fix(int L,int R,int rt,int l,int r)
{
if (flag[rt]==) return;
if (L<=l && r<=R){
flag[rt]=;
d[rt]=sf[rt];
return;
}
int mid=(l+r)>>;
if (R<=mid) fix(L,R,lson);
else
if (L>mid) fix(L,R,rson);
else{
fix(L,R,lson);
fix(L,R,rson);
}
if (flag[rt<<]== && flag[rt<<|]==){
flag[rt]=;
}
up(rt,l,r);
}
void add(int loc,LL val,int rt,int l,int r)
{
if (l>=r)
{
flag[rt]=;
d[rt]+=val;
int loc=lower_bound(f,f+,d[rt])-f;
sf[rt]=f[loc];
if (loc>){
if (d[rt]-f[loc-]<=f[loc]-d[rt]){
sf[rt]=f[loc-];
}
}
return;
}
pushdown(rt,l,r);
int mid=(l+r)>>;
if (loc<=mid) add(loc,val,lson);
else add(loc,val,rson);
up(rt,l,r);
} LL query(int L,int R,int rt,int l,int r)
{
if (L<=l && r<=R){
if (flag[rt]) return sf[rt];
else return d[rt];
}
pushdown(rt,l,r);
int mid=(l+r)>>;
if (R<=mid) return query(L,R,lson);
else
if (L>mid) return query(L,R,rson);
else
{
LL ret1=query(L,R,lson);
LL ret2=query(L,R,rson);
return ret1+ret2;
}
}
int main()
{
init();
int op;
while (scanf("%d%d",&n,&q)!=EOF)
{
build(,,n);
while (q--)
{
scanf("%d",&op);
if (op==){
int a;
LL b;
scanf("%d%I64d",&a,&b);
add(a,b,,,n);
}
else
if (op==){
int a,b;
scanf("%d%d",&a,&b);
LL ans=query(a,b,,,n);
printf("%I64d\n",ans);
}
else{
int a,b;
scanf("%d%d",&a,&b);
fix(a,b,,,n);
}
}
}
return ;
}
HDU 4893 2014多校三 线段树的更多相关文章
- HDU 4960 Handling the past 2014 多校9 线段树
首先确定的基本思想是按时间离散化后来建线段树,对于每个操作插入到相应的时间点上 但是难就难在那个pop操作,我之前对pop操作的处理是找到离他最近的那个点删掉,但是这样对于后面的peak操作,如果时间 ...
- HDU 4893 Wow! Such Sequence! (线段树)
Wow! Such Sequence! 题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4893 Description Recently, Doge ...
- hdu 5274 Dylans loves tree(LCA + 线段树)
Dylans loves tree Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Othe ...
- HDU 3074.Multiply game-区间乘法-线段树(单点更新、区间查询),上推标记取模
Multiply game Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Tot ...
- HDU 1394 Minimum Inversion Number(线段树求最小逆序数对)
HDU 1394 Minimum Inversion Number(线段树求最小逆序数对) ACM 题目地址:HDU 1394 Minimum Inversion Number 题意: 给一个序列由 ...
- HDU 5029 Relief grain(离线+线段树+启发式合并)(2014 ACM/ICPC Asia Regional Guangzhou Online)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5029 Problem Description The soil is cracking up beca ...
- hdu 1556:Color the ball(线段树,区间更新,经典题)
Color the ball Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...
- hdu 1166敌兵布阵(线段树)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1166 敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) M ...
- 暑期训练狂刷系列——Hdu 1698 Just a Hook (线段树区间更新)
题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=1698 题目大意: 有一个钩子有n条棍子组成,棍子有铜银金三种组成,价值分别为1,2,3.为了对付每场 ...
随机推荐
- redis之Set(无序)类型常用方法总结
redis之Set(无序)类型常用方法总结 存--sadd key member [member ...] 取--SMEMBERS key sadd key member [member ...] 向 ...
- 【剑指Offer面试编程题】题目1390:矩形覆盖--九度OJ
题目描述: 我们可以用2*1的小矩形横着或者竖着去覆盖更大的矩形.请问用n个2*1的小矩形无重叠地覆盖一个2*n的大矩形,总共有多少种方法? 输入: 输入可能包含多个测试样例,对于每个测试案例, 输入 ...
- ROS学习笔记5-理解节点(Node)
本文内容来源于:http://wiki.ros.org/ROS/Tutorials/UnderstandingNodes 图(Graph)概念概览 节点(Nodes):一个节点是ROS下面一个可执行程 ...
- vue的高级使用技巧
全局组件注册 一般组件应用弊端,比较笨拙繁琐低效,比如我们写了一些组件,需要引用上的时候就通过import导入,那如果是高频繁需要使用的组件,则需要在每个使用的时候都需要引入并注册 假设现在有两个组件
- computed、methods、watch
computed:计算属性将被混入到 Vue 实例中.所有 getter 和 setter 的 this 上下文自动地绑定为 Vue 实例. methods:methods 将被混入到 Vue 实例中 ...
- Zero 初识Sciter
在浏览有关Sciter技术前,您需要花点时间浏览以下内容. 您是否需要花时间学习Sciter? 如果您的工作或您想从事的工作与桌面应用开发无关,那么您不需要学习Sciter. 如果您不认同HTML\C ...
- Python dir和vars的区别
dir()和vars()的区别就是 dir()只打印属性(属性,属性......) 而vars()则打印属性与属性的值(属性:属性值......) ex. >> a='aaaaaaaaaa ...
- SSH框架搭建 笔记 (含spring注解驱动)
分类: web 开发2014-04-27 12:33 354人阅读 评论(0) 收藏 举报 框架springinterface注解 好久没有搭建框架了,今天整理下以前的知识,整合下SSH,没想到手生了 ...
- syx学习笔记
SYX复活了,在悲痛之际,希望能让自己获得更多的知识,更有进步,所以留此博客 数学 推荐blog: 1 2 原根表 FFT(快速傅里叶变换) 2019/12/05 √ 博客 blog 题目 Q1 NT ...
- Myeclipse项目出现红叉解决方案
1.右键点击你的项目.选中properties 2.选中MyEclipse下的Project Facets里面的java 此时的版本号为1.5,修改 3.选中MyEclipse下的Project Fa ...