Portal

  

  传送门

  

  

  

Solution

  

  显然的是,每一个圆的影子,就是从树上的圆按光线方向平移至地面的圆。至于两个圆之间的连接部分,则是每两个在树上相邻的圆的,对应的影子圆的,公切线围起来的部分,如下图所示

  



  

  所以我们现在要求每两个在原树上相邻的圆的影子圆构成的图形的并。只看\(x\)轴上半部分,可以把它想象成一个函数,求单点值是\(O(n)\)的,我们不妨用辛普森积分来解决......

  

  相邻圆的公切线和x轴的夹角是可以求出来的,然后就能解出公切线的解析式,以及有效范围。注意这些东西要预处理!千万不要放在求值函数里面。\(EPS\)大约设置到\(10^{-7}\)才不会出错,效率也相对比较高。

  

  

  

Code

  

#include <cstdio>
#include <cmath>
using namespace std;
const int N=505;
const double EPS=1e-7,INF=1e10,PI=3.14159265358979323846;
int n,sum;
double alpha,h[N],p[N],r[N];
double k[N],b[N],lx[N],rx[N];
bool exist[N];
inline double max(double x,double y){return x>y?x:y;}
inline double min(double x,double y){return x<y?x:y;}
inline void swap(int &x,int &y){x^=y^=x^=y;}
inline bool in(int a,int b){
if(r[a]>r[b]) swap(a,b);
return p[a]+r[a]-EPS<=p[b]+r[b]&&p[a]-r[a]+EPS>=p[b]-r[b];
}
double calc(int a,int b,double &k,double &bb,double &xl,double &xr){
if(p[a]>p[b]) swap(a,b);
double beta=asin((r[b]-r[a])/(p[b]-p[a]));
k=tan(beta);
double tx;
if(r[a]>=r[b]){
beta=-beta;
tx=p[b]+r[b]/sin(beta);
bb=-k*tx;
xr=tx-(cos(beta)*(r[b]/tan(beta)));
xl=tx-(cos(beta)*(r[a]/tan(beta)));
}
else{
tx=p[a]-r[a]/sin(beta);
bb=-k*tx;
xl=tx+(cos(beta)*(r[a]/tan(beta)));
xr=tx+(cos(beta)*(r[b]/tan(beta)));
}
}
double f(double x){
double res=0;
for(int i=1;i<=n;i++)
if(fabs(x-p[i])<=r[i])
res=max(res,sqrt(r[i]*r[i]-fabs(x-p[i])*fabs(x-p[i])));
for(int i=1;i<n;i++)
if(exist[i])
if(lx[i]<=x+EPS&&x-EPS<=rx[i])
res=max(res,k[i]*x+b[i]);
return res;
}
double simpson(double l,double r){
double mid=(l+r)*0.5;
return (f(l)+4*f(mid)+f(r))*(r-l)/6;
}
double solve(double l,double r){
double mid=(l+r)/2,lmid=(l+mid)/2,rmid=(mid+r)/2;
double val=simpson(l,r);
if(fabs(val-(simpson(l,mid)+simpson(mid,r)))<EPS) return val;
return solve(l,mid)+solve(mid,r);
}
int main(){
scanf("%d%lf",&n,&alpha);
n++;
for(int i=1;i<=n;i++){
scanf("%lf",h+i);
h[i]+=h[i-1];
p[i]=h[i]/tan(alpha);
}
for(int i=1;i<n;i++) scanf("%lf",r+i);
double xl=INF,xr=-INF;
for(int i=1;i<=n;i++){
xl=min(xl,p[i]-r[i]);
xr=max(xr,p[i]+r[i]);
}
for(int i=1;i<n;i++){
exist[i]=!in(i,i+1);
if(exist[i])
calc(i,i+1,k[i],b[i],lx[i],rx[i]);
}
printf("%.2lf\n",solve(xl,xr)*2);
return 0;
}

