题目描述

WJJ喜欢“魔兽争霸”这个游戏。在游戏中,巫妖是一种强大的英雄,它的技能Frozen Nova每次可以杀死一个小精灵。我们认为,巫妖和小精灵都可以看成是平面上的点。 当巫妖和小精灵之间的直线距离不超过R,且巫妖看到小精灵的视线没有被树木阻挡(也就是说,巫妖和小精灵的连线与任何树木都没有公共点)的话,巫妖就可以瞬间杀灭一个小精灵。 在森林里有N个巫妖,每个巫妖释放Frozen Nova之后,都需要等待一段时间,才能再次施放。不同的巫妖有不同的等待时间和施法范围,但相同的是,每次施放都可以杀死一个小精灵。 现在巫妖的头目想知道,若从0时刻开始计算,至少需要花费多少时间,可以杀死所有的小精灵?

输入

输入文件第一行包含三个整数N、M、K(N,M,K<=200),分别代表巫妖的数量、小精灵的数量和树木的数量。 接下来N行,每行包含四个整数x, y, r, t,分别代表了每个巫妖的坐标、攻击范围和施法间隔(单位为秒)。 再接下来M行,每行两个整数x, y,分别代表了每个小精灵的坐标。 再接下来K行,每行三个整数x, y, r,分别代表了每个树木的坐标。 输入数据中所有坐标范围绝对值不超过10000,半径和施法间隔不超过20000。

输出

输出一行,为消灭所有小精灵的最短时间(以秒计算)。如果永远无法消灭所有的小精灵,则输出-1。

样例输入

2 3 1
-100 0 100 3
100 0 100 5
-100 -10
100 10
110 11
5 5 10

样例输出

5


题解

计算几何+二分+网络流最大流

首先要解决的是是否能够攻击到,如果两个点形成的线段与所有圆都没有公共点,那么就可以攻击到。

线段与圆有公共点,需要满足两个条件之一:(1)线段某端点在圆内 (2)直线与圆有公共点,且以线段和端点与圆心连线的夹角是锐角。

其中直线与圆有公共点可以使用点到直线距离公式:$\frac{|ax_0+by_0+c|}{\sqrt{a^2+b^2}}$,夹角是锐角可以使用余弦定理的推论:如果$a^2<b^2+c^2$,那么$A$是锐角。

判断完以后就是经典的二分+最大流判断问题了:二分时间,S向巫妖连边,容量为mid时间攻击次数;巫妖向能够攻击到的小精灵连边,容量为1;小精灵向T连边,容量为1。如果满流则mid可行,否则不可行。

