P4160 [SCOI2009]生日快乐[dfs]

题目描述

windy的生日到了,为了庆祝生日,他的朋友们帮他买了一个边长分别为 X 和 Y 的矩形蛋糕。

现在包括windy,一共有 N 个人来分这块大蛋糕,要求每个人必须获得相同面积的蛋糕。

windy主刀,每一切只能平行于一块蛋糕的一边(任意一边),并且必须把这块蛋糕切成两块。

这样,要切成 N 块蛋糕,windy必须切 N-1 次。

为了使得每块蛋糕看起来漂亮,我们要求 N 块蛋糕的长边与短边的比值的最大值最小。

你能帮助windy求出这个比值么?

解析

这题使我深刻领悟到我有多菜。

结论:切的位置的坐标一定是\(X/N\)或\(Y/N\)的整数倍。

证明:用反证法。若某一刀切出的短边比\(X/N\)小,那么一定需要一个比\(Y\)还长的长边才能满足题目要求,显然不能切。

若某一刀切得比\(X/N\)大,比如切出来一个\(X/N+k,k<X/N\),那么切出来的这块矩形的长边一定小于\(Y\)且大于\(Y*(N-1)/N\),这样就会余出来一段\(p\),且\(0<p<Y/N\)。然而为了满足题目条件,我们势必要使\(p\)构成的矩形面积也等于\(X*Y/N\),但是它最大也就是\(p*X\),显然\(p*X<X*Y/N\),这是不可能的。

证毕。

明白这一点之后这题就简单了,但是我是个连深搜都写不对的菜鸡。

显然切一刀会分成两个子问题,而每次我们可以在所有合法位置横纵切。选择子问题所有情况中最大值最小的情况进行回溯即可。

参考代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<string>
#include<cstdlib>
#include<queue>
#include<vector>
#define INF 0x3f3f3f3f
#define PI acos(-1.0)
#define N 101
#define MOD 2520
#define E 1e-12
using namespace std;
inline int read()
{
    int f=1,x=0;char c=getchar();
    while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
    while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
    return x*f;
}
double n;
double x,y,kx,ky;
inline double dfs(double k,double nx,double ny)
{
    if(k==1){
        if(nx<ny) swap(nx,ny);
        return nx/ny;
    }
    double ans=INF,t1,t2;
    double kx=nx/k,ky=ny/k;
    for(int i=1;i<=k/2;++i){
        t1=max(dfs(k-i,nx-i*kx,ny),dfs(i,i*kx,ny));
        t2=max(dfs(k-i,nx,ny-i*ky),dfs(i,nx,i*ky));
        ans=min(ans,min(t1,t2));
    }
    return ans;
}
int main()
{
    cin>>x>>y>>n;
    printf("%.6lf",dfs(n,x,y));
    return 0;
}

原文地址:https://www.cnblogs.com/DarkValkyrie/p/11722097.html

时间: 2024-11-10 15:12:22

P4160 [SCOI2009]生日快乐[dfs]的相关文章

BZOJ 1024: [SCOI2009]生日快乐 dfs

1024: [SCOI2009]生日快乐 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/problem.php?id=1024 Description windy的生日到了,为了庆祝生日,他的朋友们帮他买了一个边长分别为 X 和 Y 的矩形蛋糕.现在包括windy,一共有 N 个人来分这块大蛋糕,要求每个人必须获得相同面积的蛋糕. windy主刀,每一切只能平行于一块蛋糕的一边(任意一边

BZOJ1024 SCOI2009 生日快乐 DFS

题意:给定一个X*Y的长方形,每次可以平行长或宽切一刀,求切N-1次能得到的最小的N个子长方形中长宽比最大的值. 题解:因为N<=10,DFS乱搞就好 #include <cstdio> #include <cstring> #include <cstdlib> #include <iostream> #include <algorithm> using namespace std; #define INF 66666.0 int N; d

P4160 [SCOI2009]生日快乐

传送门 一看 $n$ 这么小,搜就完事了... 因为最后每块小蛋糕面积固定,所以每次切完面积都必须是小蛋糕面积的倍数 那么最多只有第一次有 $10$ 个位置,之后越来越少,复杂度很低 然后注意不要乱剪枝...,每次切不一定只切长的边,枚举位置时因为左右两边是对称的所以只要枚举一半 #include<iostream> #include<cstdio> #include<algorithm> #include<cstring> #include<cmat

BZOJ 1024 [SCOI2009]生日快乐 (搜索)

1024: [SCOI2009]生日快乐 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3025  Solved: 2201[Submit][Status][Discuss] Description windy的生日到了,为了庆祝生日,他的朋友们帮他买了一个边长分别为 X 和 Y 的矩形蛋糕.现在包括windy,一共有 N 个人来分这块大蛋糕,要求每个人必须获得相同面积的蛋糕.windy主刀,每一切只能平行于一块蛋糕的一边(任意一边),并且必须把这

bzoj千题计划115:bzoj1024: [SCOI2009]生日快乐

http://www.lydsy.com/JudgeOnline/problem.php?id=1024 枚举横着切还是竖着切,一边儿分多少块 #include<cstdio> #include<algorithm> using namespace std; double S; double dfs(double x,double y,int n) { if(n==1) return max(x,y)/min(x,y); double ans=1e9; for(int i=1;i&

SCOI2009生日快乐

竟然是搜索……囧 还以为是什么神题…… 1 uses math; 2 var x,y:extended; 3 n:longint; 4 function find(x,y:extended;z:longint):extended; 5 var i:longint; 6 r1,r2:extended; 7 begin 8 find:=10000000; 9 if z=1 then 10 begin 11 if x>y then exit(x/y) else exit(y/x); 12 end; 1

[BZOJ 1024][SCOI2009]生日快乐(DFS)

Description windy的生日到了,为了庆祝生日,他的朋友们帮他买了一个边长分别为 X 和 Y 的矩形蛋糕.现在包括windy ,一共有 N 个人来分这块大蛋糕,要求每个人必须获得相同面积的蛋糕.windy主刀,每一切只能平行于一块蛋糕 的一边(任意一边),并且必须把这块蛋糕切成两块.这样,要切成 N 块蛋糕,windy必须切 N-1 次.为了使得 每块蛋糕看起来漂亮,我们要求 N块蛋糕的长边与短边的比值的最大值最小.你能帮助windy求出这个比值么? Solution 爆搜可过 #i

bzoj1024: [SCOI2009]生日快乐(DFS)

dfs(x,y,n)表示长为x,宽为y,切n块 每次砍的一定是x/n的倍数或者y/n的倍数 #include<bits/stdc++.h> using namespace std; const int inf=1e9; int x,y,n; double dfs(double x,double y,int n) { if(n==1)return max(x,y)/min(x,y); double sum=1.0*inf; double xx=x/n,yy=y/n; for(int i=1;i&

bzoj 1024 SCOI2009 生日快乐

一上来没思路…… 但是由于每块蛋糕面积相等,又因为每次切必须切成两半,所以每半的面积都是每块蛋糕面积的倍数. 枚举切下来的蛋糕应该还被分为多少块蛋糕,计算切割的地方,分X 和 Y, 递归搜索. 上代码: #include <cstdio> #include <cstring> #include <cstdlib> #include <iostream> #include <algorithm> #include <cmath> #de