poj1190 生日蛋糕(深搜+剪枝)
题目链接:poj1190 生日蛋糕
解题思路:
深搜,枚举:每一层可能的高度和半径
确定搜索范围:底层蛋糕的最大可能半径和最大可能高度
搜索顺序:从底层往上搭蛋糕,在同一层尝试时,半径和高度都是从大到小试
剪枝:
①已建好的面积已经超过目前求得的最优表面积,或者预见到搭完后面积一定会超过目前最优表面积,则停止搭建(最优性剪枝)
②预见到再往上搭,高度已经无法安排,或者半径无法安排,则停止搭建(可行性剪枝)
③还没搭的那些层的体积,一定会超过还缺的体积,则停止搭建(可行性剪枝)
④还没搭的那些层的体积,最大也到不了还缺的体积,则停止搭建(可行性剪枝)
看讨论时看见了一个神剪枝:当2 * leftVolume / r + currentS >= min停止搜索
currentS代表“已有的圆柱的侧面积之和+最底下圆柱的横截面面积”。min代表已得到的最小表面积。
假设只有一个圆柱,该圆柱的半径为r,体积为leftVolume,可知:2 * leftVolume/r 表示圆柱的侧面积。
现在我们有2个圆柱,要求这两个圆柱叠在一起之后满足题目的条件:下柱半径>上柱半径。把上柱压扁,压到和下柱的半径相等,那么根据表面积和体积公式,我们知道上柱的侧面积会减小。
多个圆柱叠立,假设最下面圆柱半径最大,该半径为r。于是,这些圆柱的侧面积之和>=等体积的半径为r的圆柱的侧面积。
假设还有k层柱要搜索,leftVolume是剩余体积,r是第k层的圆柱的最大可能半径。那么2*leftVolume/r<=k层圆柱的最小侧面积之和。
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define CLR(a,b) memset((a),(b),sizeof(a))
using namespace std; const int inf = 0x3f3f3f3f;
int ans; //最优表面积
int area; //正在搭建中的蛋糕的表面积
int N, M;//体积N,层数M
int minv[]; //第i层蛋糕最少的体积
int mins[]; //第i层蛋糕的最少侧表面积 int maxV(int n, int r, int h){
//在n层蛋糕,底层最大半径r,最高高度h的情况下,能凑出来的最大体积
int v = ;
for(int i = ; i < n; ++i)
v += (r - i) * (r - i) * (h - i);
return v;
}
void dfs(int v, int n, int r, int h){
//用n层去凑体积v,最底层半径最大为r, 高度最大为 h
//求出最小表面积放入ans
if(n == ){
if(v == && area < ans){
ans = area;
return;
}
}
if(v <= ) return; if((*v*./r+area)>=ans) return;
if(area + mins[n] >= ans ) return;
if(h < n || r < n) return;
if(v < minv[n]) return;
if(maxV(n, r, h) < v) return;
for(int rr = r ; rr >= n; --rr){//从大到小搜索!
if(n == M) //底面积(总的上表面积)
area = rr * rr;
for(int hh = n; hh <= h; ++hh){
area += * rr * hh;
dfs(v - rr*rr*hh, n-, rr-, hh-);
area -= * rr * hh;
}
}
}
int main(){
int i, j;
int maxh;//底层最大高度
int maxr;//底层最大半径
scanf("%d %d", &N, &M); CLR(minv, ); CLR(mins, ); for(i = ; i <= M; ++i){
//第i层半径至少为i,,高度至少为i
minv[i] = minv[i - ] + i * i * i;
mins[i] = mins[i - ] + * i * i;
}
if(minv[M] > N){
printf("0\n"); return ;
}
area = ;
ans = inf;
//底层体积不超过(n - minv[m - 1])
maxh = (N - minv[M - ]) / (M * M) + ; //底层半径至少为 m
maxr = sqrt(.*(N - minv[M-]) / M) + ;//底层高度至少为 m dfs(N, M, maxr, maxh); if(ans == inf) printf("0\n");
else printf("%d\n", ans);
return ;
}
poj1190 生日蛋糕(深搜+剪枝)的更多相关文章
- Hdu3812-Sea Sky(深搜+剪枝)
Sea and Sky are the most favorite things of iSea, even when he was a small child. Suzi once wrote: ...
- 一本通例题-生日蛋糕——题解<超强深搜剪枝,从无限到有限>
题目传送 显然是道深搜题.由于蛋糕上表面在最底层的半径确认后就确认了,所以搜索时的面积着重看侧面积. 找维度/搜索面临状态/对象:当前体积v,当前外表面面积s,各层的半径r[],各层的高度h[]. 可 ...
- UVA 10160 Servicing Stations(深搜 + 剪枝)
Problem D: Servicing stations A company offers personal computers for sale in N towns (3 <= N < ...
- ACM 海贼王之伟大航路(深搜剪枝)
"我是要成为海贼王的男人!" 路飞他们伟大航路行程的起点是罗格镇,终点是拉夫德鲁(那里藏匿着"唯一的大秘宝"--ONE PIECE).而航程中间,则是各式各样的 ...
- hdu 1518 Square(深搜+剪枝)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1518 题目大意:根据题目所给的几条边,来判断是否能构成正方形,一个很好的深搜应用,注意剪枝,以防超时! ...
- POJ-1724 深搜剪枝
这道题目如果数据很小的话.我们通过这个dfs就可以完成深搜: void dfs(int s) { if (s==N) { minLen=min(minLen,totalLen); return ; } ...
- 模拟赛T5 : domino ——深搜+剪枝+位运算优化
这道题涉及的知识点有点多... 所以还是比较有意思的. domino 描述 迈克生日那天收到一张 N*N 的表格(1 ≤ N ≤ 2000),每个格子里有一个非 负整数(整数范围 0~1000),迈克 ...
- POJ2044 深搜+剪枝(云彩下雨)
题意: 有一个城镇,是4*4的大小的,然后你控制一块云彩,2*2的,你每天可以有9种走的方法,上下左右,或者不动,走的时候可以走1或者2步,云彩所在的地方肯定会下雨,然后给你做多365天 ...
- HDU 1175 连连看 (深搜+剪枝)
题目链接 Problem Description "连连看"相信很多人都玩过.没玩过也没关系,下面我给大家介绍一下游戏规则:在一个棋盘中,放了很多的棋子.如果某两个相同的棋子,可以 ...
随机推荐
- 502 Proxy Error The proxy server received an invalid response from an upstream server
Proxy Error The proxy server received an invalid response from an upstream server. The proxy server ...
- [CF353C]Find Maximum(贪心)
题目链接:http://codeforces.com/contest/353/problem/C 题意:给你一串数字a[]和一个二进制串,要求找一个不超过m的二进制数,使得与对应a[]上的数字的乘积和 ...
- [Lonlife1031]Bob and Alice are eating food(递推,矩阵快速幂)
题目链接:http://www.ifrog.cc/acm/problem/1031 题意:6个水果中挑出n个,使得其中2个水果个数必须是偶数,问有多少种选择方法. 设中0代表偶数,1代表奇数.分别代表 ...
- 获取Token不完整问题
有时会遇到获取Token只能获取一半的问题,明明有两个Cookie,但只获取到一个,这个是因为301重定向跳转设置问题,设置为True就可以获取到完整的Token了. myHttpWebRequest ...
- Scrum Meeting---Ten(2015-11-5)
今日已完成任务和明日要做的任务 姓名 今日已完成任务 今日时间 明日计划完成任务 估计用时 董元财 分类页设计 4h 商品详单设计 4h 胡亚坤 首页设计 2h 滚动广告栏设计 2h 刘猛 服务器测试 ...
- aop前传之代理
一.jdk提供proxy类对目标对象实现代理,简单的说对方法的调用交给代理对象来操作. 代理目标 代理的具体实现: 代理测试; 简单说:利用proxy生成一个委托类实现代理.这个委托类是目标类的接口的 ...
- [转发] 理解 oauth 2.0
原文: http://www.ruanyifeng.com/blog/2014/05/oauth_2_0.html oauth 的各种编程语言实现: http://oauth.net/2/ 理解OAu ...
- rpm and yum commands
rpm命令 rpm包,由“-”.“.”构成,包名.版本信息.版本号.运行平台 对已安装软件信息的查询 rpm -qa 查询已安装的软件 rpm ...
- python中的for循环
打印出1到100的数,不包含100 for i in range(1,100): if i==23: print "great,you got your luncky number:&quo ...
- Android Netty框架的使用
Netty框架的使用 1 TCP开发范例 发送地址---192.168.31.241 发送端口号---9223 发送数据 { "userid":"mm910@mbk.co ...