1710 生日蛋糕(1999 noi)

1999年NOI全国竞赛

题目描述 Description

7月17日是Mr.W的生日,ACM-THU为此要制作一个体积为Nπ的M层生日蛋糕,每层都是一个圆柱体。

设从下往上数第i(1<=i<=M)层蛋糕是半径为Ri,高度为Hi的圆柱。当i<M时,要求Ri>Ri+1且Hi>Hi+1

由于要在蛋糕上抹奶油,为尽可能节约经费,我们希望蛋糕外表面(最下一层的下底面除外)的面积Q最小。

令Q= Sπ

请编程对给出的N和M,找出蛋糕的制作方案(适当的Ri和Hi的值),使S最小。

(除Q外,以上所有数据皆为正整数)

输入描述 Input Description

有两行,第一行为N(N<=10000),表示待制作的蛋糕的体积为Nπ;第二行为M(M<=20),表示蛋糕的层数为M。

输出描述 Output Description

仅一行,是一个正整数S(若无解则S=0)。

样例输入 Sample Input

100   2

样例输出 Sample Output

68

数据范围及提示 Data Size & Hint

体积V=πR2H

侧面积A’=2πRH

底面积A=πR2

题目分析:搜索题,其实不减枝的代码还是很好写的。

参考程序:

#include<stdio.h>

#include<string.h>

#define maxn 22

#define INF 100000000

int N,M,ans,maxh;

//m为蛋糕的层数, v为当前的体积, s为当前得到的面积,r和h为当前层的半径和高

void dfs(int m,int v,int s,int r,int h){

//退出条件

if(m == 0){

if(ans > s && v == N) ans = s;

return;

}

//枚举可能的解

for(int i = r-1; i >= m; i--){

for(int j = maxh; j >= m; j--){

if(m == M) s = i * i;

dfs(m-1,v+i*i*j,s+2*i*j,i,j);

}

}

}

int main(){

scanf("%d%d",&N,&M);

ans = INF;

maxh=N/M/M;//唯一的剪枝,底层蛋糕的最大高度N/(M*M)

dfs(M,0,0,N+1,N+1);

if(ans == INF) printf("0\n");

else printf("%d\n",ans);

return 0;

}

然后就是剪枝呢,四个减枝的条件:

先打表,算出每层蛋糕的最小体积和表面积(minv[i]和mins[i]),然后在来减枝

1、v+minv[m]  > V

v为已经涂的体积,那么如果v加上下一层最小的体积比总体积V还大,这显然是不可能的,减去。

2、s+min[m] > ans

s为已经涂的面积,那么s加上下一次最小的面积比当前求得的ans还大,显然不需要dfs了,减去。

3、2*(V-v)/r + s >= ans

已经涂了s,那么还剩下rest_s = sum{2*Ri*Hi} >= sum{2*Ri*Ri*Hi/Rk} = 2*(V-v)/r (设k为当前层的半径)。如果rest_s加上s大于等于ans,那么也不用在dfs了。

4、maxh = Min((N-v-minv[m-1])/(i*i),h-1)

当枚举半径为i时,当前最低的可能高为maxh = Min((N-v-minv[m-1])/(i*i),h-1)。

剪枝后的代码:

//注意:本程序蛋糕自顶向下编号(与题目相反),dfs由下向上运行.

#include<iostream>

#include<cmath>

using namespace std;

const int INF=1000000;

const int Size=22;

int N,M;

int ans;

int mins[Size],minv[Size];

int maxh;

void init(){

mins[0]=minv[0]=0;

for(int i=1;i<=M;i++){

mins[i]=mins[i-1]+2*i*i;

minv[i]=minv[i-1]+i*i*i;

}

}

void dfs(int m,int v,int s,int lastr,int lasth){

//cout<<m<<‘ ‘;

if(m==0){

if(v==N)ans=min(ans,s);

return;

}

if(v+minv[m]>N)return;

if(s+mins[m]>=ans)return;

if(2*(N-v)/lastr+s>=ans)return;

for(int r=lastr-1;r>=m;r--){

int maxh=min((N-v-minv[m-1])/(r*r),lasth-1);//注意这里是minv[m-1],因为 总体积N – 已经有的v – 将要有的最小体积minv[m+1]  才是 m这一层的最大体积,进而计算最大高度

//cout<<maxh<<endl;

for(int h=maxh;h>=m;h--){

if(m==M)s=r*r;//整个蛋糕与桌面向平的那些环面的面积之和==底层蛋糕的圆面的面积,在此加上

dfs(m-1,v+r*r*h,s+2*r*h,r,h);

}

}

}

int main(){

freopen("1.in","r",stdin);

cin>>N>>M;

ans=INF;

maxh=N/M/M;

init();

int maxr=sqrt(N);

dfs(M,0,0,maxr+1,N+1);

cout<<ans<<endl;

return 0;

}

时间: 2024-07-30 14:38:30

1710 生日蛋糕(1999 noi)的相关文章

生日蛋糕 (codevs 1710) 题解

