Just a Hook (HDU 1698)

题链

每一次都将一个区间整体进行修改,需要用到懒惰标记,懒惰标记的核心在于在查询前才更新,比如将当前点rt标记为col[rt],那么此点的左孩子和右孩子标记必然和其一致(直接替换,如果是累积则另当别论),同时这个区间也能很快求出了

线段树功能:区间更新+区间查询

#include <cstdio>
#include <utility>
#include <queue>
#include <cstring>
#define scan(x) scanf("%d",&x)
#define scan2(x,y) scanf("%d%d",&x,&y)
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
#define root 1,1,n
using namespace std;
const int Max=1e5+10;
int sum[Max<<2],col[Max<<2];
void Pushup(int rt)
{
sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}
void Build(int rt,int l,int r)
{
if(l==r)
{
sum[rt]=1;
col[rt]=0;
return;
}
int mid=(l+r)>>1;
Build(lson);
Build(rson);
Pushup(rt);
}
void Pushdown(int rt,int rage)
{
if(col[rt])
{
col[rt<<1]=col[rt<<1|1]=col[rt];
sum[rt<<1]=col[rt]*(rage-(rage>>1));
sum[rt<<1|1]=col[rt]*(rage>>1);
col[rt]=0;
}
}
void Update(int L,int R,int x,int rt,int l,int r)
{
if(L<=l&&r<=R)
{
sum[rt]=x*(r-l+1);
col[rt]=x;
return;
}
Pushdown(rt,(r-l)+1);
int mid=(l+r)>>1;
if(L<=mid) Update(L,R,x,lson);
if(mid<R) Update(L,R,x,rson);
Pushup(rt);
}
int main()
{
int T,ca=1;
for(scan(T);T;T--)
{
memset(sum,0,sizeof(sum));
memset(col,0,sizeof(col));
int l,r,k,n,m;
scan2(n,m);
Build(root);
while(m--)
{
scanf("%d%d%d",&l,&r,&k);
Update(l,r,k,root);
}
printf("Case %d: The total value of the hook is %d.\n",ca++,sum[1]);
}
return 0;
}

重新写了一次,学习了一份新的模板

#include<queue>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define ll long long
#define inf 1000000000
#define mod 1000000007
using namespace std;
int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*f;
}
int n,m;
int a[100005];
int ls[400005],rs[400005];
int M[400005],tag[400005];
void build(int k,int l,int r)
{
int mid=(l+r)>>1;
ls[k]=l;rs[k]=r;tag[k]=0;M[k]=0;
if(l==r){M[k]=1;return;}
build(k<<1,l,mid);
build(k<<1|1,mid+1,r);
M[k]=M[k<<1]+M[k<<1|1];
}
void pushdown(int k)
{
if(!tag[k]||ls[k]==rs[k])return;
tag[k<<1]=tag[k];
tag[k<<1|1]=tag[k];
int rage=(rs[k]-ls[k]+1);
M[k<<1]=(rage-(rage>>1))*tag[k];
M[k<<1|1]=(rage>>1)*tag[k];
tag[k]=0;
}
void add(int k,int x,int y,int v)
{
pushdown(k);
int l=ls[k],r=rs[k],mid=(l+r)>>1;
if(x==l&&y==r)
{
tag[k]=v;
M[k]=v*(r-l+1);
return;
}
if(x<=mid) add(k<<1,x,min(y,mid),v);
if(y>mid) add(k<<1|1,max(x,mid+1),y,v);
M[k]=M[k<<1]+M[k<<1|1];
}
int query(int k,int x,int y)
{
pushdown(k);
int l=ls[k],r=rs[k],mid=(l+r)>>1,ans=0;
if(x==l&&y==r) return M[k];
if(x<=mid)ans+=query(k<<1,x,min(y,mid));
if(y>mid)ans+=query(k<<1|1,max(x,mid+1),y);
return ans;
}
int main()
{
int T,ca=1;
for(T=read();T;T--)
{
int l,r,v,n,m;
n=read();m=read();
build(1,1,n);
while(m--)
{
scanf("%d%d%d",&l,&r,&v);
add(1,l,r,v);
}
printf("Case %d: The total value of the hook is %d.\n",ca++,query(1,1,n));
}
return 0;
}

