时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K 64bit IO Format: %lld
题目描述
小H有n个碗需要放进橱柜,她希望将他们叠起来放置。你知道每个碗都是规则的圆柱体,并且都是上宽下窄,你已经测量出了每个碗的两个半径及高,请你帮小H找出一种叠放顺序,使得叠放出来的碗堆的高度尽量小,比如:
100%数据满足n < = 9。所有输入的数绝对值不超过1000。
输入描述:
第一行一个整数n,表示碗的数目。 以下n行,每行三个整数h,r1,r2。分别表示碗高及两个半径。其中r1<r2
输出描述:
仅一个数,表示最小的高度。答案四舍五入取整。
示例1
输入
复制
3 50 30 80 35 25 70 40 10 90
输出
复制
55
题解:
n<9,所以直接枚举就行,挨个试一遍取最优解
碗的叠放分几种情况:
当前的总高度应该是上底相对于上个碗的高度+上个碗底的高度
一个一个的放,然后看看哪一种情况下高度最高,暴力就完事了
思路很简单,但实现有点麻烦
斜率:2*h/(x1-x2)
代码:
我调了一阵子都没a。。。放弃了
直接借用大佬代码,参考处理的细节
代码选自
#include<bits/stdc++.h> using namespace std; typedef long long ll; const ll inf = 0x3f3f3f3f; const int maxn = 1e6 + 10; double h[maxn], r1[maxn], r2[maxn], tmp[maxn]; double calc(int x, int y) { if (r1[x] > r2[y])//第一种情况 return h[y]; if (r2[x] < r1[y])// return 0; if ((r2[x] - r1[x]) * h[y] <= (r2[y] - r1[y]) * h[x]) { if (r1[x] <= r1[y]) return 0; return h[y] * (r1[x] - r1[y]) / (r2[y] - r1[y]); } if (r2[x] > r2[y]) return max(0.0, h[y] - h[x] * (r2[y] - r1[x]) / (r2[x] - r1[x])); return max(0.0, h[y] * (r2[x] - r1[y]) / (r2[y] - r1[y]) - h[x]); } int id[maxn]; int main() { int n; cin >> n; for (int i = 1; i <= n; i++) cin >> h[i] >> r1[i] >> r2[i], id[i] = i; double ans = inf, res = 0; do{ res = 0; for (int i = 1; i <= n; i++){ tmp[i] = 0; for (int j = 1; j < i; j++) tmp[i] = max(tmp[i], tmp[j] + calc(id[i], id[j])); res = max(tmp[i] + h[id[i]], res); } ans = min(ans, res); }while(next_permutation(id + 1, id + 1 + n)); printf("%d\n",(int)(ans+0.5)); return 0; }