【bzoj1502】月下柠檬树
Solution
额其实说实在这题我一开始卡在了。。这个阴影长啥样上QwQ
首先因为是平行光线然后投影到了一个水平面上所以这个投影一定是。。若干个圆再加上这些圆的边界这样组成的
额或者说是这样:

就是一些圆和相邻圆的切线组成的边界
然后因为这个是。。一段连续的曲线嘛然后的话浮动应该不会太太太太大所以有一种相对简单的方法就是用Simpson积分直接爆搞(虽然说。。如果你特意构造一下数据的话还是可能会被卡掉的qwq)
注意到这个图形是对称的所以我们可以先只看一半,然后最后输出的时候乘\(2\)就好了
然后这个边界的话我们可以将它看成一个函数\(f\),如果要用Simpson积分的话我们需要快速求这个函数在某个点的取值,我们记当前查询\(f(x)\),那么这个可以分成两种情况:
1、点\((x,f(x))\)在某个圆上,这种时候\(f(x)=\sqrt {r^2-(x-x_O)^2}\),其中\(x_O\)是这个圆的圆心的横坐标,\(r\)是这个圆的半径
2、点\((x,f(x))\)在某条切线上,那这个时候直接把\(x\)带进这条切线的解析式里面就可以得到\(f(x)\)了
然而实际上在实现的时候,并不需要特别分类讨论,只要返回两种算的方法的最大值就好了
那么剩下来的问题,就是如何求两个圆的切线
首先我们需要预处理一下,把被包含的圆去掉
然后接下来我们只讨论不包含的情况