Just a Hook (HDU 1698) 懒惰标记的更多相关文章

  1. hdu 1698 (延迟标记+区间修改+区间求和)

    In the game of DotA, Pudge's meat hook is actually the most horrible thing for most of the heroes. T ...

  2. E - Just a Hook - hdu 1698(区间覆盖)

    某个英雄有这样一个金属长棍,这个金属棍有很多相同长度的短棍组成,大概最多有10w节,现在这个人有一种魔法,他可以把一段区间的金属棍改变成别的物质,例如金银或者铜, 现在他会有一些操作在这个金属棍上,他 ...

  3. E - Just a Hook HDU - 1698 线段树区间修改区间和模版题

    题意  给出一段初始化全为1的区间  后面可以一段一段更改成 1 或 2 或3 问最后整段区间的和是多少 思路:标准线段树区间和模版题 #include<cstdio> #include& ...

  4. Just a Hook HDU - 1698Just a Hook HDU - 1698 线段树区间替换

    #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> us ...

  5. HDU 1698 just a hook - 带有lazy标记的线段树(用结构体实现)

    2017-08-30 18:54:40 writer:pprp 可以跟上一篇博客做个对比, 这种实现不是很好理解,上一篇比较好理解,但是感觉有的地方不够严密 代码如下: /* @theme:segme ...

  6. HDU 1698 Just a Hook(线段树成段更新)

    题目网址:http://acm.hdu.edu.cn/showproblem.php?pid=1698 题目: Problem Description   In the game of DotA, P ...

  7. HDU 1698 Just a Hook(线段树:区间更新)

    http://acm.hdu.edu.cn/showproblem.php?pid=1698 题意:给出1~n的数,每个数初始为1,每次改变[a,b]的值,最后求1~n的值之和. 思路: 区间更新题目 ...

  8. HDU 1698 Just a Hook (线段树区间更新)

    题目链接 题意 : 一个有n段长的金属棍,开始都涂上铜,分段涂成别的,金的值是3,银的值是2,铜的值是1,然后问你最后这n段总共的值是多少. 思路 : 线段树的区间更新.可以理解为线段树成段更新的模板 ...

  9. HDU 1698——Just a Hook——————【线段树区间替换、区间求和】

    Just a Hook Time Limit:2000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit  ...

随机推荐

  1. IDEA hadoop MapReduce 环境配置

    1.下载,安装,配置好Hadoop 2.在IDEA中执行MapReduc 配置: 这里将JAR包加入: JAR包是:/usr/local2/hadoop/share/hadoop 目录下:直接右边+以 ...

  2. java静态方法和实例化方法的区别(copy)

    [资料来源] http://blog.csdn.net/biaobiaoqi/article/details/6732117 方法是我们每天都在写得,很多程序员大多都使用实例化方法,而很少使用静态方法 ...

  3. P4097 [HEOI2013]Segment

    传送门 简单来说就是对于每条线段,先把它拆成\(O(logn)\)条,然后对于每一条再\(O(logn)\)判断在所有子区间的优劣程度 //minamoto #include<bits/stdc ...

  4. Akka源码分析-Cluster-Distributed Publish Subscribe in Cluster

    在ClusterClient源码分析中,我们知道,他是依托于“Distributed Publish Subscribe in Cluster”来实现消息的转发的,那本文就来分析一下Pub/Sub是如 ...

  5. May Challenge 2019 Division 2 水题讲解

    Reduce to One 这题其实蛮水的? 题意就是说: 给定一个 1~n 的序列,每次挑两个数 x y 合并,合并值为 \(x+y+xy\) ,然后求不断合并最后剩下的一个的最大值 随便搞搞发现答 ...

  6. Python中re操作正则表达式

    在python中使用正则表达式 1.转义符 正则表达式中的转义: '\('表示匹配小括号 [()+*/?&.] 在字符组中一些特殊的字符会现出原形 所有的\s\d\w\S\D\W\n\t都表示 ...

  7. HTTP的报文格式、GET和POST格式解析

    1. HTTP报文格式 HTTP报文是面向文本的,报文中的每一个字段都是一些ASCII码串,各个字段的长度是不确定的.HTTP有两类报文:请求报文和响应报文.请求报文一个HTTP请求报文由请求行(re ...

  8. 42使用NanoPiM1Plus在Android4.4.2下的录音测试

    42使用NanoPiM1Plus在Android4.4.2下的录音测试 大文实验室/大文哥壹捌陆捌零陆捌捌陆捌贰21504965 AT qq.com完成时间:2017/12/5 17:51版本:V1. ...

  9. Angular——路由参数

    基本介绍 在控制中注入$routeParams可以获取传递的参数 区别对比 angular中的路由是指#之后的内容,包括之后的?,而在之前的http地址中我们习惯性的将?放在前面 具体使用 1.形参 ...

  10. mongodb Shell 启动

    开始运行mongodb 准备 上篇说过,通过brew安装的程序目录在 /usr/local/Cellar下面 下面,我们先看一下 mongodb的可执行程序命令 cd /usr/local/Cella ...