首页 > 其他分享 >2023年蓝桥杯省赛——买二赠一

2023年蓝桥杯省赛——买二赠一

时间:2024-04-08 09:58:55浏览次数:35  
标签:队列 样例 蓝桥 商品 购买 买二赠 省赛 价格 免费

目录

题目链接:1.买二赠一 - 蓝桥云课 (lanqiao.cn)

题目描述

输入格式

输出格式

样例输入

样例输出

样例说明

思路

队列+贪心

代码实现

总结


题目链接:1.买二赠一 - 蓝桥云课 (lanqiao.cn)

题目描述


        某商场有 N 件商品,其中第 i 件的价格是 Ai。现在该商场正在进行 “买二赠一” 的优惠活动,具体规则是:

        每购买 2 件商品,假设其中较便宜的价格是 P(如果两件商品价格一样,则 P 等于其中一件商品的价格),就可以从剩余商品中任选一件价格不超过 P/2的商品,免费获得这一件商品。可以通过反复购买 2 件商品来获得多件免费商品,但是每件商品只能被购买或免费获得一次。

        小明想知道如果要拿下所有商品(包含购买和免费获得),至少要花费多少钱?

输入格式


        第一行包含一个整数 N。

        第二行包含 N 个整数,代表 A1, A2, A3, . . . ,AN。

输出格式


        输出一个整数,代表答案。

样例输入


7
1 4 2 8 5 7 1
1
2


样例输出


25
1


样例说明


        小明可以先购买价格 4 和 8 的商品,免费获得一件价格为 1 的商品;再后买价格为 5 和 7 的商品,免费获得价格为 2 的商品;最后单独购买剩下的一件价格为 1 的商品。总计花费 4 + 8 + 5 + 7 + 1 = 25。不存在花费更低的方案。

对于 30% 的数据,1 ≤ N ≤ 20。

对于 100% 的数据,1 ≤ N ≤ 5 × 105,1 ≤ Ai ≤ 109


思路

队列+贪心

        这题,我真的没有爆出来一个用例,因为它真的是需要走一步考虑一步的一道题目,而且暴力的话并不好操作,所以暴力思路在这里真的就是完全骗不到分了。

        我们可以使用队列和贪心来解决这个问题。其实一直会有同学有疑惑,你的贪心代码在哪里就像动态规划的递推公式一样,看不到一个实际上存在的东西啊。看不到是因为贪心这个思路真的就是和你思维的思路是一样的。就比如这道题,我们就是要占最大的便宜,也就是买最贵的两个东西来获取最大的便宜的价格。

        这里的队列是如何使用的,或者说是如何工作的呢?这里我们使用队列来存储有多少额度的优惠可以使用。然后根据我们现将贵的东西买走,那么队列里先进入的一定是较大的优惠力度,然后我们在循环之中每次都判断当前商品的价格是否小于或者等于在队列中存储到的,已经拥有了的优惠。如果有的话就可以不加这个商品的价格了,我们直接进入下一次循环即可。

        主循环的详细解释:

        从价格最高的商品开始向下检查商品(因为 nums 数组现在是按价格排序的):

  • 如果队列非空,并且当前商品的价格小于或等于队列头部的元素(即可以免费获取的商品的上限价格),则将队列头部元素弹出,表示已经找到了可以免费获取的商品。
  • 如果当前商品不能免费获取,则需要判断是否应该被视为第二次购买:
    • 如果是,将当前商品价格的一半加入队列(表示下一次可以免费获取的商品的价格上限),并且将 flag 设置为 false,表示下一次购买视为第一次购买。
    • 如果不是,将 flag 设置为 true,表示下一次购买视为第二次购买。

代码实现



import java.util.Arrays;
import java.util.LinkedList;
import java.util.Queue;
import java.util.Scanner;

// 1:无需package
// 2: 类名必须Main, 不可修改

public class Main {
	public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		
		// 接收整数 n
		int n = sc.nextInt();
		// 接收每件商品的价格
		int[] nums = new int[n];
		for(int i = 0; i < n; i++) {
			nums[i] = sc.nextInt();
		}
		sc.close();
		// 将商品的价格从小到大的排序
		Arrays.sort(nums);
		// 定义出我们的pos当做指针,指向操作的商品的索引
		int pos = n - 1;
		Queue<Integer> queue = new LinkedList<Integer>();
		// 信号位,用于判断是否是第二次购买
		boolean flag = false;
		// 初始化结果为0,这里使用long型,因为数据有点大
		long res = 0L;
		
		// 进入贪心的循环之中
		while(pos >= 0) {
			
			// 如果上一轮给的免费额度足够这次的商品的价格
			// 那么将商品弹出即可,不用加入到res中
			if (!queue.isEmpty() && nums[pos] <= queue.peek()) {
				queue.poll();
			}else {
				if (flag) {
					// 是第二次购买则增加免费次数
					queue.add(nums[pos] / 2);
					// 然后使用了免费的次数,再将flag修改为false
					flag = false;
				}else {
					// 这次是第一次,那么下一次就是第二次了,将flag修改为true
					flag = true;
				}
				// 加到结果中
				res += nums[pos];
			}
			pos--;
		}
		System.out.println(res);
	}
}