还是用回这张图,我们只看\(x\)轴的上半部分
其中\(A,B\)均为切点,记\(l_{AB}\)与\(x\)轴的夹角为\(\theta\),\(L=O_1O_2\),第一个圆的半径为\(r\),第二个圆的半径为\(R\)
由相似我们可以得到\(sin\theta=\frac{R-r}{L}\),进而我们可以得到\(cos\theta\)
然后知道了\(sin\theta\)和\(cos\theta\)之后,我们就可以知道\(A\)和\(B\)的坐标了:
A&=(x_{O_1}-sin\theta\cdot r,cos\theta \cdot r)\\
B&=(x_{O_2}-sin\theta\cdot R,cos\theta\cdot R)\\
\end{aligned}
\]
然后我们就可以直接求一下\(l_{AB}\)的解析式了,这里不需要担心\(l_{AB}\)的斜率是否存在,因为这个图形保证了不会出现这种情况
然后就很愉快滴做完啦ovo
代码大概长这个样子(额貌似eps要开到1e-6才不会出锅qwq):
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
const int N=510;
const double eps=1e-6,inf=2147483647;
double h[N],r[N],O[N],k[N],b[N];
double lx[N],rx[N],ly[N],ry[N];
int have[N];
int n,m;
double alpha,L,R;
double val(double x);
double simpson(double l,double r);
void prework();
void calc(int x);
bool in(int x,int y){return fabs(O[x]-O[y])<=fabs(r[x]-r[y]);}
double sqr(double x){return x*x;}
double solve(double l,double r);
int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
scanf("%d%lf\n",&n,&alpha);
++n;
for (int i=1;i<=n;++i){
scanf("%lf",h+i);
O[i]=h[i]/tan(alpha)+O[i-1];
}
L=inf,R=-inf;
for (int i=1;i<n;++i) scanf("%lf",r+i);
for (int i=1;i<=n;++i)
L=min(O[i]-r[i],L),R=max(O[i]+r[i],R);
prework();
printf("%.2lf\n",solve(L,R)*2.0);
}
double solve(double l,double r){
double mid=(l+r)*0.5,h=simpson(l,r),hmid=simpson(l,mid)+simpson(mid,r);
if (fabs(r-l)<=eps||(fabs(h-hmid)<=eps)) return hmid;
return solve(l,mid)+solve(mid,r);
}
double simpson(double l,double r){
double hl=val(l),hr=val(r),hmid=val((l+r)*0.5);
return (r-l)*(hl+hr+hmid*4.0)/6.0;
}
double val(double x){
double ret=0;
for (int i=1;i<=n;++i)
if (fabs(x-O[i])<=r[i])
ret=max(ret,sqrt(sqr(r[i])-sqr(x-O[i])));
for (int i=1;i<n;++i){
if (have[i]) continue;
if (lx[i]<=x&&x<=rx[i])
ret=max(ret,k[i]*x+b[i]);
}
return ret;
}
void prework(){
for (int i=1;i<n;++i){
have[i]=in(i,i+1);
if (have[i]) continue;
calc(i);
}
}
void calc(int x){
double l=(O[x+1]-O[x]);
double theta=asin((r[x+1]-r[x])/l);
double x1,y1,x2,y2;
x1=O[x]-sin(theta)*r[x]; y1=cos(theta)*r[x];
x2=O[x+1]-sin(theta)*r[x+1]; y2=cos(theta)*r[x+1];
lx[x]=x1; ly[x]=y1;
rx[x]=x2; ry[x]=y2;
k[x]=(y2-y1)/(x2-x1);
b[x]=y1-k[x]*x1;
}
【bzoj1502】月下柠檬树的更多相关文章
- [BZOJ1502]月下柠檬树(自适应辛普森积分)
1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1387 Solved: 739[Submit][Status] ...
- 【BZOJ1502】[NOI2005]月下柠檬树 Simpson积分
[BZOJ1502][NOI2005]月下柠檬树 Description 李哲非常非常喜欢柠檬树,特别是在静静的夜晚,当天空中有一弯明月温柔地照亮地面上的景物时,他必会悠闲地坐在他亲手植下的那棵柠檬树 ...
- 【BZOJ-1502】月下柠檬树 计算几何 + 自适应Simpson积分
1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1017 Solved: 562[Submit][Status] ...
- BZOJ 1502: [NOI2005]月下柠檬树 [辛普森积分 解析几何 圆]
1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1070 Solved: 596[Submit][Status] ...
- [NOI2005]月下柠檬树[计算几何(simpson)]
1502: [NOI2005]月下柠檬树 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1169 Solved: 626[Submit][Status] ...
- 【bzoj 1502】月下柠檬树
月下柠檬树 题意 求n个圆与他们的公切线的定积分. 解法 求出圆的公切线就可以了. 特别坑的一点 : 最两端的圆,有可能会被其他的圆所包含,所以要重新求一下最左端与最右端. 比较坑的一点 : 精度要设 ...
- [NOI2005]月下柠檬树
题意 F.A.Qs Home Discuss ProblemSet Status Ranklist Contest 入门OJ ModifyUser autoint Logout 捐赠本站 Probl ...
- 5.21 省选模拟赛 luogu P4207 [NOI2005]月下柠檬树 解析几何 自适应辛普森积分法
LINK:月下柠檬树 之前感觉这道题很鬼畜 实际上 也就想到辛普森积分后就很好做了. 辛普森积分法的式子不再赘述 网上多的是.值得一提的是 这道题利用辛普森积分法的话就是一个解析几何的问题 而并非计算 ...
- 【bzoj1502】[NOI2005]月下柠檬树 自适应Simpson积分
题目描述 李哲非常非常喜欢柠檬树,特别是在静静的夜晚,当天空中有一弯明月温柔地照亮地面上的景物时,他必会悠闲地坐在他亲手植下的那棵柠檬树旁,独自思索着人生的哲理.李哲是一个喜爱思考的孩子,当他看到在月 ...
- BZOJ1502:[NOI2005]月下柠檬树——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=1502 https://www.luogu.org/problemnew/show/P4207 李哲 ...
随机推荐
- Vue学习计划基础笔记(二) - 模板语法,计算属性,侦听器
模板语法.计算属性和侦听器 目标: 1.熟练使用vue的模板语法 2.理解计算属性与侦听器的用法以及应用场景 1. 模板语法 <div id="app"> <!- ...
- Echarts-K线图提示框改头换面
工作: 使用Hbuilder建web工程,加入echarts相关库,根据需要更改K线图及其的提示样式,去除默认提示,使用异步加载echarts的数据,数据格式为json. 需要注意的K线图和5日均线, ...
- 性能测试持续集成(Jenkins+Ant+Jmeter)
一.环境准备: 1.JDK:http://www.oracle.com/technetwork/java/javase/downloads/index.html 2.Jmeter:http://jme ...
- 亚马逊拟斥资15亿美元建航空货运中心 - Amazon to spend $1.49 bln on air cargo hub, fans talk of bigger ambitions - ReutersFebruary 1, 2017
2月1日消息,亚马逊本周二宣布将在肯塔基州开建其第一个航空货运中心,以应对高速增长的航空货运需求.亚马逊预计,该项目将带来2000个工作岗位. 据悉,该项计划总投入约为15亿美元,亚马逊或可从当地政府 ...
- Oracle 11g用exp无法导出空表的处理方法
Oracle 11G在用EXPORT导出时,空表不能导出. 11G中有个新特性,当表无数据时,不分配segment,以节省空间 解决方法: 1.insert一行,再rollback就产生segment ...
- C++:构造函数1——普通构造函数
前言:构造函数是C+中很重要的一个概念,这里对其知识进行一个简单的总结 一.构造函数的定义 1.类中的构造函数名与类名必须相同 2.构造函数没有函数的返回类值型说明符 [特别注意]: a.构造函数的返 ...
- C语言调查问卷
1.你对自己的未来有什么规划?做了哪些准备?毕业后应该不会从事编程类工作,目前有在学习感兴趣的东西.2.你认为什么是学习?学习有什么用?现在学习动力如何?为什么?学习就是把不懂变成懂,可以充实自己.没 ...
- lintcode-463-整数排序
463-整数排序 给一组整数,按照升序排序,使用选择排序,冒泡排序,插入排序或者任何 O(n2) 的排序算法. 样例 对于数组 [3, 2, 1, 4, 5], 排序后为:[1, 2, 3, 4, 5 ...
- .net 错误处理
第一步在页面中写OnError方法: protected override void OnError(EventArgs e) { Exception ex = Server.GetLastError ...
- 还原 listagg/wm_concat 后的数据 pack_split_listatt ;
1.创建表并制作测试数据: --创建测试表 : CREATE TABLE split_table ( NAME ), ID ) ); --准备测试数据 : INSERT INTO split_tabl ...