题目链接:

这个题目非常好,有难度;能够好好的多做做;

#include<iostream>
#include<string>
#include<cstdio>
#include<cstring>
#include<queue>
#include<map>
#include<cmath>
#include<stack>
#include<set>
#include<vector>
#include<algorithm>
#define LL long long
using namespace std;
int MinArea=1<<30;  //  存最优表面积。
int MinV[30];       //  存第一层到该层最小体积;
int MinA[30];       //  存第一层到该层最小面积。
int N,M;            //  体积,层数。
int area=0;         //  存增在搭建的表面积;
int MaxV(int n,int r,int h)     //  当有n层时。能够有的最大体积;当中r为最大半径,h为最大高度;
{
    int v=0;
    for(int i=0;i<n;i++) v+=(r-i)*(r-i)*(h-i);
    return v;
}
void dfs(int v,int n,int r,int h)
{
    if(n==0){           //  说明一种情况搭建层数已经完毕。
        if(v) return;
        else{
            MinArea=min(MinArea,area);      //  能够搭建,则更新最优解。
            return;
        }
    }
    if(v<=0) return;                    //  体积不够,退出。
    if(MinV[n]>v) return;               //  搭建n层的最小体积比提供的体积大。即无法搭建。退出;
    if(area+MinA[n]>=MinArea) return;   //  当前搭建表面积加上前n层最小的表面积比最优解更大。则能够退出。
    if(h<n||r<n) return;                //  最大半径,或者最大高度比层数还要多。则就说明已经无法继续搭建了。
    if(MaxV(n,r,h)<v) return;           //  这n层能够搭建的最大体积都比提供的体积要小,说明无法按要求搭建;
    for(int rr=r;rr>=n;rr--){       //  从最大半径開始枚举搜索。
        if(n==M) area=rr*rr;        //  底面积;
        for(int hh=h;hh>=n;hh--){   //  从最大高度開始枚举。
            area+=2*rr*hh;          //  搭建的面积更新;
            dfs(v-rr*rr*hh,n-1,rr-1,hh-1);
            area-=2*rr*hh;          //  回溯;
        }
    }
}
int main()
{
    while(~scanf("%d%d",&N,&M)){
        MinV[0]=0;
        MinA[0]=0;
        for(int i=1;i<=M;i++){
            MinV[i]=MinV[i-1]+i*i*i;                    //  前i层最小体积;
            MinA[i]=MinA[i-1]+2*i*i;                    //  前i层最小面积;
        }
        if(MinV[M]>N) printf("0\n");
        else{
            int MaxH=(N-MinV[M-1])/(M*M)+1;             //  底层最大高度。
            int MaxR=sqrt(double(N-MinV[M-1])/M)+1;     //  底层最大半径;
            int area=0;
            MinArea=1<<30;
            dfs(N,M,MaxH,MaxR);
            if(MinArea==1<<30)  printf("0\n");
            else  printf("%d\n",MinArea);
        }
    }
    return 0;
}