AC喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽喽!!!!!!!!!!


总结

        其实这道题的思路难就难在呃,斯~~哪里难了。。。。。。。。。。。。。。。。

好吧,其实这里如果被题目迷惑的话,会有难度。

        来我们来看这个坑die题目的样例说明,小明可以先购买4和8,然后免费获得价格为1的商品,然后再购买5和7然后获得价格为2的商品。这个其实很难不让人往一种组合的方向上寻找答案,然后找不到便只能是放弃了。但是其实这道题我们可以不着急选赠送给我们的商品,就像这个案例一样。

1 2 2 4 5 7 8

        我们直接豪掷千金买下7和8,那么7 / 2 = 3,此时就已经有了一个额度为 3 的免费的“卷”,那么接下来就可以在遇到商品的价格小于等于3的时候直接免费拿下这个商品了,因为我们的从后往前来消费的,所以这里一定是可以免费的商品中最贵的那一个。然后接着循环即可找到最后的res结果了。 

标签:队列,样例,蓝桥,商品,购买,买二赠,省赛,价格,免费
From: https://blog.csdn.net/DDDDWJDDDD/article/details/137476479

相关文章

  • 蓝桥杯嵌入式2023年第十四届省赛主观题解析
    1 题目2 代码/*Includes------------------------------------------------------------------*/#include"main.h"#include"adc.h"#include"rtc.h"#include"tim.h"#include"gpio.h"/*Privateinclud......
  • 蓝桥杯练习系统(算法训练)ALGO-963 转圈游戏
    资源限制内存限制:128.0MB C/C++时间限制:1.0s Java时间限制:3.0s Python时间限制:5.0s问题描述n个小伙伴(编号从0到n-1)围坐一圈玩游戏。按照顺时针方向给n个位置编号,从0到n-1。最初,第0号小伙伴在第0号位置,第1号小伙伴在第1号位置,……,依此类推。游戏规......
  • 蓝桥杯练习系统(算法训练)ALGO-962 积木大赛
    资源限制内存限制:128.0MB C/C++时间限制:1.0s Java时间限制:3.0s Python时间限制:5.0s问题描述THU幼儿园举办了一年一度的“积木大赛”。今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi。在搭建开始......
  • 蓝桥杯—DS1302
    目录1.管脚2.时序&官方提供的读写函数3.如何使用读写函数4.如何在数码管中显示在DS1302中读取出的数据?1.管脚2.时序&官方提供的读写函数/* # DS1302代码片段说明 1. 本文件夹中提供的驱动代码供参赛选手完成程序设计参考。 2. 参赛选手可以自行编写相关代码或......
  • 螺旋矩阵(蓝桥杯-Python)
    importosimportsys#请在此输入您的代码n,m=input().split()n=int(n)m=int(m)arr=[[0forjinrange(m)]foriinrange(n)]r,c=input().split()r=int(r)c=int(c)defdo_l():globaln,m,r,c,arr#四个方向#右下左上......
  • FQQQ的蓝桥杯
    蓝桥杯15届备战Day213届蓝桥杯省赛文章目录蓝桥杯15届备战Day2前言主观题程序设计1.CUBEMAX配置2.代码部分(分享思路和简单实现任务)总结前言备战蓝桥杯嵌入式,刷题第二天,对象为13届蓝桥杯省赛题工程代码在此网盘提取码:xrpg提示:以下是本篇文章正文内容,下面案......
  • 【每周例题】蓝桥杯 C++ 鸡哥的蛋糕大作战
    鸡哥的蛋糕大作战题目鸡哥的蛋糕大作战 题目分析1.使用一个for循环遍历全数,寻找最大洞的数2.使用一个while进行数位拆分,寻找洞的数量3.使用if从两个条件寻找最大洞的最小数符合最大洞的数洞数相同中的最小数代码#include<iostream>#include<bits/stdc++.h>using......
  • 【每周例题】蓝桥杯 C++ 鸡哥的奇特密码
    鸡哥的奇特密码题目鸡哥的奇特密码 题目分析 1.首先,我们需要想到用一个for循环去遍历整个数组,用if寻找出需要我们处理的部分2.如何处理:将重复的L丢出数组,可以运用pop_back()函数3.为了避免越界,我们可以从后往前遍历代码#include<iostream>#include<bits/stdc++.h>u......
  • 第十四届蓝桥杯省赛A组
    目录试题A:特殊日期题解试题B:分糖果试题C:三国游戏试题D:平均试题E:翻转试题F:子矩阵题解:暴力试题G:阶乘的和题解试题H:奇怪的数试题A:特殊日期题解mon=[0,31,28,31,30,31,30,31,31,30,31,30,31]defrun(x):#判断是否为闰年ifx%400==0or(x%4==0andx%100!=0):r......
  • Acwing2024蓝桥杯递归
    模板:欧几里得算法//若a,b互质则返回1,否则返回0intgcd(inta,intb){returnb?gcd(b,a%b):a;}题目:AcWing1360.有序分数暴力模拟法(AC):#include<iostream>#include<algorithm>#definexfirst#defineysecondusingnamespacestd;intn;typed......