[NOI2005][BZOJ1502][洛谷P4207]月下柠檬树(自适应Simpson积分)

题面

https://www.luogu.com.cn/problem/P4207

题解

前置知识

首先需要一些空间想象力~

1、圆形:一个空中的圆形, 其影子应该也是一个圆形。并且,其圆心距离树根的距离应该是原来高度的\(\cot \alpha\)倍。

2、圆台侧面:在画好圆台的上、下底面的影子以后,只需加上上、下底面的外公切线之间的部分。当然,如果上下底面内切甚至内含,圆台侧面的影子就整体被较大的那个圆覆盖了,此时没必要加任何东西。

所以,最终我们要计算面积的形状就是由所有的圆的影子,以及相邻两圆之间(如果外离)的两条外公切线组成的图形。

接下来只要使用自适应Simpson积分。实现时,可以只统计此图形的x轴以上的一半,最后再统一乘2。

Simpson中需要积分的函数f(x)即为横坐标x对应的高度。直接存下所有的半圆和外公切线段,然后一一判断其是否覆盖横坐标x;覆盖的话计算高度即可。

由于题目只需要1e-2的精度,本题eps我使用1e-7已经足够。

代码

#include<bits/stdc++.h>

using namespace std;

#define N 500
#define rg register
#define eps 1e-9
#define inf 1e9
#define In inline

In int sgn(double x){return x < -eps ? -1 : x > eps;}

In double sqr(double x){return x * x;}

struct vec{
	double x,y;
	vec(){}
	vec(double _x,double _y){x = _x,y = _y;}
	In friend vec operator + (vec a,vec b){
		return vec(a.x + b.x,a.y + b.y);
	}
	In friend vec operator - (vec a,vec b){
		return vec(a.x - b.x,a.y - b.y);
	}
	In friend double Dot(vec a,vec b){
		return a.x * b.x + a.y * b.y;
	}
	In friend double Cross(vec a,vec b){
		return a.x * b.y - a.y * b.x;
	}
};

struct line{
	vec p,q;
	line(){}
	line(vec _p,vec _q){p = _p,q = _q;}
	In friend double Height(line a,double x){ //计算线段a在横坐标为x时的纵坐标值
		if(sgn(x-a.p.x) < 0 || sgn(x-a.q.x) > 0)return 0;
		return (a.q.y * (x-a.p.x) + a.p.y * (a.q.x-x)) / (a.q.x - a.p.x);
	}
};

line l[N+5];
int ln;

struct cir{
	double c,r;
	cir(){}
	cir(double _c,double _r){c = _c,r = _r;}
	In friend double Height(cir a,double x){ //计算半圆a在横坐标为x时的纵坐标值
		if(sgn(fabs(x-a.c)-a.r) >= 0)return 0;
		return sqrt(sqr(a.r) - sqr(x-a.c)); 
	}
	In friend void CalcTan(cir a,cir b){ //计算半圆a,b的外公切线
		if(sgn(fabs(a.c-b.c)-fabs(a.r-b.r)) <= 0)return;
		if(sgn(a.c-b.c) > 0)swap(a,b);
		if(sgn(a.r-b.r) == 0){
			l[++ln] = line(vec(a.c,a.c+a.r),vec(b.c,b.c+b.r));
			return;
		}
		double cs = ((a.r-b.r) / (b.c-a.c));
		l[++ln] = line(vec(a.c+a.r*cs,a.r*sqrt(1-sqr(cs))),vec(b.c+b.r*cs,b.r*sqrt(1-sqr(cs))));
	}
};

cir c[N+5];
int n;

double f(double x){
	double ans = 0;
	for(rg int i = 1;i <= ln;i++)ans = max(ans,Height(l[i],x));
	for(rg int i = 1;i <= n;i++)ans = max(ans,Height(c[i],x));
	return ans;
}

double Simp(double l,double r){
	return (f(l) + 4 * f((l+r)/2) + f(r)) * (r - l) / 6;
}

double RSimp(double l,double r,double A,double Eps){
	double m = (l + r) / 2;
	double L = Simp(l,m),R = Simp(m,r);
	if(fabs(L+R-A) <= Eps)return L + R + (L + R - A) / 15;
	return RSimp(l,m,L,Eps) + RSimp(m,r,R,Eps);
}

double alpha;

int main(){
	scanf("%d%lf",&n,&alpha);
	double x = 0,h;
	for(rg int i = 1;i <= n + 1;i++){
		scanf("%lf",&h);
		x += h / tan(alpha);
		c[i].c = x;	
	}
	for(rg int i = 1;i <= n;i++)scanf("%lf",&c[i].r);
	c[n+1].r = 0;
	for(rg int i = 1;i <= n;i++)CalcTan(c[i],c[i+1]);
	double L = inf,R = 0;
	for(rg int i = 1;i <= n + 1;i++){
		L = min(L,c[i].c - c[i].r);
		R = max(R,c[i].c + c[i].r);
	}	
	printf("%.2lf\n",2 * RSimp(L,R,Simp(L,R),1e-7));
	return 0;
}
posted @ 2020-10-03 22:41  coder66  阅读(119)  评论(0编辑  收藏  举报