[SCOI2015]小凸想跑步
题目描述
小凸晚上喜欢到操场跑步,今天他跑完两圈之后,他玩起了这样一个游戏。
操场是个凸 n 边形, nn 个顶点按照逆时针从 0 ∼n−1 编号。现在小凸随机站在操场中的某个位置,标记为p点。将 p 点与 n个顶点各连一条边,形成 n个三角形。如果这时p 点, 0号点, 1号点形成的三角形的面 积是 n个三角形中最小的一个,小凸则认为这是一次正确站位。
现在小凸想知道他一次站位正确的概率是多少。
题解
我们其实是要找到一个p点,使得pp0*pp1<=ppi*ppi+1.
然后我们把上面的式子展开,然后化简,这个不难就是挺麻烦的。
最后得到了Ax+By+C<=0的形式,然后可以用(-1e9,y1)(1e9,y2)这条直线来描述这个限制,再加上凸多边形的限制,跑个半平面交就好了。
注意:要特判A或B=0的情况。
代码
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cstdlib>
#define N 200009
#define double long double
#define eq(x,y) (fabs((x)-(y))<eps)
using namespace std;
const double eps=1e-;
int n,top,tot;
double x[N],y[N],s,S;
inline int rd(){
int x=;char c=getchar();bool f=;
while(!isdigit(c)){if(c=='-')f=;c=getchar();}
while(isdigit(c)){x=(x<<)+(x<<)+(c^);c=getchar();}
return f?-x:x;
}
struct point{
double x,y;
point(double xx=,double yy=){x=xx;y=yy;}
inline point operator +(const point &b)const{return point{x+b.x,y+b.y};}
inline point operator -(const point &b)const{return point{x-b.x,y-b.y};}
inline double operator *(const point &b)const{return x*b.y-y*b.x;}
inline point operator *(const double &b)const{return point{x*b,y*b};}
}p[N];
struct line{
point x,y;double ang;
line(double x1=,double x2=,double x3=,double x4=){x.x=x1;x.y=x2;y.x=x3;y.y=x4;}
bool operator <(const line &b)const{
if(fabs(ang-b.ang)<eps)return (y-x)*(b.x-x)<eps;
else return ang<b.ang;
}
}a[N],l[N],q[N];
inline bool left(point a,line b){return (a-b.x)*(b.y-b.x)>-eps;}
inline point jiao(line a,line b){return b.x+(b.y-b.x)*(((b.x-a.x)*(a.y-a.x))/((a.y-a.x)*(b.y-b.x)));}
int main(){
n=rd();
for(int i=;i<n;++i){
x[i]=rd(),y[i]=rd();
if(i)l[++top]=line(x[i-],y[i-],x[i],y[i]);
}
l[++top]=line(x[n-],y[n-],x[],y[]);
for(int i=;i<n;++i)S+=(point(x[i],y[i])-point(x[],y[]))*(point{x[i-],y[i-]}-point{x[],y[]})/;S=fabs(S);
x[n]=x[];y[n]=y[];
for(int i=;i<n;++i){
double a=-y[]+y[]+y[i+]-y[i],b=-x[]+x[]+x[i]-x[i+],c=x[]*y[]-x[]*y[]-x[i]*y[i+]+x[i+]*y[i];
if(fabs(b)<eps){
if(fabs(a)<eps){puts("0.0000");return ;}
double xf=-c/a,xs=xf,yf=,ys=1e15;
if(a<)l[++top]=line(xs,ys,xf,yf);
else l[++top]=line(xf,yf,xs,ys);
}
else{
double xf=-1e11,xs=1e11,yf=(-c-a*xf)/b,ys=(-c-a*xs)/b;
if(b>=){l[++top]=line(xs,ys,xf,yf);
}else l[++top]=line(xf,yf,xs,ys);
}
}
for(int i=;i<=top;++i)l[i].ang=atan2(l[i].y.y-l[i].x.y,l[i].y.x-l[i].x.x);
sort(l+,l+top+);
for(int i=;i<=top;++i)if(i==||fabs(l[i].ang-l[i-].ang)>eps)a[++tot]=l[i];
int h=,t=;q[]=a[];q[]=a[];p[]=jiao(a[],a[]);
for(int i=;i<=tot;++i){
while(h<t&&left(p[t-],a[i]))t--;
while(h<t&&left(p[h],a[i]))h++;
q[++t]=a[i];p[t-]=jiao(q[t-],q[t]);
}
while(h<t&&left(p[h],q[t]))h++;
while(h<t&&left(p[t-],q[h]))t--;
p[t]=jiao(q[t],q[h]);
for(int i=h+;i<=t;++i)s+=(p[i]-p[h])*(p[i-]-p[h])/;s=fabs(s);
printf("%.4LF",s/S);
return ;
}
[SCOI2015]小凸想跑步的更多相关文章
- 【BZOJ4445】[Scoi2015]小凸想跑步 半平面交
[BZOJ4445][Scoi2015]小凸想跑步 Description 小凸晚上喜欢到操场跑步,今天他跑完两圈之后,他玩起了这样一个游戏. 操场是个凸n边形,N个顶点按照逆时针从0-n-l编号.现 ...
- 【BZOJ4445】[SCOI2015]小凸想跑步(半平面交)
[BZOJ4445][SCOI2015]小凸想跑步(半平面交) 题面 BZOJ 洛谷 题解 首先把点给设出来,\(A(x_a,y_a),B(x_b,y_b),C(x_c,y_c),D(x_d,y_d) ...
- BZOJ 4445 [Scoi2015]小凸想跑步:半平面交
传送门 题意 小凸晚上喜欢到操场跑步,今天他跑完两圈之后,他玩起了这样一个游戏. 操场是个凸 $ n $ 边形,$ n $ 个顶点 $ P_i $ 按照逆时针从 $ 0 $ 至 $ n-1 $ 编号. ...
- bzoj 4445 [SCOI2015] 小凸想跑步
题目大意:一个凸包,随机一个点使得其与前两个点组成的面积比与其他相邻两个点组成的面积小的概率 根据题意列方程,最后求n条直线的交的面积与原凸包面积的比值 #include<bits/stdc++ ...
- 洛谷P4250 [SCOI2015]小凸想跑步(半平面交)
题面 传送门 题解 设\(p\)点坐标为\(x_p,y_p\),那么根据叉积可以算出它与\((i,i+1)\)构成的三角形的面积 为了保证\(p\)与\((0,1)\)构成的面积最小,就相当于它比其它 ...
- BZOJ4445: [Scoi2015]小凸想跑步
裸半平面交. 记得把P0P1表示的半平面加进去,否则点可能在多边形外. #include<bits/stdc++.h> #define N 100009 using namespace s ...
- BZOJ4445 SCOI2015小凸想跑步(半平面交)
考虑怎样的点满足条件.设其为(xp,yp),则要满足(x0-xp,y0-yp)×(x1-xp,y1-yp)<=(xi-xp,yi-yp)×(xi+1-xp,yi+1-yp)对任意i成立.拆开式子 ...
- 2018.10.15 bzoj4445: [Scoi2015]小凸想跑步(半平面交)
传送门 话说去年的省选计算几何难度跟前几年比起来根本不能做啊(虽然去年考的时候并没有学过计算几何) 这题就是推个式子然后上半平面交就做完了. 什么? 怎么推式子? 先把题目的概率转换成求出可行区域. ...
- [bzoj4445] [SCOI2015]小凸想跑步 (半平面交)
题意:凸包上一个点\(p\),使得\(p\)和点\(0,1\)组成的三角形面积最小 用叉积来求: \(p,i,i+1\)组成的三角形面积为: (\(\times\)为叉积) \((p_p-i)\tim ...
随机推荐
- Mac 终端 显示隐藏文件
defaults write com.apple.finder AppleShowAllFiles Yes && killall Finder //显示隐藏文件 defaults wr ...
- Windows编译OpenCV4Android解决undefined reference to std错误
注意OpenCV 4.0.1 解决了这个问题请直接下载OpenCV 4.0.1 但是OpenCV 4.0.1作为模块导入Android Studio会有找不到R.styleable的问题 OpenCV ...
- 手持式停车收费管理系统全套案例,支持车牌识别,包含了android版app,微信小程序查询,响应式管理后台,云端大数据存储
先展示几个app效果图片吧,使用起来非常方便,关联了机器的快捷键操作,操作速度提高了不少,摄像头车牌自动识别,车牌识别无网络情况下离线也可以使用 再来一张后台截图,停车场信息完整显示,今日数据实时 ...
- MongoDB分片 在部署和维护管理 中常见事项的总结
分片(sharding)是MongoDB将大型集合分割到不同服务器(或者说集群)上所采用的方法,主要为应对高吞吐量与大数据量的应用场景提供了方法. 和既有的分库分表.分区方案相比,MongoDB的最大 ...
- 使用VsCode自带的Emmet语法
新建html文件,保存之后,输入"!",按Tap(或Enter)键,自动生成HTML结构 标签只要直接输入标签名(不要输入<>),按Tap(或Enter)键自动生成完整 ...
- Linux内存描述之内存区域zone--Linux内存管理(三)
1 内存管理域zone 为了支持NUMA模型,也即CPU对不同内存单元的访问时间可能不同,此时系统的物理内存被划分为几个节点(node), 一个node对应一个内存簇bank,即每个内存簇被认为是一个 ...
- c/c++ linux 进程间通信系列1,使用signal,kill
linux 进程间通信系列1,使用signal,kill 信号基本概念: 软中断信号(signal,又简称为信号)用来通知进程发生了异步事件.进程之间可以互相通过系统调用kill发送软中断信号.内核 ...
- Windows 下安装drozer(Windows 10),连接手机(红米note4X)
Windows 下安装drozer(Windows 10),连接手机(红米note4X) 首先下载drozer(http://mwr.to/drozer). 红米手机开发者模式 遇到第一个问题,红米手 ...
- Saltstack_使用指南03_配置管理
1. 主机规划 注意事项 修改了master或者minion的配置文件,那么必须重启对应的服务. 2. 了解YAML 具体地址 https://docs.saltstack.com/en/latest ...
- python进阶之time模块详解
Time模块 Time模块包含的函数 Time模块包含了一下内置的函数,既有时间处理的,也有转换时间格式的: 序号 函数及描述 1 time.altzone 返回格林威治西部的夏令时地区的偏移秒数.如 ...