【bzoj3476-懒惰的奶牛】线段树

题解:
感觉这题和别人的做法不一样。。。呵呵呵。。。调了一百年。。
设家坐标为(a,b),对于每个点(x,y),可以转化为|a-x|+|b-y|<=k
对于每个点,它的影响范围是一个菱形(也就是一个正方形啦。。),也就是一个图上有若干个正方形。

然后我就把这个坐标轴选择了45度。
好难画不画了,正交分解一下就可以了。
然后题目就转化成正方形各种交里的最大值。

正方形有x和y两个元素,但是很明显我们只能维护一个。。
所以我以x轴建立线段树,对于每个正方形按照y从小到大排序。
维护一个指针j,表示当前前j个正方形已经和现在在处理的第i个正方形没有交集。每次都要先把j更新(看看它是否能后移)。
然后我们在当前正方形的两端a[i].x~a[i].y这一段+a[i].d
每个点维护当前的和d以及该区间的最大值mx。每次做完之后,用t[1].mx更新答案。
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<ctime>
#include<queue>
#include<algorithm>
using namespace std; const int N=,INF=(int)1e9;
int n,K,L,tl,ans;
struct trnode{
int l,r,lc,rc,d,mx,lazy;
}t[*N];
struct node{
int x,y,d;
}a[N]; bool cmp(node x,node y){return x.y<y.y;}
int maxx(int x,int y){return x>y ? x:y;}
int minn(int x,int y){return x<y ? x:y;} int read()
{
int x=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-; ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-''; ch=getchar();}
return x*f;
} int bt(int l,int r)
{
int x=++tl;
t[x].l=l;t[x].r=r;
t[x].lc=t[x].rc=;
t[x].mx=;t[x].d=;
t[x].lazy=;
if(l<r)
{
int mid=(l+r)/;
t[x].lc=bt(l,mid);
t[x].rc=bt(mid+,r);
}
return x;
} void pd(int x)
{
if(t[x].lazy==) return ;
int d=t[x].lazy,lc=t[x].lc,rc=t[x].rc;
t[x].lazy=;
t[x].d+=d;
t[x].mx+=d;
if(lc) t[lc].lazy+=d;
if(rc) t[rc].lazy+=d;
} void change(int x,int l,int r,int d)
{
pd(x);
if(t[x].l==l && t[x].r==r)
{
t[x].lazy+=d;
pd(x);
return ;
}
int lc=t[x].lc,rc=t[x].rc,mid=(t[x].l+t[x].r)/;
if(r<=mid) change(lc,l,r,d);
else if(l>mid) change(rc,l,r,d);
else
{
change(lc,l,mid,d);
change(rc,mid+,r,d);
}
pd(x);pd(lc);pd(rc);
t[x].mx=maxx(t[lc].mx,t[rc].mx);
} int main()
{
// freopen("a.in","r",stdin);
// freopen("me.out","w",stdout);
freopen("lazy.in","r",stdin);
freopen("lazy.out","w",stdout);
n=read();K=read();
// scanf("%d%d",&n,&K);
int x,y,tx,ty,nx=INF,ny=INF,mx=;tl=;L=*K;
for(int i=;i<=n;i++)
{
a[i].d=read();x=read();y=read();
// scanf("%d%d%d",&a[i].d,&x,&y);
tx=x,ty=y+K;
a[i].x=tx-ty;
a[i].y=tx+ty; nx=minn(nx,a[i].x);
ny=minn(ny,a[i].y);
}
for(int i=;i<=n;i++)
{
a[i].x-=nx-;
a[i].y-=ny-;
mx=maxx(mx,a[i].x);
}
mx+=;ans=;
sort(a+,a++n,cmp);
bt(,mx);
int j=;
change(,a[].x,minn(mx,a[].x+L),a[].d);
ans=maxx(ans,t[].mx);
for(int i=;i<=n;i++)
{
while(j<i && a[i].y-L>a[j].y)
{
change(,a[j].x,minn(mx,a[j].x+L),-a[j].d);
ans=maxx(ans,t[].mx);
j++;
}
change(,a[i].x,minn(mx,a[i].x+L),a[i].d);
ans=maxx(ans,t[].mx);
}
printf("%d\n",ans);
return ;
}
【bzoj3476-懒惰的奶牛】线段树的更多相关文章
- Codeforces 444C 线段树 懒惰标记
前天晚上的CF比赛div2的E题,很明显一个线段树,当时还在犹豫复杂度的问题,因为他是区间修改和区间查询,肯定是要用到懒惰标记. 然后昨天真的是给这道题跪了,写了好久好久,...我本来是写了个add标 ...
- D - Mayor's posters(线段树+离散化)
题目: The citizens of Bytetown, AB, could not stand that the candidates in the mayoral election campai ...
- 【HDU 4614】Vases and Flowers(线段树区间更新懒惰标记)
题目0到n-1的花瓶,操作1在下标a开始插b朵花,输出始末下标.操作2清空[a,b]的花瓶,求清除的花的数量.线段树懒惰标记来更新区间.操作1,先查询0到a-1有num个空瓶子,然后用线段树的性质,或 ...
- 【POJ】3468 A Simple Problem with Integers ——线段树 成段更新 懒惰标记
A Simple Problem with Integers Time Limit:5000MS Memory Limit:131072K Case Time Limit:2000MS Descr ...
- 【HDU】4092 Nice boat(多校第四场1006) ——线段树 懒惰标记
Nice boat Time Limit: 30000/15000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) To ...
- poj3468 线段树的懒惰标记
题目链接:poj3468 题意:给定一段数组,有两种操作,一种是给某段区间加c,另一种是查询一段区间的和 思路:暴力的方法是每次都给这段区间的点加c,查询也遍历一遍区间,复杂度是n*n,肯定过不去,另 ...
- HDU 4107 Gangster(线段树 特殊懒惰标记)
两种做法. 第一种:标记区间最大值和最小值,若区间最小值>=P,则本区间+2c,若区间最大值<P,则本区间+c.非常简单的区间更新. 最后发一点牢骚:最后query查一遍就行,我这个2B竟 ...
- POJ 3468 线段树 成段更新 懒惰标记
A Simple Problem with Integers Time Limit:5000MS Memory Limit:131072K Case Time Limit:2000MS Descr ...
- Transformation HDU - 4578(线段树——懒惰标记的妙用)
Yuanfang is puzzled with the question below: There are n integers, a 1, a 2, …, a n. The initial val ...
随机推荐
- 【转】c++面试基础
1,关于动态申请内存 答:内存分配方式三种: (1)从静态存储区域分配:内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在. 全局变量,static变量. (2)在栈上创建:在执行函 ...
- c# word 删除指定内容
1.首先简单的是获取得到的range,直接rangge.delete() 2.文本框的删除: foreach (Microsoft.Office.Interop.Word.Shape shape in ...
- Internet Secure
key Secret key(密钥):ues the same key to Encryption and decryption(symmetric-key)对称加密,需要绝对安全的空间来传递key ...
- .net改善程序性能建议
对改善程序性能的建议. 文章:https://msdn.microsoft.com/zh-cn/library/ms973838.aspx
- 再看RCU
从昨天晚上开始,我就立志要把RCU拿下,昨晚加今天早上看了RCU的东西,太细节的原理我就不扣了,放弃,知道RCU是怎么用的就可以了,赶紧看文件系统中dcache的管理了. 说说RCU,RCU其实是很简 ...
- 安装多个版本JDK相关问题
一.前言 因敝人计算器上面安装了多个版本的JDK,其中包括JDK1.6.JDK1.7.JDK1.8,想通过变换环境变量(JAVA_HOME)的形式切换不同的JDK,但是我在安装了JDK1.7并且配置了 ...
- Android OCR文字识别 实时扫描手机号(极速扫描单行文本方案)
身份证识别:https://github.com/wenchaosong/OCR_identify 遇到一个需求,要用手机扫描纸质面单,获取面单上的手机号,最后决定用tesseract这个开源OCR库 ...
- 【.Net】c# 中config.exe 引发的一些问题
public static void CreateConfig(){ //c#可以添加内置的app.config,我们通过ConfigrationManager类可以 //可以很轻松的操作相关节点,操 ...
- BZOJ2151 种树(贪心+堆+链表/wqs二分+动态规划)
dp容易想到,但没法进一步优化了. 考虑贪心,每次选出价值最大的物品.但这显然是不对的因为会影响其他物品的选择. 于是考虑加上反悔操作.每次选出一个物品后,将其相邻两物品删除,再将原物品价值变为相邻两 ...
- [洛谷P3550][POI2013]TAK-Taxis
题目大意:一条路上有三个点,$0$为起始位置,$d$为总部,$m$为家.有$n$辆车,每辆车最多行驶$x_i$,都从$d$出发,可以在任意位置结束,问最少几辆车可以到家. 题解:贪心,发现当人在$[0 ...