注意攻击次数的计算公式:$\lfloor\frac{mid}t\rfloor+1$,即开场是没有冷却的。(为这个问题纠结了半天= =)

  1. #include <queue>
  2. #include <cstdio>
  3. #include <cstring>
  4. #define N 410
  5. #define M 200010
  6. using namespace std;
  7. typedef long long ll;
  8. queue<int> q;
  9. ll xn[N] , yn[N] , rn[N] , tn[N] , xm[N] , ym[N] , xk[N] , yk[N] , rk[N];
  10. int n , m , k , v[N][N] , head[N] , to[M] , val[M] , next[M] , cnt , s , t , dis[N];
  11. inline ll squ(ll x)
  12. {
  13. return x * x;
  14. }
  15. bool connect(int a , int b)
  16. {
  17. if(squ(xn[a] - xm[b]) + squ(yn[a] - ym[b]) > squ(rn[a])) return 0;
  18. int i;
  19. for(i = 1 ; i <= k ; i ++ )
  20. if(squ(xn[a] - xk[i]) + squ(yn[a] - yk[i]) <= squ(rk[i]) || squ(xm[b] - xk[i]) + squ(ym[b] - yk[i]) <= squ(rk[i]) || (
  21. squ(yn[a] * (xk[i] - xm[b]) - ym[b] * (xk[i] - xn[a]) - yk[i] * (xn[a] - xm[b])) <= squ(rk[i]) * (squ(xn[a] - xm[b]) + squ(yn[a] - ym[b])) &&
  22. squ(xn[a] - xm[b]) + squ(yn[a] - ym[b]) + squ(xn[a] - xk[i]) + squ(yn[a] - yk[i]) >= squ(xm[b] - xk[i]) + squ(ym[b] - yk[i]) &&
  23. squ(xn[a] - xm[b]) + squ(yn[a] - ym[b]) + squ(xm[b] - xk[i]) + squ(ym[b] - yk[i]) >= squ(xn[a] - xk[i]) + squ(yn[a] - yk[i])))
  24. return 0;
  25. return 1;
  26. }
  27. void add(int x , int y , int z)
  28. {
  29. to[++cnt] = y , val[cnt] = z , next[cnt] = head[x] , head[x] = cnt;
  30. to[++cnt] = x , val[cnt] = 0 , next[cnt] = head[y] , head[y] = cnt;
  31. }
  32. bool bfs()
  33. {
  34. int x , i;
  35. memset(dis , 0 , sizeof(dis));
  36. while(!q.empty()) q.pop();
  37. dis[s] = 1 , q.push(s);
  38. while(!q.empty())
  39. {
  40. x = q.front() , q.pop();
  41. for(i = head[x] ; i ; i = next[i])
  42. {
  43. if(val[i] && !dis[to[i]])
  44. {
  45. dis[to[i]] = dis[x] + 1;
  46. if(to[i] == t) return 1;
  47. q.push(to[i]);
  48. }
  49. }
  50. }
  51. return 0;
  52. }
  53. int dinic(int x , int low)
  54. {
  55. if(x == t) return low;
  56. int temp = low , i , k;
  57. for(i = head[x] ; i ; i = next[i])
  58. {
  59. if(val[i] && dis[to[i]] == dis[x] + 1)
  60. {
  61. k = dinic(to[i] , min(temp , val[i]));
  62. if(!k) dis[to[i]] = 0;
  63. val[i] -= k , val[i ^ 1] += k;
  64. if(!(temp -= k)) break;
  65. }
  66. }
  67. return low - temp;
  68. }
  69. bool judge(int mid)
  70. {
  71. int i , j , sum = 0;
  72. memset(head , 0 , sizeof(head)) , cnt = 1;
  73. for(i = 1 ; i <= n ; i ++ ) add(s , i , mid / tn[i] + 1);
  74. for(i = 1 ; i <= m ; i ++ ) add(i + n , t , 1);
  75. for(i = 1 ; i <= n ; i ++ )
  76. for(j = 1 ; j <= m ; j ++ )
  77. if(v[i][j])
  78. add(i , j + n , 1);
  79. while(bfs()) sum += dinic(s , 1 << 30);
  80. return sum == m;
  81. }
  82. int main()
  83. {
  84. int i , j , l = 0 , r = 2000000 , mid , ans = -1;
  85. scanf("%d%d%d" , &n , &m , &k) , s = 0 , t = n + m + 1;
  86. for(i = 1 ; i <= n ; i ++ ) scanf("%lld%lld%lld%lld" , &xn[i] , &yn[i] , &rn[i] , &tn[i]);
  87. for(i = 1 ; i <= m ; i ++ ) scanf("%lld%lld" , &xm[i] , &ym[i]);
  88. for(i = 1 ; i <= k ; i ++ ) scanf("%lld%lld%lld" , &xk[i] , &yk[i] , &rk[i]);
  89. for(i = 1 ; i <= n ; i ++ )
  90. for(j = 1 ; j <= m ; j ++ )
  91. v[i][j] = connect(i , j);
  92. while(l <= r)
  93. {
  94. mid = (l + r) >> 1;
  95. if(judge(mid)) ans = mid , r = mid - 1;
  96. else l = mid + 1;
  97. }
  98. printf("%d\n" , ans);
  99. return 0;
  100. }

