一堆圆台平行光的投影

在草稿纸上画一下,发现对于一个圆,它投影完还是一个半径不变的圆。

定义树的轴在投影平面上经过的点为原点,定一个正方向,建立平面直角坐标系,

能发现,对于一个半径为\(r\),高度为\(h\)的圆,投影到平面上是圆心坐标为\((cot(\alpha)h, 0)\),半径为\(r\)的圆

想象有一个水平的平面,竖直向上移,可以把树切出一堆圆,对于这些圆,把它们投影求个并就是答案

对于每个圆台,它一堆圆的并就是先求上下两个面的圆的投影,再对投影求外公切线,围成的图形

如图,就是\(BEHDGF\)围成的面积

注意对于两个圆的内含或内切关系,是没有切线的

对于树顶,我们把它当做一个半径为\(0\)的圆

那么大概可以画成这样

首先因为它是轴对称的,所以只用算出x轴上方的

但是这面积并怎么求呢?求出所有交点?

这个图挺特殊,所以可以对不规则的函数下方面积考虑使用自适应simpson

然后就做完了

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath> const double eps = 1e-6;
const double STOP = 15;
const int MAXN = 510;
double xs[MAXN], rs[MAXN], theta;
double tx1[MAXN], ty1[MAXN], tx2[MAXN], ty2[MAXN];
int bak;
inline double absx(double x) { return x < 0 ? -x : x; }
inline double sqrx(double x) { return x * x; }
int n;
double f(double at) {
double res = 0;
for (int i = 1; i <= n; ++i) {
double t = absx(xs[i] - at);
if (t < rs[i])
res = std::max(res, sqrt(sqrx(rs[i]) - sqrx(t)));
}
for (int i = 1; i <= bak; ++i) {
if (tx1[i] - eps <= at && at <= tx2[i] + eps) {
double tanx = (ty2[i] - ty1[i]) / (tx2[i] - tx1[i]);
res = std::max(res, ty1[i] + tanx * (at - tx1[i]));
}
}
return res;
}
inline double calc(double l, double mid, double r) {
return (l + r + mid * 4) / 6;
}
double simpson(double l, double r, double eps, double ll, double midv, double lr) {
// if (absx(r - l) < 0.1) return 213213;
const double lans = calc(ll, midv, lr) * (r - l);
const double mid = (l + r) / 2;
const double lmid = (l + mid) / 2, rmid = (mid + r) / 2;
const double lmidv = f(lmid), rmidv = f(rmid);
const double lv = calc(ll, lmidv, midv) * (mid - l);
const double rv = calc(midv, rmidv, lr) * (r - mid);
// std::cout << "SIMP " << l << " " << r << " " << eps << " " << lans << " " << lv << " " << rv << std::endl;
if (absx(lv + rv - lans) <= eps * STOP) return lans;
return simpson(l, mid, eps / 2, ll, lmidv, midv) +
simpson(mid, r, eps / 2, midv, rmidv, lr);
}
int main() {
double L = 1e10, R = -1e10;
scanf("%d%lf", &n, &theta); ++n;
const double transform = 1 / tanl(theta);
double now = 0, t;
for (int i = 1; i <= n; ++i) {
scanf("%lf", &t);
now += t;
xs[i] = now * transform;
}
for (int i = 1; i != n; ++i) scanf("%lf", rs + i);
for (int i = 1; i <= n; ++i) {
L = std::min(L, xs[i] - rs[i]);
R = std::max(R, xs[i] + rs[i]);
}
rs[n] = 0;
for (int i = 1; i != n; ++i) {
if (absx(xs[i] - xs[i + 1]) < absx(rs[i + 1] - rs[i]) + eps) continue;
const double sina = (rs[i + 1] - rs[i]) / (xs[i + 1] - xs[i]);
if (1 - sqrx(sina) < eps) continue;
const double cosa = sqrt(1 - sqrx(sina));
++bak;
tx1[bak] = xs[i] - sina * rs[i];
ty1[bak] = cosa * rs[i];
tx2[bak] = xs[i + 1] - sina * rs[i + 1];
ty2[bak] = cosa * rs[i + 1];
}
printf("%.2lf\n", simpson(L, R, eps, f(L), f((L + R) / 2), f(R)) * 2);
return 0;
}

