2024第十五届蓝桥杯C++大学A组压轴题解:封印宝石

2024-05-15 16:52

本文主要是介绍2024第十五届蓝桥杯C++大学A组压轴题解:封印宝石,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目:第十五届蓝桥杯C++大学A组省赛压轴题

题目传送门

题意:将n个数放在n个位置上,每个数只能放在它自己之前的位置上,且离自己多远就花费多少代价,可以有没放的数,给出最大代价要求最后放的数排成的字典序最大。

字典序最大带来的就必定是贪心,必须每次都把能放最大的一个值放到前面,同时为了节省体力,需要选相同的这个最大值最前面的一个。

对于当前位置i,也就是求i到i+k(当前体力)最大且最靠前的一个数,可以用线段树。同时,题目要求不能连续放一样的数,也就是说最大值有时候放不进去,我们需要放严格次大值,因此同时维护严格次大值,线段树可以实现。最后,在每次放入后要把放入的删掉。

tip:这道题的修改和维护其实挺简单的,但是注意两点

1:注意次大值的一些特殊情况

2:别忘了位置要最靠前

AC代码:

#include<bits/stdc++.h>
using namespace std;
int n,m;
int a[1000005],ans[1000005];
bool f[1000005];
struct node
{int id,val;node(){val=id=0;}
}Zero;
bool operator<(node x,node y)
{if(x.val==y.val)return x.id>y.id;return x.val<y.val;
}y
bool operator==(node x,node y)
{return x.val==y.val;
}
struct Tree
{int l,r;node mx1,mx2;Tree(){mx1.val=mx1.id=0,mx2.val=mx2.id=0;}
}t[1000005];
node a1[4];
inline void Pushup(int i)
{a1[0]=t[i*2].mx1,a1[1]=t[i*2].mx2,a1[2]=t[i*2+1].mx1,a1[3]=t[i*2+1].mx2;sort(a1,a1+4);t[i].mx1=a1[3];for(int j=2;j>=0;j--)if(a1[j].val!=a1[j+1].val){t[i].mx2=a1[j];break;}
}
inline void Build(int i,int l,int r)
{if(l==r){t[i].mx1.val=a[l],t[i].mx1.id=l,t[i].l=l,t[i].r=r;return;}int mid=(l+r)/2;Build(i*2,l,mid);Build(i*2+1,mid+1,r);Pushup(i);
}
node m1,m2;
inline node Max1(int i,int l,int r,int l1,int r1)
{if(l1<=l&&r1>=r){a1[0]=t[i].mx1,a1[1]=t[i].mx2,a1[2]=m1,a1[3]=m2;sort(a1,a1+4);m1=a1[3];for(int j=2;j>=0;j--)if(a1[j].val!=a1[j+1].val){m2=a1[j];break;}return t[i].mx1;}if(l1>r||r1<l)return Zero;int mid=(l+r)/2;return max(Max1(i*2,l,mid,l1,r1),Max1(i*2+1,mid+1,r,l1,r1));
}
inline void Modify(int i,int l,int r,int x1)
{if(l>x1||r<x1)return;if(l==x1&&r==x1){t[i].mx1=t[i].mx2=Zero;return;}int mid=(l+r)/2;Modify(i*2,l,mid,x1);Modify(i*2+1,mid+1,r,x1);Pushup(i);
}
int main()
{scanf("%d%d",&n,&m);for(int i=1;i<=n;i++)scanf("%d",&a[i]);Build(1,1,n);for(int i=1;i<=n;i++){m1=m2=Zero;Max1(1,1,n,i,min(i+m,n));if(m1.val==ans[i-1]){if(m2.val){ans[i]=m2.val;Modify(1,1,n,m2.id);m-=m2.id-i;}}if(m1.val!=ans[i-1]&&m1.val){ans[i]=m1.val;Modify(1,1,n,m1.id);m-=m1.id-i;}}for(int i=1;i<=n;i++)printf("%d ",ans[i]?ans[i]:-1);return 0;
}

这篇关于2024第十五届蓝桥杯C++大学A组压轴题解:封印宝石的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/992388

相关文章

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决