蒟蒻的第一篇黑题题解(学了这么长时间了才第一道也是没谁了。)

题目链接

Solution

朴素:

 根据题目描述,我们可以处理每一个x节点左右爆炸半径范围内的点,然后模拟一次爆炸 (for),遍历每一个点。每当我们遍历到一个点,我们就对这个点在进行一次处理半径+dfs直到没有能遍历的了,直接统计遍历的点数然后输出。

时间复杂度:O(n^2)级别。

于是乎:

  

是不是可以暴力踩标算了呢?

显然完全接受不了。

因为这个算法要考虑到连边操作。对于每一个点,我们要将它自己连向能够遍历到的边,那么意味着,一个边要被连很多次。

就如下图:

其中编号为1的节点能炸到1 2 3 4 5,2能炸到1 2 3,3能炸到1 2 3 4 5,其中2 3 4挺惨的。

于是有了这么一个线段树建图的思路:

考虑线段树的性质,上层节点可以代表下层节点的一段区间,那么我们是不是也可以建这类的上层边,表示通过这条边能遍历到它对应的下层边的所有点。(就是这句精华)

这样我们先把所有点进行从1到n的编号,然后跑线段树的build建边,然后依次对每一个点进行从底层(只能这么理解)到上层节点的连边表示能遍历到上层节点能遍历到的节点。

这样的话,会形成很多环。在对每一个点都进行范围连边以后,对于一个环(或者强连通分量)内的点,自然能互相遍历到,为了方便统计,我们需要跑一个tarjan记录强连通分量内的点能遍历到的左右区间范围。

跑完tarjan就可以O(n)遍历一遍按照题目所说进行统计了呢。

部分代码

build:

void build(int now,int l,int r)
{
canl[now]=l;canr[now]=r;//can表示当前节点能遍历到的左右区间范围
if(l==r)//线段树日常操作
{
id[r]=now;
return;
}
int mid=(l+r)>>;
build(now<<,l,mid);build(now<<|,mid+,r);
add(now,now<<);add(now,now<<|);//不同于线段树的地方,向左右两个节点连边,往下层伸展
}

下层往上连边部分:

void lb(int mb,int l,int r,int L,int R,int now)
{
if(l>=L&&r<=R)
{
if(mb==now)//mb表示目标节点,也就是当前初始节点
return;
add(mb,now);//now表示线段树上当前节点,从mb到now连边表示mb这个节点可以爆炸到now节点代表的区间
return;
}
int mid=(l+r)>>;
if(L<=mid)lb(mb,l,mid,L,R,now<<);//左右区间连边
if(R>mid)lb(mb,mid+,r,L,R,now<<|);
}

tarjan部分:

void tarjan(int now)
{
dfn[now]=++tarjannum;//dfn都知道是什么吧
low[now]=tarjannum;
vis[now]=;//设置入栈
st[++top]=now;
for(int i=hea[now];i;i=edge[i].nex)
{
int v=edge[i].to;
if(!dfn[v])
{
tarjan(v);low[now]=min(low[now],low[v]);
}
else
if(vis[v]) //在栈中
low[now]=min(low[now],dfn[v]);//以上正常tarjan操作
}
if(low[now]==dfn[now])
{
numdag++;//标记为同一个dag
do
{
int topp=st[top--];
dagbh[topp]=numdag;
leff[numdag]=min(leff[numdag],canl[topp]);
//在统计tarjan的同时,把这个强联通分量里面所有的点能到的左右区间范围统计一下
rigg[numdag]=max(rigg[numdag],canr[topp]);
vis[topp]=;//出栈标记
}while(st[top+]!=now);//dowhile保证能够便历到,因为上面top已经--了所以要判断top+1
}
}

高清无码完整代码:

