【寒假每日一题】洛谷 P7471 [NOI Online 2021 入门组] 切蛋糕

文章描述了一个关于如何公平地按照比例切分蛋糕给Alice、Bob和Cindy的问题,提出了根据比例中0的个数来分类解决的策略。解题思路包括分析比例中的0数量以及比例关系,给出不同情况下最少切刀次数的判断条件。代码示例展示了如何实现这个解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目链接:P7471 [NOI Online 2021 入门组] 切蛋糕 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

题目描述

Alice、Bob 和 Cindy 三个好朋友得到了一个圆形蛋糕,他们打算分享这个蛋糕。

三个人的需求量分别为 a,b,c,现在请你帮他们切蛋糕,规则如下:

1. 每次切蛋糕可以选择蛋糕的任意一条直径,并沿这条直径切一刀(注意切完后不会立刻将蛋糕分成两部分)。

2. 设你一共切了 n 刀,那么你将得到 2n 个扇形的蛋糕(特别地,切了 0 刀被认为是有一个扇形,即整个圆形蛋糕),将这些蛋糕分配给 Alice,Bob 和 Cindy,要求每个扇形蛋糕只能完整地分给一个人。

3. 三人分到的蛋糕面积比需要为 a:b:c(不保证是最简比例,且如果 a:b:c 中某个数为 0,表示那个人不吃蛋糕)。

为了完成这个任务,你至少需要切几刀?

输入格式

本题单个测试点包含多组数据。

第一行包含一个整数 T,表示数据组数。

接下来 T 行,每行包含三个整数 a,b,c,表示三人的需求量。

输出格式

输出 T 行,第 i 行的输出表示第 i 组数据中你至少需要切蛋糕的次数。

样例 #1

样例输入 #1

6
0 0 8
0 5 3
9 9 0
6 2 4
1 7 4
5 8 5

样例输出 #1

0
2
1
2
3
2

提示

样例 1 解释

数据范围与提示

30% 的数据满足:a = b = 0。

60% 的数据满足:a = 0。

100% 的数据满足:1 <= T <= 10^4,0 <= a,b,c <= 10^8,保证 a + b + c > 0。

数据由 SSerxhs 提供。

解题思路

这道题可以通过观察测试样例,并且根据不同比例尝试分割蛋糕。我们可以找出以下规律:

可以按照比例中0的个数,分为三大情况:

1)若比例中有2个0,则返回0(样例 0 0 8)

2)若比例中只有1个0

①若剩下2个比例相等,则返回1(样例 9 9 0)

②若剩下2个比例不相等,则返回2(样例 0 5 3)

3)若比例中三个数都不为0

①若3个比例都相同,则返回2(三个部分各为120°)

②若只有2个比例相同,则返回2(样例 5 8 5)

③若3个比例各不相同,且没有2个比例相加等于另一个比例,则输出3(样例 1 7 4)

④若3个比例各不相同,且有2个比例相加等于另一个比例,则输出2(样例 6 2 4)

AC code:

#include<iostream>
#include<algorithm>
#include<vector>
#include<unordered_set>

using namespace std;

int solution(vector<int> &a)
{
	unordered_set<int> s;
	int cntzero = 0; 
	for(auto &i : a)
	{
		if(i == 0)
			cntzero ++; // 统计比例中0的个数 
		s.insert(i);
	}
	if(cntzero == 2) // 若比例中有2个0,则返回0 
		return 0;
	else if(cntzero == 1) // 若比例中只有1个0 
	{
		if(s.size() == 2) // 若剩下2个比例相等,则返回1 
			return 1;
		else if(s.size() == 3) // 若剩下2个比例不相等,则返回2 
			return 2;
	}
	else if(cntzero == 0) // 若比例中三个数都不为0 
	{
		if(s.size() == 1) // 若3个比例都相同,则返回2 
			return 2;
		else if(s.size() == 2) // 若3个比例中,只有2个比例相同,则返回2 
			return 2;
		else if(s.size() == 3) // 若3个比例各不相同 
		{
			sort(a.begin(),a.end()); // 排序,方便进行比例计算 
			if(a[0] + a[1] == a[2]) // 若有2个比例相加等于另一个比例,则返回2 
				return 2;
			else // 若没有2个比例相加等于另一个比例,则返回3 
				return 3;
		}
	}
}

int main()
{
	vector<int> v(3); 
	int tt;
	cin>>tt;
	while(tt --)
	{
		cin>>v[0]>>v[1]>>v[2];
		cout<<solution(v)<<endl;
	} 
	
	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值