1502: [NOI2005]月下柠檬树的更多相关文章

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

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

  2. 1502: [NOI2005]月下柠檬树 - BZOJ

    Description Input 文件的第1行包含一个整数n和一个实数alpha,表示柠檬树的层数和月亮的光线与地面夹角(单位为弧度).第2行包含n+1个实数h0,h1,h2,…,hn,表示树离地的 ...

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

    题目链接: BZOJ - 1502 题目分析 这是我做的第一道 Simpson 积分的题目.Simpson 积分是一种用 (fl + 4*fmid + fr) / 6 * (r - l) 来拟合 fl ...

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

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

  5. [NOI2005]月下柠檬树

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

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

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

  7. 【bzoj 1502】月下柠檬树

    月下柠檬树 题意 求n个圆与他们的公切线的定积分. 解法 求出圆的公切线就可以了. 特别坑的一点 : 最两端的圆,有可能会被其他的圆所包含,所以要重新求一下最左端与最右端. 比较坑的一点 : 精度要设 ...

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

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

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

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

随机推荐

  1. SVN随笔记录(一)

    svn是版本控制系统 为何使用svn? ~团队在开发同一个项目时对项目进行模块划分,在第一阶段结束后进行部分整合时,提交至服务器上合并. ~多人对同一版本的同一代码进行修改后,合并时会出现冲突,此是需 ...

  2. Django中ORM操作提升性能

    提升orm操作性能注意的点 优化一:尽量不查对象,能用values就是用values 直接使用对象查询的结果是5条sql语句 def youhua(request): # 使用对象查 obj_list ...

  3. java限流工具类

    代码 import com.google.common.util.concurrent.RateLimiter; import java.util.concurrent.ConcurrentHashM ...

  4. WordPress网站搬家数据迁移完整教程

    用本地环境搭建好的WordPress网站在做好之后如何从本地迁移到网络空间或者网络服务器上呢? 首先请确认你在本地建站的时候只做了themes里面的模版文件,如果只是自己改了下模版,那么网站在搬到服务 ...

  5. python使用Flask作为MockServer的方法

    日常开发/测试过程中,需要对相关服务添加挡板--Mock 简单介绍一下使用python的Flask插件,对相关的服务进行Mock # coding:utf-8 import os from flask ...

  6. leecode刷题(28)-- 二叉树的前序遍历

    leecode刷题(28)-- 二叉树的前序遍历 二叉树的前序遍历 给定一个二叉树,返回它的 前序 遍历. 示例: 输入: [1,null,2,3] 1 \ 2 / 3 输出: [1,2,3] 思路 ...

  7. 测试用例管理工具-TestLink

    TestLink是基于web的测试用例管理系统,主要功能是测试用例的创建.管理和执行,并且还提供了一些简单的统计功能,主要功能包括: 测试需求管理 测试用例管理 测试用例对测试需求的覆盖管理 测试计划 ...

  8. pl_sql例题

    0. 准备工作: set serveroutput on hellowrold 程序 begindbms_output.put_line('hello world');end;/ [语法格式]--de ...

  9. DHCP服务部署

    目录 DHCP服务部署 一. 简介 二. 用途及功能 三. 原理+示意图 四. 实战搭建 相关文件 配置基础DHCP服务器 配置DHCP保留地址 配置DHCP超级作用域 配置DHCP中继 五. 小结 ...

  10. 最完美ThinkPHP Nginx 配置文件

    一个配置文件,完美支持普通,兼容,pathinfo,rewrite4种url模式,别怪我没提醒你收藏哦. 常见的静态文件404时也不会再去跑一遍fastcgi浪费资源. server { listen ...