描述

  这是一枚平凡的骰子。它是一个均质凸多面体,表面有n个端点,有f个面,每一面是一个凸多边形,且任意两面不共面。将这枚骰子抛向空中,骰子落地的时候不会发生二次弹跳(这是一种非常理想的情况)。你希望知道最终每一面着地的概率。
  每一面着地的概率可以用如下的方法计算:我们假设O为骰子的重心,并以O为球心,做半径为1的单位球面(记为S)。我们知道S的表面积即单位球的表面积,为4*pi,这里pi为圆周率。对于骰子的某一面C来说,球面S上存在一块区域T满足:当下落时若骰子所受重力方向与S的交点落在T中,则C就是最终着地的一面。那么C着地的概率为区域T的面积除以4*pi。

  为了能更好地辅助计算球面上一块区域的面积,我们给出单位球面 S 上三角形的面积计算公式。考虑单位球面 S 上的三个两两相交的大圆,交点依次为A,B 和 C。则曲面三角形 ABC 的面积为 Area(ABC)=alpha+beta+gamma-pi,其中 alpha,beta 和 gamma 分别对应了三个二面角的大小。如下图所示。

  我们保证:每一面着地的时候,重心的垂心都恰好在这一面内。也就是说不会出现摆不稳的情况。

格式

输入格式

  第一行输入两个整数,分别表示端点总数n与表面总数f,分别从1开始编号。
  之后n行,每行有三个浮点数x,y和z,给出了每一个端点的坐标。之后f行依次描述了每一块表面,首先给出不小于3的整数d,表示这一面的端点个数,之后d个整数按照逆时针方向(视角在骰子的外面)给出了每一个端点的编号。

输出格式

  输出f行,第i行有一个浮点数,表示第i个面着地的概率。
 本题中您的输出应该保留距离答案最近的7位小数,即在需要保留7位小数的前提之下与标准答案最接近。数据保证可以避免对小数点后第八位四舍五入后产生的精度误差。

样例1

样例输入1

8 6
1 0 0
1 1 0
1 0 1
1 1 1
0 0 0
0 1 0
0 0 1
0 1 1
4 1 2 4 3
4 2 6 8 4
4 6 5 7 8
4 5 1 3 7
4 3 4 8 7
4 1 5 6 2

样例输出1

0.1666667
0.1666667
0.1666667
0.1666667
0.1666667
0.1666667

限制

首先存在20%的数据,骰子为长方体。
其次存在20%的数据,骰子为四面体。
余下的数据中有30%的数据,每一面都是三角形。
对于100%的数据,4<=n<=100且4<=m<=100,所有坐标的绝对值都在10000以内。

来源

SDOI 2016 round2 day2

  • 三维计算几何。
  • 需要混合积求四面体体积;
  • 四面体剖分后合并带权重心求总重心;
  • 四面体重心的横纵坐标是四个顶点的横纵坐标的平均数;
  • 三维差积求平面的法向量;
  • 点积求法向量夹角(二面角)
  • 这些知识就可以了AC此题了。
  • 时间复杂度O(nf)O(nf),注意n,f≤100n,f≤100,题面描述有误。
