2017 Multi-University Training Contest - Team 9 1003&&HDU 6163 CSGO【计算几何】
CSGO
Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)
Total Submission(s): 127 Accepted Submission(s): 20

Senior
Pan is crazy about CSGO and she is so skilled. She can kill every enemy
she spots at the same time, even if they are in different positions.
There
are some walls in a map. The locations of enemies are represented by
some points and walls are represented by some line segments in the XY
plane. The position of senior Pan is also described as a point. Senior
Pan can spot an enemy if and only if the segment between them doesn’t
intersect with any wall segment. But if there are other enemies on this
segment, she can kill all of them.
For some given positions, senior
Pan wants to know how many enemies she can kill in these positions. Your
Task is to write a program to figure it out.
∙
The first line contains three integers N, M, Q representing
respectively the number of enemies, the number of walls and the number
of positions senior Pan chooses. For the next N lines, each line contain
two integers X and Y, indicating the location of enemies is (X, Y). The
next M lines, each line contain four integers X1, Y1, X2, Y2,
indicating two endpoints of the wall are (X1, Y1) and (X2, Y2). The last
Q lines, each line contain two integers X and Y, indicating the
location of senior Pan chooses is (X, Y).
∙
You can assume that walls don’t intersect even if in their endpoints,
no enemies have the same location and no enemies are in walls.
∙ N, M <= 10 ^ 4.
∙ Q <= 12.
∙ -10 ^ 6 <= X, Y, X1, Y1, X2, Y2 <= 10 ^ 6
For
next Q lines, each line output a number which represents the number of
enemies senior Pan can kill in her i-th chosen location.
3 2 1
1 1
2 2
-1 1
0 1 -1 0
2 -2 2 0
0 0
5 4 2
29 -8
19 33
-46 -44
-38 19
9 -20
40 45 38 18
9 -32 -8 46
33 20 35 -19
22 17 -5 40
19 -38
-17 -21
Case #1:
2
Case #2:
3
2
平面上有一些点和一些线段,保证这些线段互不相交,点不在线段上。
每次问从一个点能看到多少个点。一个点能看到另一点当且仅当这两点连线的线段不和任何一个已知的线段相交。(这里的相交均指非规范相交)
每次以询问点为极点极角排序。线段两端点拆成两个事件,一次出现,一次消失。对于枚举的某一个角度,能否看见当前角度的点仅取决于当前角度上离极点最近的线段,由于保证线段不相交,一条线段在插入时和还存在线段的相对位置是不会改变的,所以可以用set维护,每次先处理当前角度上所有的点是否被线段遮挡就可以了。
+ M)log(N + M))。
#include<bits/stdc++.h> using namespace std;
const int N = ;
const double eps = 1e-;
const double pi = acos(-1.0);
typedef complex<double> Point; double Det(const Point & a, const Point & b) {return (conj(a) * b).imag();}
double Dot(const Point & a, const Point & b) {return (conj(a) * b).real();}
int sgn(double x) {if(fabs(x) < eps) return ; if(x > eps) return ; return -;} double _ang;
Point ori = (Point) {, };
struct Line :public vector<Point>{
double k;
Line(){}
Line(Point a, Point b){
push_back(a), push_back(b);
k = atan2((b - a).imag(), (b - a).real());
}
};
Point Vec(Line a) {return a[] - a[];}
Point LineIntersection(const Line & a, const Line & b){
double k1 = Det(Vec(a), Vec(b));
double k2 = Det(Vec(b), a[] - b[]);
if(!sgn(k1)){
if(sgn(abs(a[] - b[]) - abs(a[] - b[])) > ) return a[];
else return a[];
}
return a[] + Vec(a) * k2 / k1;
} bool operator < (const Line & a, const Line & b){
Line cur = (Line) {ori, (Point) {cos(_ang), sin(_ang)}};
if(sgn(abs(LineIntersection(a, cur)) - abs(LineIntersection(b, cur))) < ) return ;
return ;
} struct Event{
double k;
int id, typ;
bool operator < (const Event & a) const{
if(sgn(k - a.k)) return k < a.k;
return typ < a.typ;
}
}; double CalcAng(const Point & a) {return atan2(a.imag(), a.real());}
Point p[N];
Line w[N], seg[N];
Event e[N << ]; int q, n, m, tot;
set<Line> s;
int rec[N]; int Calc(int x){
tot = ;
s.clear();
for(int i = ; i <= n; i ++)
e[++ tot] = (Event){CalcAng(p[i] - p[x]), i, }; bool flag = ;
for(int i = ; i <= m; i++){
seg[i] = (Line) {w[i][] - p[x], w[i][] - p[x]};
if(sgn(CalcAng(seg[i][]) - CalcAng(seg[i][])) > ) swap(seg[i][], seg[i][]);
flag = ;
if(sgn(Det(Vec(seg[i]), Point(-, ))) != && sgn(abs(CalcAng(seg[i][]) - CalcAng(seg[i][])) - pi) > ) flag = ;
if(flag) e[++ tot] = (Event) {CalcAng(seg[i][]), i, }, e[++tot] = (Event) {CalcAng(seg[i][]), i, };
else{
e[++ tot] = (Event) {-pi, i, };
e[++ tot] = (Event) {CalcAng(seg[i][]), i, };
e[++ tot] = (Event) {CalcAng(seg[i][]), i, };
e[++ tot] = (Event) {pi, i, };
}
} sort(e + , e + tot + );
int cnt = ;
Point dir, dd;
Line t;
for(int i = ; i <= tot; i ++){
_ang = e[i].k;
if(e[i].typ & ){
dir = (Point) {cos(_ang), sin(_ang)};
if(s.empty() || sgn(abs(LineIntersection(*s.begin(), (Line) {ori, dir})) - abs(p[e[i].id] - p[x])) > ){
cnt ++, rec[cnt] = e[i].id;
}
}
else if(!e[i].typ) s.insert(seg[e[i].id]);
else s.erase(seg[e[i].id]);
} return cnt;
} int Tcase; char fi[] = "pcx.in", fo[] = "pcx.ans";
void Solve(){
printf("Case #%d:\n", ++ Tcase);
double x, y;
Point a, b;
for(int i = ; i <= n; i ++){
scanf("%lf%lf", &x, &y);
p[i] = (Point) {x, y};
}
for(int i = ; i <= m; i ++){
scanf("%lf%lf", &x, &y);
a = (Point) {x, y};
scanf("%lf%lf", &x, &y);
b = (Point) {x, y};
w[i] = (Line) {a, b};
} int ans;
for(int i = ; i <= q; i ++){
scanf("%lf%lf", &x, &y);
p[n + ] = (Point) {x, y};
ans = Calc(n + );
printf("%d\n", ans);
} return;
} int main(){
// freopen("_pc.in", "r", stdin);
// freopen("_pc.ans", "w", stdout);
while(~scanf("%d%d%d", &n, &m, &q))
Solve();
//printf("--->%lf\n", (double) clock() / CLOCKS_PER_SEC);
return ;
}
2017 Multi-University Training Contest - Team 9 1003&&HDU 6163 CSGO【计算几何】的更多相关文章
- 2017 Multi-University Training Contest - Team 1 1003&&HDU 6035 Colorful Tree【树形dp】
Colorful Tree Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)T ...
- 2017 Multi-University Training Contest - Team 9 1005&&HDU 6165 FFF at Valentine【强联通缩点+拓扑排序】
FFF at Valentine Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) ...
- 2017 Multi-University Training Contest - Team 9 1004&&HDU 6164 Dying Light【数学+模拟】
Dying Light Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)Tot ...
- 2017 Multi-University Training Contest - Team 9 1002&&HDU 6162 Ch’s gift【树链部分+线段树】
Ch’s gift Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total S ...
- 2017 Multi-University Training Contest - Team 9 1001&&HDU 6161 Big binary tree【树形dp+hash】
Big binary tree Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)T ...
- 2017 Multi-University Training Contest - Team 1 1006&&HDU 6038 Function【DFS+数论】
Function Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)Total ...
- 2017 Multi-University Training Contest - Team 1 1002&&HDU 6034 Balala Power!【字符串,贪心+排序】
Balala Power! Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)T ...
- 2017 Multi-University Training Contest - Team 1 1011&&HDU 6043 KazaQ's Socks【规律题,数学,水】
KazaQ's Socks Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)T ...
- 2017 Multi-University Training Contest - Team 1 1001&&HDU 6033 Add More Zero【签到题,数学,水】
Add More Zero Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)T ...
随机推荐
- MVC系列——一个异常消息传递引发的思考
前言:最近在某个项目里面遇到一个有点纠结的小问题,经过半天时间的思索和尝试,问题得到解决.在此记录一下解决的过程,以及解决问题的过程中对.net里面MVC异常处理的思考.都是些老生常谈的问题,不多说, ...
- 【java设计模式】【行为模式Behavioral Pattern】模板方法模式Template Method Pattern
package com.tn.pattern; public class Client { public static void main(String[] args) { AbstractClass ...
- ASP.NET Core使用SkiaSharp实现验证码
前言 本文并没有实现一个完成的验证码样例,只是提供了在当前.NET Core 2.0下使用Drawing API的另一种思路,并以简单Demo的形式展示出来. Skia Skia是一个开源的二维图形库 ...
- PE格式第三讲扩展,VA,RVA,FA(RAW),模块地址的概念
PE格式第三讲扩展,VA,RVA,FA的概念 作者:IBinary出处:http://www.cnblogs.com/iBinary/版权所有,欢迎保留原文链接进行转载:) 一丶VA概念 VA (vi ...
- lesson - 5 Linux用户和组管理
1. /etc/passwd由 : 分隔成7个字段(1) 用户名 规则:大小写字母.数字.减号(不能出现在首位).点以及下划线,其他字符不合法 (2) x 放密码,安全起见放到 /etc/shadow ...
- js随机数生成,生成m-n的随机数
使用js生成n到m间的随机数字,主要目的是为后期的js生成验证码做准备,Math.random()函数返回0和1之间的伪随机数 var random = Math.random(); console. ...
- django2 快速安装指南
django2 快速安装指南 1. 安装 作为一个 Python Web 框架,Django需要Python的支持.请参阅 我可以在Django中使用哪些Python版本?了解详情.Python包含一 ...
- Q:算法(第四版)—第一章
1.1.14:编写一个静态方法lg(),接受一个整型参数N,返回不大于log2N的最大整数(ps:不使用Math库) 分析: 利用将公式k=log2N转化为N=2k的原理,不断的逼近其输入的值N,当N ...
- python科学计算_numpy_广播与下标
多维数组下标 多维数组的下标是用元组来实现每一个维度的,如果元组的长度比维度大则会出错,如果小,则默认元组后面补 : 表示全部访问: 如果一个下标不是元组,则先转换为元组,在转换过程中,列表和数组的转 ...
- vue2 过渡 轮播图
---恢复内容开始--- Vue主要渲染条件: v-if:是将元素删除再创造出来进行渲染. v-show:是将元素的display=none掉,再进行渲染: 要点知识:v-key:唯一元素标识,若不设 ...