HNOI2019 鱼 fish
本来想写个改题记录的然后想了想改不完所以就分开写了= =
https://www.luogu.org/problemnew/show/P5286
显然枚举A,D,然后鱼头和鱼身分开来考虑。
鱼身:先枚举B,C,那么BC的中点一定在线段AD(不包含端点)上,对于每一条直线维护一个vector存所有的点,将这个BC的中点插入进线段BC的垂直平分线的vector,然后对于一组AD,鱼身的方案数是vector上AD中间的点数,可以用upper_bound求。
鱼尾:枚举D,对所有其他点极角排序,用双指针扫一遍所有点即可。具体难以描述请直接看代码
这题细节巨多,考场上一看就会写,写了3h还没拍上以为暴力写挂走了,结果水到40分,如果没被卡常有60,然后今天又调了一个上午才过的= =说不定还有bug只是数据太水= =
// luogu-judger-enable-o2
#include<bits/stdc++.h>
#define il inline
#define rg register
#define vd void
#define ll long long
il int gi(){
int x=0,f=0;char ch=getchar();
while(!isdigit(ch))f^=ch=='-',ch=getchar();
while(isdigit(ch))x=x*10+ch-'0',ch=getchar();
return f?-x:x;
}
int n;
template<class T>il T gcd(T a,T b){return b?gcd(b,a%b):a;}
struct number{
ll x,y;
number(){}
number(const ll&a,const ll&b){
ll g=gcd(llabs(a),llabs(b));
x=a/g,y=b/g;
if(y<0)x=-x,y=-y;
}
};
il bool operator<(const number&a,const number&b){
if(a.x^b.x)return a.x<b.x;
return a.y<b.y;
}
il bool operator!=(const number&a,const number&b){return a.x!=b.x||a.y!=b.y;}
struct line{number k,b;};
il bool operator<(const line&a,const line&b){
if(a.k!=b.k)return a.k<b.k;
return a.b<b.b;
}
std::map<line,int>M;int cnt;
int x[1010],y[1010];
struct yyb{int i;double at2;}s[2010];
il bool operator<(const yyb&a,const yyb&b){return a.at2<b.at2;}
std::vector<int>vec[1000010];
il ll getdist(int a,int b){
return 1ll*(x[a]-x[b])*(x[a]-x[b])+1ll*(y[a]-y[b])*(y[a]-y[b]);
}
int tail[1010][1010];
std::map<ll,int>mmp;
int res,nowi;
const double eps=1e-10,pi=acos(-1);
il vd insert(int x){res+=mmp[getdist(nowi,x)]++;}
il vd delet(int x){res-=--mmp[getdist(nowi,x)];}
int main(){
//freopen("fish.in","r",stdin);
//freopen("fish.out","w",stdout);
n=gi();
for(int i=1;i<=n;++i)x[i]=gi(),y[i]=gi();
for(int i=1;i<=n;++i)
for(int j=i+1;j<=n;++j)
if(y[i]==y[j]){
if(x[i]+x[j]&1)continue;
number k=(number){1,0},b=(number){(x[i]+x[j])/2,1};
if(M.find((line){k,b})==M.end())M[(line){k,b}]=++cnt;
vec[M[(line){k,b}]].push_back(y[i]*2);
}else{
number k=(number){x[i]-x[j],y[j]-y[i]},b=(number){-1ll*(x[i]-x[j])*(x[i]+x[j])+1ll*(y[j]-y[i])*(y[j]+y[i]),2ll*(y[j]-y[i])};
if(M.find((line){k,b})==M.end())M[(line){k,b}]=++cnt;
vec[M[(line){k,b}]].push_back(x[i]==x[j]?x[i]*2:y[i]+y[j]);
}
for(int i=1;i<=n;++i){
int m=0;
for(int j=1;j<=n;++j)if(j!=i)s[++m]=(yyb){j,atan2(y[j]-y[i],x[j]-x[i])};
std::sort(s+1,s+m+1);
for(int i=1;i<=m;++i)s[i+m]=s[i],s[i+m].at2+=pi*2;
mmp.clear();
res=0;nowi=i;
for(int j=1,p=0,q=0;j<=m;++j){
while(s[p+1].at2+eps<s[j].at2+1.5*pi)insert(s[++p].i);
while(s[q+1].at2<s[j].at2+0.5*pi+eps)delet(s[++q].i);
tail[s[j].i][i]=res;
}
}
for(int i=1;i<=cnt;++i)std::sort(vec[i].begin(),vec[i].end());
ll ans=0;
for(int i=1;i<=n;++i)
for(int j=i+1;j<=n;++j){
int xa=y[i],xb=y[j];
if(xa==xb)xa=x[i],xb=x[j];
if(xa>xb)std::swap(xa,xb);
number k,b;
if(x[i]==x[j])k=(number){1,0},b=(number){x[i],1};
else k=(number){y[i]-y[j],x[i]-x[j]},b=(number){-1ll*x[i]*(y[i]-y[j])+1ll*y[i]*(x[i]-x[j]),x[i]-x[j]};
if(M.find((line){k,b})==M.end())continue;
int veci=M[(line){k,b}];
ans+=(std::upper_bound(vec[veci].begin(),vec[veci].end(),xb*2-1)-std::upper_bound(vec[veci].begin(),vec[veci].end(),xa*2))*(tail[i][j]+tail[j][i]);
}
printf("%lld\n",ans*4);
return 0;
}
HNOI2019 鱼 fish的更多相关文章
- [HNOI2019]鱼
Luogu5286 \(2019.4.14\),新生第一题,改了\(3\)个小时 题解-租酥雨,和出题人给的正解一模一样 枚举\(AD\),分别考虑鱼身\(BC\)和鱼尾\(EF\) 到\(E\),\ ...
- [HNOI2019]鱼(计算几何)
看到数据范围n<=1000,但感觉用O(n^2)不现实,所以考虑方向应该是O(n^2logn). 一种暴力做法:用vector存到1点相同的2点和到2点相同的1点,然后枚举A,枚举BC,再枚举D ...
- luogu P5286 [HNOI2019]鱼
传送门 这题真的牛皮,还好考场没去刚( 这题口胡起来真的简单 首先枚举D点,然后对其他所有点按极角排序,同时记录到D的距离.然后按照极角序枚举A,那么鱼尾的两个点的极角范围就是A关于D对称的那个向量, ...
- 【洛谷5286】[HNOI2019] 鱼(计算几何)
点此看题面 大致题意: 给你\(n\)个点,让你求鱼形图的数量. 核心思路 首先,考虑到\(n\)这么小,我们可以枚举线段\(AD\),再去找符合条件的\(BC,EF\). 然后,不难发现\(BC\) ...
- 个人对【依赖倒置(DIP)】、【控制反转(IOC)】、【依赖注入(DI)】浅显理解
一.依赖倒置(Dependency Inversion Principle) 依赖倒置是面向对象设计领域的一种软件设计原则.(其他的设计原则还有:单一职责原则.开放封闭原则.里式替换原则.接口分离原则 ...
- 人工鱼群算法超详细解析附带JAVA代码
01 前言 本着学习的心态,还是想把这个算法写一写,给大家科普一下的吧. 02 人工鱼群算法 2.1 定义 人工鱼群算法为山东大学副教授李晓磊2002年从鱼找寻食物的现象中表现的种种移动寻觅特点中得到 ...
- 第三十篇 面向对象的三大特性之继承 supre()
继承 一 .什么是继承? 类的继承跟现实生活中的父.子.孙子.重孙子的继承关系一样,父类又称基类. Python中类的继承分为:单继承 和 多继承. # 定义父类 class ParentClass ...
- Unity基础:AR(增强现实)的学习
版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...
- 【loj3054】【hnoi2019】鱼
题目 描述 难以描述.......慢慢看..: https://loj.ac/problem/3054 范围 $6 \le n \le 1000 , 1 \le |x| , |y| \ ...
随机推荐
- Git的安装配置(win环境)
安装: 首先安装win版本的git msysgit:https://git-for-windows.github.io 注:安装时要勾选生成桌面快捷方式. 默认安装完后依次执行: $ git conf ...
- Python CNN卷积神经网络代码实现
# -*- coding: utf-8 -*- """ Created on Wed Nov 21 17:32:28 2018 @author: zhen "& ...
- [20171206]rman与truncate2.txt
[20171206]rman与truncate2.txt --//上午测试发现truncate的表在做rman备份时还要做8个extents的备份.--//不知道自己的猜测是否正确,选择一个使用UNI ...
- python第四十二天 socket ---ssh
用scoket 写一个简版的ssh 服务端: #!usr/bin/env python #-*-coding:utf-8-*- # Author calmyan import socket,os s= ...
- AspNetCore2身份验证
1.在Startup类的Configure方法,添加身份验证的中间件AuthenticationMiddleware app.UseAuthentication(); 2.在Startup类的Conf ...
- 理解inode 以及 软链接和硬链接概念区分
inode简单理解 本文来源自网络文章,并针对文章内容加以批注和修改.希望能帮到你! 一. 磁盘设备 说到inode,首先必须要提及下<操作系统>中磁盘存储器的管理一节.磁盘设备是一种相当 ...
- Sqoop-1.4.7-部署与常见案例
该文章是基于 Hadoop2.7.6_01_部署 . Hive-1.2.1_01_安装部署 进行的 1. 前言 在一个完整的大数据处理系统中,除了hdfs+mapreduce+hive组成分析系统的核 ...
- MySQL基本简单操作02
MySQL基本简单操作 先进入Mysql容器. [root@promote ~]# docker exec -it mysql /bin/bash root@30d60b852cf5:/# mysql ...
- hubilder打包+C#服务端个推服务实现
关于推送鼓捣了好长时间,这里不再写helloworld了,只讲里面遇到的问题. 1.关于苹果开发者平台上的注册 网上很多的教程,只要按照步骤来设置就行了,在 iOS证书(.p12)和描述文件(.mob ...
- 17秋 软件工程 团队第五次作业 Alpha Scrum9
17秋 软件工程 团队第五次作业 Alpha Scrum9 今日完成的任务 世强:APP后端部门申请状态: 港晨:主页面代码实现: 树民:完善超级管理员web后端: 伟航:设置页面和侧边栏的原型: 陈 ...