出题人题解 | #球的体积并#

球的体积并

https://ac.nowcoder.com/acm/problem/22564

原题解链接:https://ac.nowcoder.com/discuss/157310

设两个球 A\ A B \ B的体积分别为 Va\ V_{a} Vb\ V_{b} ,半径分别为  ra\ r_{a}  rb\ r_{b} ,两球球心距离为 d\ d

分三种情况讨论: 两球/相离/外切,那么答案是  Va+Vb\ V_{a}+V_{b}

两球内含/内切时,答案是 max(Va,Vb)\ max(V_{a} , V_{b})

两球相交时,我们可以认为相交部分就是两个球被平面所截的部分,我们称之为球冠。

对于球冠体积有公式:V=πh(3a2+h2)6=πh2(3rh)3V=\frac{\pi h\left(3 a^{2}+h^{2}\right)}{6}=\frac{\pi h^{2}(3 r-h)}{3}

其中 A\ A为球半径, a\ a为截面圆半径,  h\ h为垂直于截面的一条直径,即球冠的高 ;

那么  A\ A对应的球冠参数为: r=ra,h=rara2rb2+d22dr=r_{a}, h=r a-\frac{r a^{2}-r b^{2}+d^{2}}{2 d}

BB对应的球冠参数为:r=rb,h=rbrb2ra2+d22dr=r b, h=r b-\frac{r b^{2}-r a^{2}+d^{2}}{2 d}

代入球冠体积公式便可得到答案。

#include<bits/stdc++.h>
using namespace std;
const double PI = acos(-1.0);
typedef struct point
{
    double x,y,z;
    point() { } point(double a, double b,double c)
    {
        x = a;
        y = b;
        z = c;
    } point operator -(const point &b)const    //返回减去后的新点
    {
        return point(x - b.x, y - b.y,z-b.z);
    } point operator +(const point &b)const   //返回加上后的新点
    {
        return point(x + b.x, y + b.y,z+b.z);
    } //数乘计算
    point operator *(const double &k)const   //返回相乘后的新点
    {
        return point(x * k, y * k,z*k);
    }
    point operator /(const double &k)const   //返回相除后的新点
    {
        return point(x / k, y / k,z/k);
    }
    double operator *(const point &b)const   //点乘
    {
        return x*b.x + y*b.y+z*b.z;
    }
} point;
double dist(point p1, point p2)   //返回平面上两点距离
{
    return sqrt((p1 - p2)*(p1 - p2));
}
typedef struct sphere  //球
{
    double r;
    point centre;
} sphere;
double SphereInterVS(sphere a, sphere b,double v,double s)
{
    double d = dist(a.centre, b.centre);//球心距
    if(d-a.r-b.r>0)return PI*4.0/3*a.r*a.r*a.r+PI*4.0/3*b.r*b.r*b.r;
    double r1=a.r,r2=b.r;
    if(d+r1<r2||d+r2<r1)return max(PI*4.0/3*a.r*a.r*a.r,PI*4.0/3*b.r*b.r*b.r);
    double t = (d*d + a.r*a.r - b.r*b.r) / (2.0 * d);//
    double h = sqrt((a.r*a.r) - (t*t)) * 2;//h1=h2,球冠的高
    double angle_a = 2 * acos((a.r*a.r + d*d - b.r*b.r) / (2.0 * a.r*d)); //余弦公式计算r1对应圆心角,弧度
    double angle_b = 2 * acos((b.r*b.r + d*d - a.r*a.r) / (2.0 * b.r*d)); //余弦公式计算r2对应圆心角,弧度
    double l1 = ((a.r*a.r - b.r*b.r) / d + d) / 2;
    double l2 = d - l1;
    double x1 = a.r - l1, x2 = b.r - l2;//分别为两个球缺的高度
    double v1 = PI*x1*x1*(a.r - x1 / 3);//相交部分r1圆所对应的球缺部分体积
    double v2 = PI*x2*x2*(b.r - x2 / 3);//相交部分r2圆所对应的球缺部分体积
    v = v1 + v2;//相交部分体积
    double s1 = PI*a.r*x1; //r1对应球冠表面积
    double s2 = PI*a.r*x2; //r2对应球冠表面积
    s = 4 * PI*(a.r*a.r + b.r*b.r) - s1 - s2;//剩余部分表面积
    return PI*4.0/3*a.r*a.r*a.r+PI*4.0/3*b.r*b.r*b.r-v;
}
int main()
{
    sphere a,b;
    cin>>a.centre.x>>a.centre.y>>a.centre.z>>a.r;
    cin>>b.centre.x>>b.centre.y>>b.centre.z>>b.r;
    printf("%.7f",SphereInterVS(a,b,0.0,0.0));
}


全部评论

相关推荐

03-06 18:20
门头沟学院 Java
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
正在热议
更多
# 春招至今,你的战绩如何? #
11505次浏览 99人参与
# 你的实习产出是真实的还是包装的? #
2022次浏览 43人参与
# 巨人网络春招 #
11387次浏览 223人参与
# 军工所铁饭碗 vs 互联网高薪资,你会选谁 #
7702次浏览 43人参与
# 简历第一个项目做什么 #
31809次浏览 344人参与
# 重来一次,我还会选择这个专业吗 #
433639次浏览 3926人参与
# MiniMax求职进展汇总 #
24236次浏览 310人参与
# 当下环境,你会继续卷互联网,还是看其他行业机会 #
187273次浏览 1122人参与
# 牛客AI文生图 #
21459次浏览 238人参与
# 不考虑薪资和职业,你最想做什么工作呢? #
152519次浏览 888人参与
# 研究所笔面经互助 #
118985次浏览 577人参与
# 简历中的项目经历要怎么写? #
310489次浏览 4226人参与
# AI时代,哪些岗位最容易被淘汰 #
64003次浏览 834人参与
# 面试紧张时你会有什么表现? #
30527次浏览 188人参与
# 你今年的平均薪资是多少? #
213204次浏览 1039人参与
# 你怎么看待AI面试 #
180271次浏览 1263人参与
# 高学历就一定能找到好工作吗? #
64348次浏览 620人参与
# 你最满意的offer薪资是哪家公司? #
76614次浏览 374人参与
# 我的求职精神状态 #
448221次浏览 3129人参与
# 正在春招的你,也参与了去年秋招吗? #
363624次浏览 2638人参与
# 腾讯音乐求职进展汇总 #
160712次浏览 1112人参与
# 校招笔试 #
471516次浏览 2964人参与
牛客网
牛客网在线编程
牛客网题解
牛客企业服务