问题描述
小明和小红同月同日生,今天是他们的生日~
但是只有一个生日蛋糕,切成了 n 块(每块是角度为 ai 的扇形)。
现在他们两人要拿走连续的若干块蛋糕(最终没有蛋糕剩余)。他们想知道怎样分,才能使得两人得到的扇形角度的总和之差最小。
输出两人获得蛋糕总和的最小角度差。(可能出现其中一人分不到任何蛋糕的情况。)
输入形式
第一行一个整数 n(1≤n≤360),表示蛋糕切成的扇形块数
第二行 n 个整数 ai(1≤ai≤360),表示每块扇形的角度,保证角度之和为 360 。输入数据是一个环。
输出形式
输出一个非负整数,表示小明与小红获得蛋糕角度之差的最小值。
样例输入
3
100 160 100
样例输出
40
样例说明
红色与黄色部分分别表示小 明 与小 红 分得的蛋糕,小 名 拿第 1、3 块,小 红 拿第 2 块。
解题思路:
要知道两人无论怎么分,蛋糕总共也就360
假设小明分到的蛋糕为x
那么两人蛋糕的差就是 x - (360 - x) = 2 * x -360; 如果为负数的话取绝对值就可以了
那么这样问题就转化为了求 abs(2 * x -360)的最小值的问题了,显然,当 x 最接近 180 时,两人分到蛋糕的差最小
所以遍历每一块蛋糕,找到第一块小于180 的蛋糕,从这块蛋糕开始左右扩展(也可以不是第一块小于180的蛋糕,随便一块都可以)
当然,如果遍历到一块180的蛋糕,那就可以直接结束了,差为0
如果这块蛋糕加上它左右相邻的较小的一块蛋糕还小于等于180的话,那就继续扩展
当加上左右相邻的蛋糕中较小的那一块,和仍然大于180的时候,那就不加了,退出while循环
记录一下小于等于180最接近180的数 ,记为x1;
当然,若 x1 = 180 也是可以直接结束了,最小的差为0
然后 x 加上 左右相邻的蛋糕中较小的那一块
然后 x 和 x1 比较,看谁最接近180
然后 x 就等于 最接近180的那个数
最后 abs(2 * x -360) 就是所求的结果了。AC
贴代码
#include <cstdio>
#include <iostream>
#include <cstdlib>
#include <cmath>
using namespace std;
struct cake
{
int data;
int flag;
} c[361];
int main()
{
int n, i;
cin >> n;
for (i = 1; i <= n; i++)
{
cin >> c[i].data;
c[i].flag = 0;
}
int x = 0; //x代表小明获得的蛋糕
int k = 0;
for (i = 1; i <= n; i++)
{
if (c[i].data == 180)
{
printf("0\n");
return 0;
}
else if (c[i].data < 180)
{
k = i;
break;
}
}
x = x + c[k].data;
int kl = k; //小明分到蛋糕的左边界
int kr = k; //小明分到蛋糕的右边界
int l = kl - 1;
int r = kr + 1;
while (x < 180)
{
l = kl - 1;
if (l == 0)
{
l = n;
}
r = kr + 1;
if (r == n + 1)
{
r = 1;
}
if (c[l].data < c[r].data && x + c[l].data <= 180)
{
x = x + c[l].data;
kl = l;
}
else if (c[l].data >= c[r].data && x + c[r].data <= 180)
{
x = x + c[r].data;
kr = r;
}
else
{
break;
}
}
if (x == 180)
{
printf("0\n");
return 0;
}
int x1 = x; //小于180最接近180的数
if (c[l].data < c[r].data)
{
x = x + c[l].data;
}
else if (c[l].data >= c[r].data)
{
x = x + c[r].data;
}
if (x-180 > 180-x1) //说明x1更接近180
{
x = x1;
}
x = 2 * x - 360;
printf("%d\n",abs(x));
system("pause");
return 0;
}
结束语:
这道题呢,难度一般,甚至可以说有点简单,因为没有用到什么高深的算法,一开始思路错了花费我很多时间,所以贴个文自己检讨一下,第一篇帖子哈哈,希望大家多多支持