#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#define N 500003
#define modd 1000000007
using namespace std;
long long read()
{
long long ans=;
char ch=getchar(),last=' ';
while(ch<''||ch>'')last=ch,ch=getchar();
while(ch>=''&&ch<='')ans=(ans<<)+(ans<<)+ch-'',ch=getchar();
return last=='-'?-ans:ans;
}
int n,id[N],hea[N*],num,vis[N*],dagbh[N*],lef,rig,top,canl[N*],canr[N*];
int st[N*],dfn[N*],low[N*],leff[N*],rigg[N*],tarjannum,numdag;
long long r[N],x[N],ans;
struct edg{
int nex,to;
}edge[N*];
inline void add(int from,int to)
{
num++;
edge[num]={hea[from],to};
hea[from]=num;
}
void build(int now,int l,int r)
{
canl[now]=l;canr[now]=r;//can表示当前节点能遍历到的左右区间范围
if(l==r)//线段树日常操作
{
id[r]=now;
return;
}
int mid=(l+r)>>;
build(now<<,l,mid);build(now<<|,mid+,r);
add(now,now<<);add(now,now<<|);//不同于线段树的地方,向左右两个节点连边,往下层伸展 void lb(int mb,int l,int r,int L,int R,int now)
{
if(l>=L&&r<=R)
{
if(mb==now)//mb表示目标节点,也就是当前初始节点
return;
add(mb,now);//now表示线段树上当前节点
return;
}
int mid=(l+r)>>;
if(L<=mid)lb(mb,l,mid,L,R,now<<);//左右区间连边
if(R>mid)lb(mb,mid+,r,L,R,now<<|);
}
void tarjan(int now)
{
dfn[now]=++tarjannum;//dfn都知道是什么吧
low[now]=tarjannum;
vis[now]=;//设置入栈
st[++top]=now;
for(int i=hea[now];i;i=edge[i].nex)
{
int v=edge[i].to;
if(!dfn[v])
{
tarjan(v);low[now]=min(low[now],low[v]);
}
else
if(vis[v]) //在栈中
low[now]=min(low[now],dfn[v]);//以上正常tarjan操作
}
if(low[now]==dfn[now])
{
numdag++;//标记为同一个dag
do
{
int topp=st[top--];
dagbh[topp]=numdag;
leff[numdag]=min(leff[numdag],canl[topp]);
//在统计tarjan的同时,把这个强联通分量里面所有的点能到的左右区间范围统计一下
rigg[numdag]=max(rigg[numdag],canr[topp]);
vis[topp]=;//出栈标记
}while(st[top+]!=now);//dowhile保证能够便历到,因为上面top已经--了所以要判断top+1
}
}
int main()
{
memset(leff,0x3f,sizeof(leff));
n=read();
for(int i=;i<=n;i++)
{
x[i]=read(),r[i]=read();//x为横轴坐标,r为爆炸半径
}
x[n+]=0x3f3f3f3f3f3f3f3fll;
build(,,n);
for(int i=;i<=n;i++)
{
if(!r[i]) continue;
rig=upper_bound(x+,x++n,x[i]+r[i])-x-;
lef=lower_bound(x+,x++n,x[i]-r[i])-x;
lb(id[i],,n,lef,rig,);
}
tarjan();//从1号节点往下跑tarjan缩点跑成dag
for(int i=;i<=n;i++)
{
int fromm=dagbh[id[i]];
ans=(ans+((long long)i*(rigg[fromm]-leff[fromm]+))%modd)%modd;//ll
}
printf("%lld",ans%modd);
}

完结。。

可能是最水的黑题了其他的都不会做。

希望讲解过后能对泥萌有所帮助qwq

P5025 [SNOI2017]炸弹 题解的更多相关文章

  1. P5025 [SNOI2017]炸弹

    原题链接  https://www.luogu.org/problem/P5025 闲话时刻: 第一道 AC 的黑题,虽然众人皆说水... 其实思路不是很难,代码也不是很难打,是一些我们已经学过的东西 ...

  2. [LOJ#2255][BZOJ5017][Snoi2017]炸弹

    [LOJ#2255][BZOJ5017][Snoi2017]炸弹 试题描述 在一条直线上有 N 个炸弹,每个炸弹的坐标是 Xi,爆炸半径是 Ri,当一个炸弹爆炸时,如果另一个炸弹所在位置 Xj 满足: ...

  3. [bzoj5017][Snoi2017]炸弹 tarjan缩点+线段树优化建图+拓扑

    5017: [Snoi2017]炸弹 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 608  Solved: 190[Submit][Status][ ...

  4. [SNOI2017]炸弹[线段树优化建图]

    [SNOI2017]炸弹 线段树优化建图,然后跑一边tarjan把点全部缩起来,炸一次肯定是有连锁反应的所以整个连通块都一样-于是就可以发现有些是只有单向边的不能忘记更新,没了. #include & ...

  5. BZOJ5017题解SNOI2017炸弹--玄学递推

    题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=5017 分析 老师讲课谈到了这道题,课上想出了个连边建图然后乱搞的操作,被老师钦定的递推方 ...

  6. BZOJ5017 [SNOI2017]炸弹 - 线段树优化建图+Tarjan

    Solution 一个点向一个区间内的所有点连边, 可以用线段树优化建图来优化 : 前置技能传送门 然后就得到一个有向图, 一个联通块内的炸弹可以互相引爆, 所以进行缩点变成$DAG$ 然后拓扑排序. ...

  7. 【bzoj5017】[Snoi2017]炸弹 线段树优化建图+Tarjan+拓扑排序

    题目描述 在一条直线上有 N 个炸弹,每个炸弹的坐标是 Xi,爆炸半径是 Ri,当一个炸弹爆炸时,如果另一个炸弹所在位置 Xj 满足:  Xi−Ri≤Xj≤Xi+Ri,那么,该炸弹也会被引爆.  现在 ...

  8. 洛谷P5364 [SNOI2017]礼物 题解

    传送门 /* 热情好客的小猴子请森林中的朋友们吃饭,他的朋友被编号为 1∼N,每个到来的朋友都会带给他一些礼物:大香蕉.其中,第一个朋友会带给他 11 个大香蕉,之后,每一个朋友到来以后,都会带给他之 ...

  9. BZOJ5017 [Snoi2017]炸弹[线段树优化建边+scc缩点+DAG上DP/线性递推]

    方法一: 朴素思路:果断建图,每次二分出一个区间然后要向这个区间每个点连有向边,然后一个环的话是可以互相引爆的,缩点之后就是一个DAG,求每个点出发有多少可达点. 然后注意两个问题: 上述建边显然$n ...

随机推荐

  1. 最新 华云数据java校招面经 (含整理过的面试题大全)

    从6月到10月,经过4个月努力和坚持,自己有幸拿到了网易雷火.京东.去哪儿.华云数据等10家互联网公司的校招Offer,因为某些自身原因最终选择了华云数据.6.7月主要是做系统复习.项目复盘.Leet ...

  2. Android netty客户端入门

    新建项目,加入netty库 implementation 'io.netty:netty-all:4.1.36.Final'

  3. The timeout period elapsed prior to completion of the operation or the server is not responding.

    问题:更新数据的状态值时,部分报出如下异常: 即时有成功更新,时有报错问题出现. 在LOG中发现成功更新的数据,存在更新时间过长问题,将近30秒(EF默认的CommandTimeout为30秒): 代 ...

  4. table固定头部,tbody内容滚动

    直觉的感受是修改thead与tbody,尝试了以下几种方法,但均告失败. 1. 将tbody设置为块状元素,然后设置表格的高度与溢出: 1. 将thead设置为绝对定位,然后设置表格的高度与溢出: 1 ...

  5. javaSE总结(一)-java数据类型和运算符

    一.注释 (1)单行注释: // (2)多行注释:/*  */  (3)文档注释:/**  */ 二.标识符和关键字 (1)分隔符:分号; 花括号{} 方括号[] 圆括号() 空格 圆点(.)     ...

  6. Mac命令行启动关闭Tomcat

    在Tomcat目录的bin目录下执行sh startup.sh或./startup.sh命名启动Tomcat,执行sh shutdown.sh或./shutdown.sh命令关闭Tomcat 浏览器输 ...

  7. 洛谷P2178 [NOI2015]品酒大会 后缀数组+单调栈

    P2178 [NOI2015]品酒大会 题目链接 https://www.luogu.org/problemnew/show/P2178 题目描述 一年一度的"幻影阁夏日品酒大会" ...

  8. pt-table-checksum和pt-table-sync使用

    pt-table-checksum和pt-table-sync使用 数据库版本:5.6.25 pt工具版本:2.2.14 主从关系一:不同机器同一端口 10.10.228.163:4306(rescs ...

  9. S02_CH04_User_IP实验Enter a post title

    S02_CH04_User_IP实验 4.1 创建IP 在之前的教程中,我们通过MIO与EMIO来控制LED,所使用的也是官方的IP,实际当中,官方提供的IP不可能涵盖到方方面面,用户需要自己编写硬件 ...

  10. MySQL 聚合函数(四)检测功能依赖

    源自MySQL 5.7 官方手册:12.20.4 Detection of Functional Dependence 本节提供了MySQL检测功能依赖的方式的几个示例.这些示例使用此表示法: {X} ...