#include<cmath>
#include<cstdio>
using namespace std;
inline void read(int &x){
register char ch=getchar();x=;
while(ch<''||ch>'') ch=getchar();
while(ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
}
const int N=;
typedef double real;
const real pi=acos(-);
struct point{
real x,y,z;
point(){}
point(real _x,real _y,real _z):x(_x),y(_y),z(_z){}
point operator +(const point &a)const{
return point(x+a.x,y+a.y,z+a.z);
}
point operator -(const point &a)const{
return point(x-a.x,y-a.y,z-a.z);
}
point operator ^(const point &a)const{
return point(y*a.z-z*a.y,z*a.x-x*a.z,x*a.y-y*a.x);
}
real operator *(const point &a)const{
return x*a.x+y*a.y+z*a.z;
}
point operator /(const real &a)const{
return point(x/a,y/a,z/a);
}
const real len(){
return sqrt(x*x+y*y+z*z);
}
}P[N],H[N*N];
int n,m,Htot,f[N][N];
real val[N*N];
int main(){
read(n);read(m);
for(int i=;i<=n;i++) scanf("%lf%lf%lf",&P[i].x,&P[i].y,&P[i].z);
for(int i=;i<=m;i++){
read(f[i][]);
for(int j=;j<=f[i][];j++){
read(f[i][j]);
}
}
point u=P[];
for(int i=;i<=m;i++){
point u2=P[f[i][]],v1,v2;
for(int j=;j<f[i][];j++){
v1=P[f[i][j]];
v2=P[f[i][j+]];
H[++Htot]=(u+u2+v1+v2)/;//四面体重心
val[Htot]=fabs(((u2-v1)^(u2-v2))*(u-u2));//四面体体积
}
}
u=point(,,);
real valtot=;
for(int i=;i<=Htot;i++){
valtot+=val[i];
u=u+point(H[i].x*val[i],H[i].y*val[i],H[i].z*val[i]);
}
u=u/valtot;//球心坐标
for(int i=;i<=m;i++){
point u1,u2,u3;real co,ans=;
for(int j=,s1,s2;j<=f[i][];j++){//该平面拆成三角形,计算所有不同夹角
s1= j+;if(s1>f[i][]) s1=;
s2=s1+;if(s2>f[i][]) s2=;
u1=P[f[i][j]]-u;
u2=P[f[i][s1]]-u;
u3=P[f[i][s2]]-u;
u1=u1^u2;//面(u1,u2)的法向量
u3=u3^u2;//面(u2,u3)的法向量
co=u1*u3/u1.len()/u3.len();//二面角夹角
ans+=acos(co);
}
ans-=pi*(f[i][]-);
printf("%.7lf\n",ans//pi);
}
return ;
}

[Sdoi2016]平凡的骰子的更多相关文章

  1. [LOJ 2070] 「SDOI2016」平凡的骰子

    [LOJ 2070] 「SDOI2016」平凡的骰子 [题目链接] 链接 [题解] 原题求的是球面面积 可以理解为首先求多面体重心,然后算球面多边形的面积 求重心需要将多面体进行四面体剖分,从而计算出 ...

  2. 【LOJ】#2070. 「SDOI2016」平凡的骰子

    题解 用了一堆迷之复杂的结论结果迷之好写的计算几何???? 好吧,要写立体几何了 如果有名词不懂自己搜吧 首先我们求重心,我们可以求带权重心,也就是x坐标的话是所有分割的小四面体的x坐标 * 四面体体 ...

  3. LOJ#2070. 「SDOI2016」平凡的骰子(计算几何)

    题面 传送门 做一道题学一堆东西不管什么时候都是美好的体验呢-- 前置芝士 混合积 对于三个三维向量\(a,b,c\),定义它们的混合积为\((a\times b)\cdot c\),其中$\time ...

  4. 【Vijos 1998】【SDOI 2016】平凡的骰子

    https://vijos.org/p/1998 三维计算几何. 需要混合积求四面体体积: 四面体剖分后合并带权重心求总重心: 四面体重心的横纵坐标是四个顶点的横纵坐标的平均数: 三维差积求平面的法向 ...

  5. bzoj 4603 平凡的骰子

    题目大意: 思路: 首先我们需要求出整个凸多面体的重心 可以通过把多面体剖分为四面体 求出每个四面体的重心 四面体的重心为四个点的坐标和/4 对每个四面体的重心 加上它们体积的权 加权平均数即为整个的 ...

  6. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  7. BZOJ1298[SCOI2009]骰子的学问

    Description Input 第一行为两个整数n, m.第二行有n个整数,为a1,a2, -, an. Output 包含n行,每行m个1~n×m的正整数,各不相同,以空格分开.如果有多解,输出 ...

  8. Frida HOOK微信实现骰子作弊

    由于微信摇骰子的功能在本地进行随机后在发送,所以存在可以hook掉判断骰子数的方法进行修改作弊. 1.frida实现hook java层函数1)写个用来测试的demo,当我们点击按钮的时候会弹出窗口显 ...

  9. BZOJ4516: [Sdoi2016]生成魔咒 后缀自动机

    #include<iostream> #include<cstdio> #include<cstring> #include<queue> #inclu ...

随机推荐

  1. Linux中的软链接与硬链接

    软链接相当于windows的快捷方式,当源文件不存在时,软链接失效. 链接是指向文件名,当指向的文件名字删除的时候,就找不到源文件了.硬链接是指向文件本身,删除一个文件名字,还是可以找到源文件的.ls ...

  2. Logback中文文档(二):体系结构

    logback Logback 的基本结构充分通用,可应用于各种不同环境.目前,logback 分为三个模块:Core.Classic 和 Access. Core模块是其他两个模块的基础.Class ...

  3. pycharm破解补丁的使用

    转自 https://www.cnblogs.com/lhuser/p/8040163.html 闲来无事,想学学python的爬虫 http://idea.lanyus.com/  破解补丁下载 或 ...

  4. android中sharedPreferences的用法(转)

    SharedPreferences介绍:   做软件开发应该都知道,很多软件会有配置文件,里面存放这程序运行当中的各个属性值,由于其配置信息并不多,如果采用数据库来存放并不划算,因为数据库连接跟操作等 ...

  5. BearSkill纯代码搭建iOS界面

    欢迎相同喜欢动效的project师/UI设计师/产品增加我们 iOS动效特攻队–>QQ群:547897182 iOS动效特攻队–>熊熊:648070256 浅谈一下 关于iOS兼容布局一直 ...

  6. 03-Linux各目录及每个目录的详细介绍

    Linux各目录及每个目录的详细介绍 [常见目录说明] 目录 /bin 存放二进制可执行文件(ls,cat,mkdir等),常用命令一般都在这里. /etc 存放系统管理和配置文件 /home 存放所 ...

  7. flexbox父盒子align-content属性

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  8. 本地Chrome测试JS代码报错:XMLHttpRequest cannot load

    这种file跨域问题在火狐下是不存在的 解决Chrome下file跨域问题: 在Chrome应用程序下,右键属性,目标处添加"--allow-file-access-from-files&q ...

  9. Python中的yield和Generators(生成器)

    本文目的 解释yield关键字到底是什么,为什么它是有用的,以及如何来使用它. 协程与子例程 我们调用一个普通的Python函数时,一般是从函数的第一行代码开始执行,结束于return语句.异常或者函 ...

  10. junit的简单用法

    之前测试一个方法总要写一个main函数来调用,感觉既费事又有点low.今天来简单研究一下怎么使用junit来进行单元测试. 1. 依赖包 <dependency> <groupId& ...