[问题描述] 7月17日是Mr.W的生日,ACM-THU为此要制作一个体积为Nπ的M层生日蛋糕,每层都是一个圆柱体. 设从下往上数第i(1<=i<=M)层蛋糕是半径为Ri,高度为Hi的圆柱.当i<M时,要求Ri>Ri+1且Hi>Hi+1. 由于要在蛋糕上抹奶油,为尽可能节约经费,我们希望蛋糕外表面(最下一层的下底面除外)的面积Q最小. 令Q= Sπ 请编程对给出的N和M,找出蛋糕的制作方案(适当的Ri和Hi的值),使S最小. (除Q外,以上所有数据皆为正整数) [样例输入]

【线段树】【NOI 1999】【cogs 284】内存分配

284. [NOI1999] 内存分配 ★★☆ 输入文件:memory.in 输出文件:memory.out 简单对比 时间限制:1 s 内存限制:128 MB 内存是计算机重要的资源之一,程序运行的过程中必须对内存进行分配. 经典的内存分配过程是这样进行的: 内存以内存单元为基本单位,每个内存单元用一个固定的整数作为标识,称为地址.地址从0开始连续排列,地址相邻的内存单元被认为是逻辑上连续的.我们把从地址i开始的s个连续的内存单元称为首地址为i长度为s的地址片. 运行过程中有若干进程需要占用内

BZOJ NOI 1999 钉子和小球 动态规划+分数类

题目大意:不太好描写叙述,自己看吧.. 思路:首先从最上面的点開始考虑.由于球一定是从最上面開始往下掉,所以球经过最上面的点的概率是1,然后他会有1/2的几率向左,1/2的几率向右,也就是以下的两个点均分上面点的几率. 当然这是全部的点都存在的情况.假设有哪里的点不存在了,那么求落到这个点的几率不变,然后它的全部几率都会加在在它以下两行且在正下方的点. 依照这样写dp方程.显然是不难的.之后就是恶心的输出了.两个方案,1.遇到小数就*2,保证它是整数.可是最高有50层.就要考虑一下2^50这么大

POJ1190生日蛋糕[DFS 剪枝]

生日蛋糕 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 18236   Accepted: 6497 Description 7月17日是Mr.W的生日,ACM-THU为此要制作一个体积为Nπ的M层生日蛋糕,每层都是一个圆柱体. 设从下往上数第i(1 <= i <= M)层蛋糕是半径为Ri, 高度为Hi的圆柱.当i < M时,要求Ri > Ri+1且Hi > Hi+1. 由于要在蛋糕上抹奶油,为尽可能

POJ1190 生日蛋糕

Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 18103   Accepted: 6439 Description 7月17日是Mr.W的生日,ACM-THU为此要制作一个体积为Nπ的M层生日蛋糕,每层都是一个圆柱体. 设从下往上数第i(1 <= i <= M)层蛋糕是半径为Ri, 高度为Hi的圆柱.当i < M时,要求Ri > Ri+1且Hi > Hi+1. 由于要在蛋糕上抹奶油,为尽可能节约经费,

整理小朋友在noi.openjudge上的作业(4)

第四章(含小学奥数)计97题,已完成8题 4.1 算法之排序和算法性能   题目ID 标题 分数 尝试人数   1625 Sequence Median 10 90   1754 字符串数组排序问题 10 49   1999 日志排序 10 50   4363 瑞士轮 10 101 4.2 算法之数论   题目ID 标题 分数 尝试人数   1350 Euclid's Game 10 223   1486 A Funny Game 10 70   185 反正切函数的应用 10 44   241

HDU 1710

http://acm.hdu.edu.cn/showproblem.php?pid=1710 题意:给二叉树的先序遍历和中序遍历,确定后序遍历 解法:复习专业课找的一题,根据先序遍历和中序遍历建树,再对树做后序遍历 #include <iostream> #include <cstdio> using namespace std; struct Tree { Tree *lc, *rc; int data; }; Tree *Create(int *preorder, int *i

linux出现提示信息Display all 1710 possibilities? (y or n)的总结

出现Display all 1710 possibilities? (y or n)的原因 现象如下 出现这种现象的原因是在什么也没有输入的情况下连续按两次tab键. 出现Display all 1710 possibilities? (y or n)之后,只要按n就会返回命令行了,对系统或其他进程不会产生任何影响.

[NOI2011]Noi嘉年华

题目描述 NOI2011 在吉林大学开始啦!为了迎接来自全国各地最优秀的信息学选手,吉林大学决定举办两场盛大的 NOI 嘉年华活动,分在两个不同的地点举办.每个嘉年华可能包含很多个活动,而每个活动只能在一个嘉年华中举办. 现在嘉年华活动的组织者小安一共收到了 n个活动的举办申请,其中第 i 个活动的起始时间为 Si,活动的持续时间为Ti.这些活动都可以安排到任意一个嘉年华的会场,也可以不安排. 小安通过广泛的调查发现,如果某个时刻,两个嘉年华会场同时有活动在进行(不包括活动的开始瞬间和结束瞬间)