题目大意:

给定n个爆破点的信息 x y r w

表示爆破点位置为 (x,y) 爆破范围是以位置为圆心 半径为r的圆 引爆这个点的代价为w

引爆某个点时 其他位置在该爆破范围内的爆破点也会被引爆

求引爆所有爆破点的最小的爆破代价

这道题跟 上一篇的 OJ 22833(POJ 2186) 差不多

那题是缩点再计算出度 这题是缩点再计算入度

以爆破关系建图 即若引爆 i 点能使 j 点被引爆 那么连一条 i 到 j 的边

若存在点 k 的位置被包含在点 j 的爆破范围内,点 j 的位置被包含在点 i 的爆破范围内,但点 k 的位置不被包含在点 i 的爆破范围内

此时若引爆 i 点 那么 j 点会被引爆,而 j 点被引爆 k 点也会被引爆,即爆破存在传递性

所以求完这个图的强联通分量并缩点后 此时引爆这个图的里一个强联通分量

那么这个强联通分量能到达的其他强联通分量也会被引爆

又因为一个强联通分量内任意两点能互达 所以引爆一个强联通分量只需要引爆它内部的其中一个点

那么引爆一个强联通分量 应该贪心地选择它内部爆破代价最小的那个点

所以此时需要引爆的就是那些 入度为0的强联通分量内爆破代价最小的点 (由它们开始发生连环引爆)(图内的独立点入度也为0)

#include <stdio.h>
#include <cstring>
#include <algorithm>
#include <stack>
#define LL long long
#define INF 0x3f3f3f3f
using namespace std; const int N=;
struct EDGE { int to, nt; }e[N*N];
int head[N], tot;
int dfn[N], low[N], ind;
int col[N], id;
bool vis[N];
stack <int> s; int n, m, du[N];
LL x[N], y[N], r[N];
LL w[N], val[N]; void init() {
while(!s.empty()) s.pop();
for(int i=;i<=n;i++) {
head[i]=dfn[i]=low[i]=col[i]=-;
vis[i]=du[i]=; val[i]=INF;
}
tot=ind=id=;
}
void addE(int u,int v) {
e[tot].to=v;
e[tot].nt=head[u];
head[u]=tot++;
} void tarjan(int u) {
dfn[u]=low[u]=ind++;
s.push(u); vis[u]=;
for(int i=head[u];i!=-;i=e[i].nt) {
int v=e[i].to;
if(dfn[v]==-) {
tarjan(v);
low[u]=min(low[u],low[v]);
}
else if(vis[v])
low[u]=min(low[u],low[v]);
}
if(dfn[u]==low[u]) {
col[u]=++id;
vis[u]=;
while(s.top()!=u) {
col[s.top()]=id;
vis[s.top()]=;
s.pop();
} s.pop();
}
} bool uni(int i,int j) {
LL dis=(x[i]-x[j])*(x[i]-x[j])+(y[i]-y[j])*(y[i]-y[j]);
if(dis<=r[i]*r[i]) return ;
return ;
} int main()
{
int t, tcase=;
scanf("%d",&t);
while(t--) {
scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%lld%lld%lld%lld",&x[i],&y[i],&r[i],&w[i]);
init();
for(int i=;i<=n;i++)
for(int j=i+;j<=n;j++) {
if(uni(i,j)) addE(i,j); // j在i的爆破范围内 连i->j
if(uni(j,i)) addE(j,i); // i在j的爆破范围内 连j->i
} for(int i=;i<=n;i++)
if(dfn[i]==-) tarjan(i); for(int i=;i<=n;i++) {
val[col[i]]=min(val[col[i]],w[i]);
// 颜色相同说明在同个强联通分量内
// 保存这个强联通分量内爆破代价最小的点
for(int j=head[i];j!=-;j=e[j].nt)
if(col[e[j].to]!=col[i])
du[col[e[j].to]]++; // 计算各个强联通分量的入度
} LL ans=0LL;
for(int i=;i<=id;i++)
if(du[i]==) ans+=val[i]; // 入度为0 引爆
printf("Case #%d: %lld\n",++tcase,ans);
} return ;
}

最新文章

  1. 兼容javascript和C#的RSA加密解密算法,对web提交的数据进行加密传输
  2. zookeeper 故障重连机制
  3. 兵家必争之地——关于O2O商业模式的一点遐想
  4. js函数动态传参
  5. C#实现对指定文件夹中文件按修改时间排序
  6. vim技巧之快速进入引号删除至右引号前的内容
  7. CSS笔记(十三)CSS3之过渡
  8. MySQL数据库最大连接数
  9. PHP CURL参数详解
  10. chain pattern
  11. [ACM] HDU 3395 Special Fish (最大重量二分图匹配,KM算法)
  12. postman断言作用及怎么使用
  13. 开发servlet三种方式
  14. svg 图片
  15. mybatis框架(1)---mybatis入门
  16. Ubuntu 下 Galera cluster for MySQL 集群安装
  17. 【blog】SpringBoot普通类中如何获取其他bean例如Service、Dao
  18. 关掉 ubuntu bug 报告功能
  19. 矩阵NumPy
  20. [转] HTML5利用WebRTC的getUserMedia获取摄像头信息模拟拍照及视频(完整示例)

热门文章

  1. Python的从头再来
  2. 10. Tasks and functions
  3. Gerrit(0): Install and Config
  4. 【react】---Hooks的基本使用---【巷子】
  5. MFC VC 双缓冲绘图基本原理与实现,详细解释
  6. 高程(三)----数组Array
  7. Summary 报告
  8. 利用left join 筛选B表中不包含A表记录
  9. 笔记30 视图解析 ——TilesViewResolver
  10. nodejs mysql 连接数据库