暂无图片
暂无图片
暂无图片
暂无图片
暂无图片

【寒假每日一题】AcWing 4656. 技能升级(难到飞起)

原创 . 2023-06-21
181


目录
一、题目

二、解题报告

三、知识风暴

一、题目
1、原题链接

2、题目描述

3 6
10 5
9 2
8 1

47
二、解题报告
1、思路分析
原始思路:

1)每次贪心选择提升攻击力最高的进行升级,即可以将每次可升级的攻击力存到一个multiset中,然后选择最后M项作为M次升级的方法,对后M项求和即为最多提升的攻击力。

2)该方法时间复杂度为O(n^2),所以必超时(十个测试数据过了四个),但并没有想出优化方法。

标准思路:

1)原始思路中因为我们每次都选择的是提升攻击力最多的,我们可以假设每个技能每次所能增加的攻击力是一个递减的等差数列,而我们每次都是选择最大的,也就是从这N个等差序列的头部进行选择,所以我们每次都是能够保证是从最初的技能开始一步步升级的,不会越级升级,因为针对某个技能在把它升到下一级之前,它一定是已经达到了下一级的前一个级别,因为我们每次都选择攻击力提升幅度最大的技能进行升级,而针对某个技能,它的攻击力提升是逐渐下降的,所以我们按照原贪心思路得到的解一定是正确的,下面我们需要对这个算法实现方法进行优化。

2)针对原始思路,我们需要求升序序列前M项的总和,我们直接暴力求肯定超时,我们可以在该 序列中找到某个数x前面正好存在M项,而这个x在该序列中可能存在多个,我们经过分析可知,这个x必定满足:在该序列中大于等于x的数的个数一定大于等于M个,而大于等于x+1的数的个数一定小于M个,而我们可以通过二分查找来查找x。

3)在找到x后,我们可以通过分析,将每个技能的每次升级的攻击力序列视为N个等差序列,根据等差序列的性质我们可知,假设其中某个等差序列首项为a0,公差为-d,则该序列中大于等于x的数的个数为𠃊(a-x)/b 𠃎+1个((a-x)/b下取整+1个),所以我们之后也可以根据等差数列求和公式来计算每个序列大于等于x的数的代数和,最后用这个和减去和x相等的数的总和即为所求结果。

4)模拟上述过程,输出结果,即为所求。

2、时间复杂度
原始思路时间复杂度为O(n^2)

标准思路时间复杂度O(nlogn)

3、代码详解 
原始思路代码:

#include
#include
using namespace std;
int A[1000100],B[1000100];
int main() {
multisets;
int N,M;
cin>>N>>M;
for(int i=0;i<N;i++){
cin>>A[i]>>B[i];
for(int j=0;j<A[i]/B[i]+1;j++){
int k=j;
s.insert(A[i]-k*B[i]);
}
}
int sl=s.size();
int num=0;
auto it=s.end();
it–;
int sum=0;
while(num<M){
sum+=*it;
it–;
num++;
}
cout<<sum;
return 0;
}
标准思路代码:

#include
using namespace std;
int A[100010],B[100010];
int N,M;
//判断大于等于mid的数的个数是否有M个
bool check(int mid){
long long res=0;
for(int i=0;i<N;i++){
if(A[i]>=mid){
res+=(A[i]-mid)/B[i]+1;
}
}
return res>=M;
}
int main()
{
cin>>N>>M;
for(int i=0;i<N;i++){
cin>>A[i]>>B[i];
}
int l=0,r=1e6;//二分从0开始,因为0可能为第M个数
while(l<r){
int mid=l+r+1>>1;//因为下一步l=mid,所以要上取整
if(check(mid)) l=mid; //如果大于等于mid的数的个数有大于等于M个,则答案在[mid,1e6]
else r=mid-1;
}
long long res=0,cnt=0;//cnt代表总项数
for(int i=0;i<N;i++){
if(A[i]>=r){
int c=(A[i]-r)/B[i]+1;//每个序列中大于r(x)的项数
int end=A[i]-(c-1)*B[i];
cnt+=c;
res+=(long long)(A[i]+end)*c/2;//每个等差数列中大于r(x)的总和
}
}
cout<<res-(cnt-M)*r;
return 0;
}
三、知识风暴
1、贪心算法

2、归并算法

3、二分查找法

「喜欢这篇文章,您的关注和赞赏是给作者最好的鼓励」
关注作者
【版权声明】本文为墨天轮用户原创内容,转载时必须标注文章的来源(墨天轮),文章链接,文章作者等基本信息,否则作者和墨天轮有权追究责任。如果您发现墨天轮中有涉嫌抄袭或者侵权的内容,欢迎发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

评论