BZOJ 1859 Luogu P2589 [ZJOI2006]碗的叠放 (计算几何)
woc, 13年前的ZJOI就这么毒瘤的嘛。。。
题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=1859
(luogu)https://www.luogu.org/problemnew/show/P2589
题解: 大分类讨论,预处理\(ints[i][j]\)表示碗\(i\)叠在\(j\)上,\(i\)的底部比\(j\)的底部要高多少
然后分类讨论求此数组:
(1) \(i\)的下半径比\(j\)的上半径大,则等于\(j\)的高度
(2) \(i\)的上半径比\(j\)的下半径小,则等于零
(3) \(i\)的碗壁斜率大于\(j\)的碗壁斜率
(3.1) 若\(i\)的下半径小于等于\(j\)的下半径,则为零
(3.2) 若\(i\)的下半径大于\(j\)的下半径,则接触点横坐标为\(i\)的下半径,通过\(j\)的一次函数计算高度即可
(4) \(i\)的碗壁斜率小于\(j\)的碗壁斜率
(4.1) 若\(i\)的上半径大于等于\(j\)的上半径,则接触点为\(j\)的口处,通过\(i\)的一次函数计算高度
(4.2) 若\(i\)的上半径小于\(j\)的上半径,则(不妨先不考虑碗底)接触点横坐标为\(i\)的上半径,通过\(j\)的一次函数计算高度
情况(4)最后求出来的高度和\(0\)取Max.
(5) 若\(i\)和\(j\)碗壁斜率相等
(5.1) 若\(i\)的下半径小于等于\(j\)的下半径,则高度为\(0\).
(5.2) 若\(i\)的下半径大于\(j\)的上半径,则接触点横坐标为\(i\)的下半径,通过一次函数计算高度即可。
然后一个显然的性质是一个碗叠在一摞碗上,它的叠放高度为和每个碗分别叠放的高度的Max.
时间复杂度\(O(n!\times n)\)
这题……我还能说什么呢23333
代码
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cassert>
#include<algorithm>
using namespace std;
const int N = 9;
struct Element
{
int h,r1,r2;
double calc(double x) {return h*(x-(double)r1)/((double)r2-r1);}
} a[N+3];
double hh[N+3];
int permu[N+3];
double ints[N+3][N+3];
int n;
double intersect(int x,int y)
{
// printf("Intersect (%d,%d)\n",x,y);
double ret;
if(a[x].r1>a[y].r2) {ret = a[y].h;/* printf("Case 1\n");*/}
else if(a[x].r2<a[y].r1) {ret = 0;/* printf("Case 2\n");*/}
else if((a[y].r2-a[y].r1)*a[x].h>(a[x].r2-a[x].r1)*a[y].h)
{
if(a[x].r1<=a[y].r1) {ret = 0;/* printf("Case 3.1\n");*/}
else {ret = a[y].calc((double)a[x].r1);/* printf("Case 3.2\n");*/}
}
else if((a[y].r2-a[y].r1)*a[x].h<(a[x].r2-a[x].r1)*a[y].h)
{
if(a[x].r2>=a[y].r2) {ret = a[y].h-a[x].calc(a[y].r2);/* printf("Case 4.1\n");*/}
// else if(a[x].h*(a[y].r2-a[y].r1)<=a[y].h*(a[x].r2-a[y].r1)) {ret = 0;}
else {ret = a[y].calc(a[x].r2)-a[x].calc(a[x].r2);/* printf("Case 4.2\n");*/}
if(ret<0) ret = 0;
}
else
{
if(a[x].r1<=a[y].r1) {ret = 0;/* printf("Case 5.1\n");*/}
else {ret = a[y].calc(a[x].r1);/* printf("Case 5.2\n");*/}
}
// printf("ret=%lf\n",ret);
return ret;
}
double calc()
{
// printf("calc: "); for(int i=1; i<=n; i++) printf("%d ",permu[i]); puts("");
double ret = 0.0;
for(int i=1; i<=n; i++)
{
hh[i] = 0;
for(int j=1; j<i; j++)
{
hh[i] = max(hh[i],hh[j]+ints[permu[i]][permu[j]]);
}
ret = max(ret,hh[i]+a[permu[i]].h);
}
// printf("Total=%lf\n",ret); puts("");
return ret;
}
int main()
{
scanf("%d",&n);
for(int i=1; i<=n; i++) scanf("%d%d%d",&a[i].h,&a[i].r1,&a[i].r2);
for(int i=1; i<=n; i++) permu[i] = i;
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
if(i==j) continue;
ints[i][j] = intersect(i,j);
}
}
double ans = 1e6;
do
{
ans = min(ans,calc());
} while(next_permutation(permu+1,permu+n+1));
printf("%d\n",(int)(ans+0.5));
return 0;
}
BZOJ 1859 Luogu P2589 [ZJOI2006]碗的叠放 (计算几何)的更多相关文章
- [ZJOI2006]碗的叠放
Description 小H有n个碗需要放进橱柜,她希望将他们叠起来放置.你知道每个碗都是规则的圆柱体,并且都是上宽下窄,你已经测量出了每个碗的两个半径及高,请你帮小H找出一种叠放顺序,使得叠放出来的 ...
- 【BZOJ1859】【ZJOI2006】碗的叠放
题目大意:给你n个碗,求如何堆叠,使得它们的总高度最低. 首先,我们枚举碗的叠放顺序. 假设我们已经堆好了前i个碗,那么在堆第i+1个碗时,我们要将第i+1个碗与前i个碗比较,确定第i+1个碗的离地高 ...
- [Luogu 2596] ZJOI2006 书架
[Luogu 2596] ZJOI2006 书架 第一次指针写 FHQ_Treap(省选噩梦数据结构)AC 啦! 省选试机写它,紧张过度失败了. 省选 Day 1 考场写它,写挂了. 省选 Day 1 ...
- BZOJ 3052/Luogu P4074 [wc2013]糖果公园 (树上带修莫队)
题面 中文题面,难得解释了 BZOJ传送门 Luogu传送门 分析 树上带修莫队板子题... 开始没给分块大小赋初值T了好一会... CODE #include <bits/stdc++.h&g ...
- BZOJ 3931 / Luogu P3171 [CQOI2015]网络吞吐量 (最大流板题)
题面 中文题目,不解释: BZOJ传送门 Luogu传送门 分析 这题建图是显然的,拆点后iii和i′i'i′连容量为吞吐量的边,根据题目要求,111和nnn的吞吐量看作∞\infty∞. 然后用di ...
- BZOJ 3894 / Luogu P4313 文理分科 (拆点最小割)
题面 中文题面- BZOJ 传送门 Luogu 传送门 分析 这道题类似于BZOJ 3774 最优选择,然后这里有一篇博客写的很好- Today_Blue_Rainbow's Blog 应该看懂了吧- ...
- BZOJ 2039 / Luogu P1791 [2009国家集训队]employ人员雇佣 (最小割)
题面 BZOJ传送门 Luogu传送门 分析 考虑如何最小割建图,因为这仍然是二元关系,我们可以通过解方程来确定怎么建图,具体参考论文 <<浅析一类最小割问题 湖南师大附中 彭天翼> ...
- BZOJ 2127 / Luogu P1646 [国家集训队]happiness (最小割)
题面 BZOJ传送门 Luogu传送门 分析 这道题又出现了二元关系,于是我们只需要解方程确定怎么连边就行了 假设跟SSS分在一块是选文科,跟TTT分在一块是选理科,先加上所有的收益,再来考虑如何让需 ...
- [BZOJ 1535] [Luogu 3426]SZA-Template (KMP+fail树+双向链表)
[BZOJ 1535] [Luogu 3426]SZA-Template (KMP+fail树+双向链表) 题面 Byteasar 想在墙上涂一段很长的字符,他为了做这件事从字符的前面一段中截取了一段 ...
随机推荐
- etcd常用命令-增删改查
增删改查key-values 插入数据测试 # etcdctl put name1 james# etcdctl put name11 alice# etcdctl put name12 seli # ...
- Mycat+Mysql主从复制实现双机热备
Mycat+Mysql主从复制实现双机热备 一.mysql主从配置原理 双机热备的概念简单说一下,就是要保持两个数据库的状态自动同步.对任何一个数据库的操作都自动应用到另外一个数据库,始终保持两个数据 ...
- CF1142B Lynyrd Skynyrd
题目 有两种做法: 第一种是\(O(nlog\ n)\)的. 我们预处理两个数组: \(pre_i\)表示\(p\)中\(i\)前面的那个数是\(pre_i\). \(lst_i\)表示\(a\)中\ ...
- 【提高组NOIP2008】双栈排序 (twostack.pas/c/cpp)
[题目描述] Tom最近在研究一个有趣的排序问题.如图所示,通过2个栈S1和S2,Tom希望借助以下4种操作实现将输入序列升序排序. 操作a 如果输入序列不为空,将第一个元素压入栈S1 操作b 如果栈 ...
- phpstorm配置phpunit进行单元测试
1.配置单元测试目录: (1)autoload.php <?php function autoloader($dir){ spl_autoload_register(function($name ...
- RabbitMQ入门教程(十六):RabbitMQ与Spring集成
原文:RabbitMQ入门教程(十六):RabbitMQ与Spring集成 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https: ...
- C#实现Web链接启动应用程序
C#实现Web链接启动应用程序 最近需要配合Web端实现用户点击链接来启动应用程序并且需要能够传参数给应用程序. 那么就可以使用注册表来实现这个功能 编写注册表可以在软件安装程序中加入,也可以在软件启 ...
- div css 布局对seo 影响 布局原则
一.代码精简 使用DIV+CSS布局,页面代码精简,这一点相信对XHTML有所了解的都知道.代码精简所带来的直接好处有两点:一是提高蜘蛛爬行效率,能在最短的时间内爬完整个页面,这样对收录质量有一定好处 ...
- docker之本地连接
docker安装成功,之后我们需要连接进入docker中,这里罗列连接方式 1. Windows7 一般的虚拟ip地址: 192.168.99.100 查看ip地址: 1) C:\Users\thi ...
- ConditionalOnProperty的使用
时间 2018-02-23 标签 ConditionalOnPropert SpringBoot 栏目 Spring 原文 http://blog.csdn.net/u010002184/art ...