P9234 [蓝桥杯 2023 省 A] 买瓜 题解

题目传送门

前言

说实话这题根本用不到什么折半……,今天看机房大佬写了半天加了一堆剪枝还以为很难,其实是你们想复杂了

20分钟不到从看题到代码实现

这题其实只需要可行性剪枝加排序 哦还有个后缀和

进入正题

小木棍子都听说过吧 没错就是小波上课打挂那道

跟这题没多大关系,不过如果你切了小木棍,就会觉得这道题很简单

讲讲我一开始的思路

一开始因为机房大佬在各种卡常,玄学剪枝,大叫折半是个好东西,还以为是个和小木棍一样的毒瘤

讲真我不喜欢打折半

第一眼看,排序,然后和埃及分数一样根据后续的瓜全买能不能满足剪枝,然后搜索的时候加个二分寻找当前第一个切开比剩下小的值

后面发现因为数据水所以加不加二分没差多少

最后清晰的讲述一下我的思路

第一步,先将所有的元素从大到小进行排序,然后做一下后缀和(后面可行性剪枝用)

第二步,开始搜索。

搜索的时候注意顺序要从前往后搜,也就是说后面被搜到的元素不能大于前面的(这里感性理解一下,如果大的搜了搜小的,然后搜完小的又去搜大的就重复了,排序就没有意义了)

关于可行性剪枝自然就是用第一步求出的后缀和直接判断一下后面所有的瓜加起来有没有剩下需要的瓜多

然后就结束了

关于一些小技巧

可以在读入的时候就把数据乘 2 ,这样就可以用 l o n g l o n g long long longlong 存下了(机房大佬说double常数很大)

然后就是把题目看清楚,求的是 需要切开的瓜,还有如果不行要输出 − 1 -1 1 不然你会因此 W A WA WA 一个点

Code

#include <bits/stdc++.h>
#define int long long//记得开 long long 
#define ull unsigned long long

const int N = 1e6+10;
const int M = 1e4+10;
const int mod = 1e9+7;
const int INF = 0x3f3f3f3f;

using namespace std;
int a[40],n,m,b[40];
bool esmite(int pos,int res){
	return b[pos+1] >= res;
}
int ans = INF;
int find(int x){// STL熟练的可以使用 upper_bound 或者 lower_bound 本蒟蒻这两玩意用法分不清故手写 
	int l = 1, r = n;
	while(l < r){
		int mid = (l+r) >> 1;
		if(a[mid] / 2 <= x){
			r = mid;
		}else{
			l = mid+1;
		}
	}
	return l;
}
void dfs(int num,int rest,int pos){//num 当前切开了几个瓜,rest 还剩下需要多少瓜,pos当前搜到哪个位置了,防止往前搜 
	if(rest == 0){//统计答案 
		ans = min(ans,num);
		return;
	}
	if(!esmite(pos,rest)) return;//可行性剪枝 
	for(int i = max(pos+1,find(rest));i <= n; i++){//当然这里也可以直接pos+1(说过了数据水) 
		if(a[i] / 2 > rest) continue;
		dfs(num+1,rest - a[i] / 2, i);
		if(a[i] > rest) continue;
		dfs(num,rest - a[i], i);
	}
}
signed main(){
	cin >> n >> m;
	m *=2;//乘2小技巧 
	for(int i = 1; i <= n; i++){
		cin>> a[i];
		a[i] *= 2;
	}
	sort(a+1,a+1+n,greater<int>());//排序 
	for(int i = n; i > 0; i--){//后缀和 
		b[i] = b[i+1] + a[i];
	}
	dfs(0,m,0);
	if(ans == INF) cout<< -1;
	else cout << ans;
	return 0;
}

后记

瓜瓜永远的神! 吃瓜教万岁!

  • 6
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 15
    评论
蓝桥杯是一个国内著名的计算机比赛,为了帮助参赛者更好地准备和了解比赛的题型,组委会会公布历年的真题并提供相应的题解。 首先,我们需要了解蓝桥杯是一个综合性的计算机比赛,测试的对象包括计算机基础知识、编程能力以及解决实际问题的能力。 在历年的真题中,参赛者将面临不同类型的题目,包括算法设计与优化问题、数据结构与算法问题、编程题等。其中针对Python B组的题目主要考察的是对Python语言的掌握和应用能力。 题目解答一般会包含以下几个方面的内容: 1. 题目分析与理解:读取题目,理解题目的要求和限制条件。通过仔细分析题目,确定题目的输入与输出,以及问题的核心。 2. 设计解决方案:根据题目要求和限制条件,设计一个合适的解决方案。可以使用合适的算法和数据结构来解决问题,并做出相应的性能优化。 3. 编写代码实现:根据设计的方案编写相应的代码实现。需要注意的是,Python语言有其独特的语法和特性,掌握好这些特性可以更好地完成编程任务。 4. 调试与测试:编写完代码后,需要进行调试和测试。通过运行样例输入和输出,检查代码是否符合题目要求,并且没有逻辑上的错误。 5. 总结与优化:在完成题目解答后,可以进行总结和优化。包括分析算法复杂度、代码风格和可读性等方面,以便在比赛中更好地表现。 在准备蓝桥杯时,可以通过阅读历年的真题和题解来了解比赛的难度和类型,针对性地进行练习和提高。同时也可以参加相关的培训班和讨论活动,与其他参赛者交流经验和技巧。 总而言之,历年蓝桥杯真题的解答对于提高自己的编程能力和应对比赛非常有帮助。通过认真分析和实践,可以更好地理解并掌握Python编程,并在比赛中取得更好的成绩。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Bamboo_Day

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

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

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

打赏作者

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

抵扣说明:

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

余额充值