两人分蛋糕,角度差最小 算法问题

问题描述

小明和小红同月同日生,今天是他们的生日~
但是只有一个生日蛋糕,切成了 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;
}
结束语:

这道题呢,难度一般,甚至可以说有点简单,因为没有用到什么高深的算法,一开始思路错了花费我很多时间,所以贴个文自己检讨一下,第一篇帖子哈哈,希望大家多多支持

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

鲸鱼的眼泪

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值