【BZOJ1502】【NOI2005】月下柠檬树的更多相关文章

  1. BZOJ1502: [NOI2005]月下柠檬树

    Simpson法相当好用啊!神奇的骗分算法! /************************************************************** Problem: 1502 ...

  2. [日常摸鱼]bzoj1502[NOI2005]月下柠檬树-简单几何+Simpson法

    关于自适应Simpson法的介绍可以去看我的另一篇blog http://www.lydsy.com/JudgeOnline/problem.php?id=1502 题意:空间里圆心在同一直线上且底面 ...

  3. 【BZOJ1502】[NOI2005]月下柠檬树 Simpson积分

    [BZOJ1502][NOI2005]月下柠檬树 Description 李哲非常非常喜欢柠檬树,特别是在静静的夜晚,当天空中有一弯明月温柔地照亮地面上的景物时,他必会悠闲地坐在他亲手植下的那棵柠檬树 ...

  4. BZOJ 1502: [NOI2005]月下柠檬树 [辛普森积分 解析几何 圆]

    1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1070  Solved: 596[Submit][Status] ...

  5. 【BZOJ-1502】月下柠檬树 计算几何 + 自适应Simpson积分

    1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1017  Solved: 562[Submit][Status] ...

  6. [NOI2005]月下柠檬树[计算几何(simpson)]

    1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1169  Solved: 626[Submit][Status] ...

  7. [NOI2005]月下柠檬树

    题意 F.A.Qs Home Discuss ProblemSet Status Ranklist Contest 入门OJ ModifyUser  autoint Logout 捐赠本站 Probl ...

  8. 5.21 省选模拟赛 luogu P4207 [NOI2005]月下柠檬树 解析几何 自适应辛普森积分法

    LINK:月下柠檬树 之前感觉这道题很鬼畜 实际上 也就想到辛普森积分后就很好做了. 辛普森积分法的式子不再赘述 网上多的是.值得一提的是 这道题利用辛普森积分法的话就是一个解析几何的问题 而并非计算 ...

  9. 【bzoj1502】[NOI2005]月下柠檬树 自适应Simpson积分

    题目描述 李哲非常非常喜欢柠檬树,特别是在静静的夜晚,当天空中有一弯明月温柔地照亮地面上的景物时,他必会悠闲地坐在他亲手植下的那棵柠檬树旁,独自思索着人生的哲理.李哲是一个喜爱思考的孩子,当他看到在月 ...

  10. BZOJ1502:[NOI2005]月下柠檬树——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=1502 https://www.luogu.org/problemnew/show/P4207 李哲 ...

随机推荐

  1. [Processing]点到线段的最小距离

    PVector p1,p2,n; float d = 0; void setup() { size(600,600); p1 = new PVector(150,30);//线段第一个端点 p2 = ...

  2. Linux命令的那些事(三)

    回顾linux命令那些事,前面大致总结了常用的Linux命令 回顾Linux命令那些事(一) clear/mkdir/rmdir/ls/rm/pwd/cd/touch/tree/man/--help ...

  3. phpcmsv9广告版位调用方法

    <div class="ya"> <?php // pc:get 使用sql语句获取指定条件的广告版位! ?> {pc:get sql="SELE ...

  4. SparkRDD编程实战

    通过spark实现点击流日志分析案例 1. 访问的pv package cn.itcast import org.apache.spark.rdd.RDD import org.apache.spar ...

  5. 工程能力之C4模型

    概述 刚在InfoQ上看到一篇介绍C4Model的文章,觉得这个模型设计的很赞,很有指导意义,做个简单的记录. Why,为什么需要架构图? ThoughtWorks中国 文章中有几句话我觉得很有道理, ...

  6. dvwa——命令注入&文件包含

    命令注入 commond_injection 源码.分析.payload: low: <?php if( isset( $_POST[ 'Submit' ] ) ) { // Get input ...

  7. Python之并发编程-concurrent

    方法介绍 #1 介绍 concurrent.futures模块提供了高度封装的异步调用接口 ThreadPoolExecutor:线程池,提供异步调用 ProcessPoolExecutor: 进程池 ...

  8. Daily Scrum (2015/10/23)

    这天晚上PM和我一起细算下来这周的确做了不少事儿.由于这天是周五,有的组员今晚有外出活动,有的组员忙了一周想休息一下.所以PM与我讨论提出今晚就布置些阅读的任务,给组员们一些自由分配的时间: 成员 今 ...

  9. 《Spring 2之站立会议3》

    <Spring 2之站立会议3> 昨天,查找了本机的端口号,并对代码作进一步的了解. 今天,对我们项目的基本框架进行了了解,即主界面和各个分界面的基本架构: 遇到的问题,虽然了解了基本框架 ...

  10. beta冲刺(7/7)

    目录 组员情况 组员1:胡绪佩 组员2:胡青元 组员3:庄卉 组员4:家灿 组员5:恺琳 组员6:翟丹丹 组员7:何家伟 组员8:政演 组员9:黄鸿杰 组员10:何宇恒 组员11:刘一好 展示组内最新 ...