【bzoj1822】[JSOI2010]Frozen Nova 冷冻波 计算几何+二分+网络流最大流的更多相关文章

  1. 【BZOJ1822】[JSOI2010]Frozen Nova 冷冻波 几何+二分+网络流

    [BZOJ1822][JSOI2010]Frozen Nova 冷冻波 Description WJJ喜欢“魔兽争霸”这个游戏.在游戏中,巫妖是一种强大的英雄,它的技能Frozen Nova每次可以杀 ...

  2. Bzoj1822 [JSOI2010]Frozen Nova 冷冻波

    Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 1933  Solved: 608 Description WJJ喜欢“魔兽争霸”这个游戏.在游戏中,巫妖 ...

  3. bzoj1822: [JSOI2010]Frozen Nova 冷冻波网络流

    思路比较显然:二分答案,流流流 但是实现的时候感觉自己数学捉急.. 一开始算了个直线到点距离.... 应该是线段到点距离 #include <bits/stdc++.h> #define ...

  4. 【计算几何】【二分答案】【最大流】bzoj1822 [JSOI2010]Frozen Nova 冷冻波

    用三角形面积什么的算算点到直线的距离之类……其实相切的情况是可行的……剩下的就跟某SDOI2015一样了. #include<cstdio> #include<cmath> # ...

  5. BZOJ1822 [JSOI2010]Frozen Nova 冷冻波 二分+最大流

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=1822 题解 好久没做网络流的,都没有想到网络流... 首先暴力判断一下一个巫妖和一个精灵之间能 ...

  6. 1822: [JSOI2010]Frozen Nova 冷冻波 二分最大流

    1822: [JSOI2010]Frozen Nova 冷冻波 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 585  Solved: 175[Subm ...

  7. BZOJ 1822[JSOI2010]Frozen Nova 冷冻波

    网络流+二分. n^3枚举判断每个巫妖可以攻击的精灵,向其连1的边,每个精灵向汇点连1的边. 二分答案,修改源点流向每个巫妖的cap,跑最大流看是否等于精灵数. 恩,看起来没什么毛病. 然后狂WA不止 ...

  8. BZOJ-1822 Frozen Nova 冷冻波 计(jie)算(xi)几何+二分+最大流判定+经典建图

    这道逼题!感受到了数学对我的深深恶意(#‵′).... 1822: [JSOI2010]Frozen Nova 冷冻波 Time Limit: 10 Sec Memory Limit: 64 MB S ...

  9. BZOJ1822 Frozen Nova 冷冻波

    1822: [JSOI2010]Frozen Nova 冷冻波 Time Limit: 10 Sec  Memory Limit: 64 MB Description WJJ喜欢“魔兽争霸”这个游戏. ...

随机推荐

  1. UVA - 1639 Candy (概率,精度)

    X表示剩下的糖数量,如果最后打开的是p对应的盒子.划分:Xi表示剩下i个糖,最后一次选的概率为p, 前面的服从二项分布.根据全概率公式和期望的线性性,求和就好了. 精度处理要小心,n很大,组合数会很大 ...

  2. 【BZOJ4196】[NOI2015] 软件包管理器(树链剖分)

    点此看题面 大致题意: 有\(n\)个软件包,它们的依赖关系形成一棵树.现在,问你安装或卸载一个软件包,会影响多少个软件包的安装状态. 树链剖分 这道题应该是 树链剖分 算法比较入门的题目吧. 对于安 ...

  3. 在mininet上基于ovs,ovx,pox搭建三点虚拟网络

    本次试验基于mininet平台,在平台中利用ovs新建1个交换机,以pox为控制器,ovx作为中间层实现虚拟化. 基础请参照http://ovx.onlab.us/getting-started/tu ...

  4. el-upload控件一次接口请求上传多个文件

    el-upload组件默认情况下上传多少个文件就会请求多少次上传接口,如何一次上传多个文件而不必多次请求上传接口呢?直接看代码 html <el-upload :action="act ...

  5. Uva 组装电脑 12124 - Assemble

    主要运用二分法查找最优解 #include<iostream> #include<string> #include<vector> #include<map& ...

  6. Linux IO调度方法

    目录 I/O调度的4种算法 I/O调度程序的测试 ionice IO调度器的总体目标是希望让磁头能够总是往一个方向移动,移动到底了再往反方向走,这恰恰就是现实生活中的电梯模型,所以IO调度器也被叫做电 ...

  7. hashlib模块常用功能

    什么是hash hash是一种算法,该算法接受传入的内容,经过运算得到一串hash值 如果把hash算法比喻为一座工厂 那传给hash算法的内容就是原材料 生成的hash值就是生产出的产品 2.为何要 ...

  8. 数据结构-单链表(Linked List)

    #include <stdio.h> #include <stdlib.h> #define LIST_INIT_SIZE 10 #define LISTINCREMENT 1 ...

  9. Crane UVA - 1611 思路+构造

    题目:题目链接 思路:思路+构造,假设 i  在pos 位置,那么如果 (pos-i-1)*2+i+1 <= n,那么可以操作一次换过来,如果他们之间元素个数是偶数,那么交换 i - pos,如 ...

  10. 笔记-爬虫-robots.txt

    笔记-爬虫-robots.txt 1.      robots.txt文件简介 1.1.    是什么 robots.txt是用来告诉搜索引擎网站上哪些内容可以被访问.哪些不能被访问.当搜索引擎访问一 ...