这道题目的数据卡得好厉害。

题目明显是考察线段树延迟标记的,但是因为要考虑到p的值,这种延迟是有条件的:在该节点下所有的数据对于p都应该位于p的同一侧。要么都比p大,要么都比p小。

开始的时候我用一个flag来标记节点下面的值是否相同,这个想法其实不对,在最恶劣的情况下,这种方式几乎会直接退化到单点更新的程度,而且随着数据的输入,算法的效率会越来越低,因为整个树从上到下都是在一次性使用,没办法维护。

但是我还是提交了一下,没有任何悬念的TLE。

我又开始正常的思路,不再考虑一个节点下面的值是否相同,而是去想这些值是否在p的同一侧。想了一下,这样的话,我们只需要知道节点下面的最小值和最大值是不是在p的同一侧就行了,而维护最大值最小值之类的事简直是线段树最擅长的了。代码写好后提交,还是TLE。

这我就有点儿郁闷了,这种方法确实在很高程度上实现了数据的成段更新。我的建树和查找是完全相同的操作,在两个地方不太可能有什么优化的空间,能优化的地方只有更新操作。不过我实在想不来该怎么优化。

我去看了下别人的题解,发现博主用跟我同样地思路过了,不过过得很勉强,在G++下TLE,在C++下AC。我看了下他的代码,相对于我的代码来说,做了两个地方的优化,一个是线段树向下分发的时候加了一个if(a[t].dam)的操作,这个操作能够提高的效率微乎其微。另一个是减少了一个判断操作,这个地方我觉得能提高一些效率。

提交C++928msAC。

#include<stdio.h>
#include<string.h>
#define N 200005
struct node
{
int x,y;
int dam;
int min,max;
}a[N*3];
int b[N];
int m,n,p;
int Max(int x,int y)
{
if(x>y)
return x;
else
return y;
}
int Min(int x,int y)
{
if(x<y)
return x;
else
return y;
}
void CreatTree(int t,int x,int y)
{
a[t].x=x;
a[t].y=y;
a[t].dam=0;
a[t].max=0;
a[t].min=0;
if(x==y)
return ;
int temp=t*2;
int mid=(x+y)/2;
CreatTree(temp,x,mid);
CreatTree(temp+1,mid+1,y);
return ;
}
void InsertTree(int t,int x,int y,int k)
{
if(a[t].x==x&&a[t].y==y)
{
if(a[t].min>=p)
{
a[t].dam+=2*k;
a[t].max+=2*k;
a[t].min+=2*k;
return ;
}
else if(a[t].max<p)
{
a[t].dam+=k;
a[t].min+=k;
a[t].max+=k;
return ;
}
}
int temp=t*2;
int mid=(a[t].x+a[t].y)/2;
if(a[t].dam>0)
{
a[temp].dam+=a[t].dam;
a[temp+1].dam+=a[t].dam;
a[temp].min+=a[t].dam;
a[temp+1].min+=a[t].dam;
a[temp].max+=a[t].dam;
a[temp+1].max+=a[t].dam;
a[t].dam=0;
}
if(y<=mid)
InsertTree(temp,x,y,k);
else if(x>mid)
InsertTree(temp+1,x,y,k);
else
{
InsertTree(temp,x,mid,k);
InsertTree(temp+1,mid+1,y,k);
}
a[t].max=Max(a[temp].max,a[temp+1].max);
a[t].min=Min(a[temp].min,a[temp+1].min);
return ;
}
void FindTree(int t,int x,int y)
{
if(a[t].x==a[t].y)
{
b[a[t].x]=a[t].dam;
//printf("%d %d %d %d %d\n",a[t].x,a[t].y,a[t].dam,a[t].max,a[t].min);
return ;
}
int temp=t*2;
int mid=(a[t].x+a[t].y)/2;
a[temp].dam+=a[t].dam;
a[temp+1].dam+=a[t].dam;
a[t].dam=0;
FindTree(temp,x,mid);
FindTree(temp+1,mid+1,y);
//printf("%d %d %d %d %d\n",a[t].x,a[t].y,a[t].dam,a[t].max,a[t].min);
return ;
}
int main()
{
while(scanf("%d%d%d",&n,&m,&p)!=EOF)
{
CreatTree(1,1,n);
while(m--)
{
int x,y,k;
scanf("%d%d%d",&x,&y,&k);
InsertTree(1,x,y,k);
}
FindTree(1,1,n);
int i;
for(i=1;i<n;i++)
printf("%d ",b[i]);
printf("%d\n",b[i]);
}
return 0;
}

hdu 4107 Gangster(线段树,时间卡得很严)的更多相关文章

  1. hdu 4031 attack 线段树区间更新

    Attack Time Limit: 5000/3000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Others)Total Subm ...

  2. HDU 4578 Transformation --线段树,好题

    题意: 给一个序列,初始全为0,然后有4种操作: 1. 给区间[L,R]所有值+c 2.给区间[L,R]所有值乘c 3.设置区间[L,R]所有值为c 4.查询[L,R]的p次方和(1<=p< ...

  3. hdu 4288 离线线段树+间隔求和

    Coder Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Su ...

  4. hdu 3016 dp+线段树

    Man Down Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  5. HDU 3308 (线段树区间合并)

    http://acm.hdu.edu.cn/showproblem.php?pid=3308 题意: 两个操作  : 1 修改 单点  a 处的值. 2 求出 区间[a,b]内的最长上升子序列. 做法 ...

  6. HDU 4107 Gangster(线段树 特殊懒惰标记)

    两种做法. 第一种:标记区间最大值和最小值,若区间最小值>=P,则本区间+2c,若区间最大值<P,则本区间+c.非常简单的区间更新. 最后发一点牢骚:最后query查一遍就行,我这个2B竟 ...

  7. HDU 4107 Gangster Segment Tree线段树

    这道题也有点新意,就是须要记录最小值段和最大值段,然后成段更新这个段,而不用没点去更新,达到提快速度的目的. 本题过的人非常少,由于大部分都超时了,我严格依照线段树的方法去写.一開始竟然也超时. 然后 ...

  8. HDU 4107 Gangster

    Gangster Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged on HDU. Original ID: 4 ...

  9. HDU 5861 Road 线段树区间更新单点查询

    题目链接: http://acm.split.hdu.edu.cn/showproblem.php?pid=5861 Road Time Limit: 12000/6000 MS (Java/Othe ...

随机推荐

  1. 阿里云修改CentOS Linux服务器的主机名

    阿里云主机的默认主机名是为AY开头的随机名称,如何修改为易于区分的友好名称呢?请看下面的操作步骤: 1. vi /etc/hosts i键,修改主机名,esc键,:wq键保存退出 2. vi /etc ...

  2. COCO 数据集使用说明书

    下面的代码改写自 COCO 官方 API,改写后的代码 cocoz.py 被我放置在 Xinering/cocoapi.我的主要改进有: 增加对 Windows 系统的支持: 替换 defaultdi ...

  3. kafka和springboot整合应用

    加载依赖 <dependency> <groupId>org.springframework.kafka</groupId> <artifactId>s ...

  4. Java 关于集合框架那点事儿

     1.引入集合框架  采用数组存在的一些缺陷:   1.数组长度固定不变,不能很好地适应元素数量动态变化的情况.   2.可通过数组名.length获取数组的长度,却无法直接获取数组中真实存储的个数. ...

  5. tftp协议

    <前言> 嵌入式开发是一个交叉开发的模式,需要将宿主机上的文件烧写到目标机上. 方式: JTAG USB 串口 网络 <tftp下载> 首先需要将宿主机架成一个TFTP的服务器 ...

  6. [BZOJ5291][BJOI2018]链上二次求和(线段树)

    感觉自己做的麻烦了,但常数似乎不算差.(只是Luogu最慢的点不到2s本地要跑10+s) 感觉我的想法是最自然的,但不明白为什么网上似乎找不到这种做法.(不过当然所有的做法都是分类大讨论,而我的方法手 ...

  7. 【平面图最小割】BZOJ1001- [BeiJing2006]狼抓兔子

    [题目大意]左上角点为(1,1),右下角点为(N,M)(上图中N=4,M=5).有以下三种类型的道路 1:(x,y)<==>(x+1,y) 2:(x,y)<==>(x,y+1) ...

  8. 【初识】KMP算法入门

    举个例子 模式串S:a s d a s d a s d f a s d 匹配串T:a s d a s d f 如果使用朴素匹配算法—— 1 2 3 4 5 6  8 9 a s d a s d a s ...

  9. 热爱编程,热爱生活,快速适应IT新技术

    现在新技术层出不穷,我们没有那么多时间去深入学习每一门.对于大多数技术,我们只需要搞懂概念部分,从宏观上了解一下,决定我们要不要深入地去学习它.有了这些了解,就能轻松地与别人聊天了,也可以为以后技术选 ...

  10. bzoj 3252: 攻略 -- 长链剖分+贪心

    3252: 攻略 Time Limit: 10 Sec  Memory Limit: 128 MB Description 题目简述:树版[k取方格数]   众所周知,桂木桂马是攻略之神,开启攻略之神 ...