当前位置: 首页 > news >正文

2023NOIP A层联测28-大眼鸹猫

给你两个长度为 n n n 的序列 a , b a,b a,b,这两个序列都是单调不降的。

你可以对 a a a 进行不超过 m m m 次操作,每次操作你可以选择一个 i i i 满足 1 ≤ i ≤ n 1\le i\le n 1in,然后选择一个整数(可以是负数) x x x,将 a i a_i ai 加上 x x x,这一次操作需要花费 x 2 x^2 x2 的代价。

在做操作的过程中,你需要保证 a a a 始终单调不降。

最后,你需要将 a a a 序列变成 b b b 序列,即对任意 i i i 满足 1 ≤ i ≤ n 1\le i\le n 1in,都有 a i = b i a_i=b_i ai=bi

求最小需要花费的总代价之和。

n , m ≤ 1 0 5 n,m\le10^5 n,m105


首先如果 m m m 小于 a i ≠ b i a_i\not=b_i ai=bi 的个数,就是无解了。

操作过程中,要保证 a a a 始终单调不降,看上去很难满足,但结论是一定存在一种合法操作方案。可以考虑一个极大的区间 [ l , r ] [l,r] [l,r] ∀ i ∈ [ l , r ] , a i < b i \forall i\in[l,r],a_i< b_i i[l,r],ai<bi,此时就从 r r r 操作到 l l l,若 ∀ i ∈ [ l , r ] , a i > b i \forall i\in[l,r],a_i>b_i i[l,r],ai>bi,就从 l l l 操作到 r r r

所以单独考虑每一个数 ∣ a i − b i ∣ |a_i-b_i| aibi,分配了 x i x_i xi 次操作,显然每次操作将 ∣ a i − b i ∣ |a_i-b_i| aibi 减少 ∣ a i − b i ∣ x i \dfrac{|a_i-b_i|}{x_i} xiaibi 是代价最小的。设 f ( x ) f(x) f(x) 表示给 ∣ a i − b i ∣ |a_i-b_i| aibi 分配了 x x x 次操作的最小代价,发现 f ( x ) f(x) f(x) 是下凸函数,意思是随着 x x x 的增大, f ( x ) f(x) f(x) 减小的幅度越来越小。

那么我们可以先给每个数分配一次操作,对于剩下的操作用大根堆维护每个数再分配一次操作减少的代价,每次就贪心的去选代价减少得最多的,然后给它分配一次操作再放进堆里。最后把堆里的数拿出来算答案即可。

时间复杂度 O ( m log ⁡ n ) O(m\log n) O(mlogn)

#include<bits/stdc++.h>
#define ll long long
using namespace std;
constexpr int N=1e5+1;
constexpr ll mod=998244353;
int n,m;
ll a[N],b[N],dis[N],ans;
inline ll getval(ll x,ll n)
{return (x/n)*(x/n)*(n-x%n)+(x/n+1)*(x/n+1)*(x%n);
}
struct node
{int num,id;bool operator<(const node &a)const{return getval(dis[id],num)-getval(dis[id],num+1)<getval(dis[a.id],a.num)-getval(dis[a.id],a.num+1);}
};
priority_queue<node> q;
int main()
{freopen("attend.in","r",stdin);freopen("attend.out","w",stdout);cin.tie(0)->sync_with_stdio(0);cin>>n>>m;for(int i=1;i<=n;i++) cin>>a[i];for(int i=1;i<=n;i++) cin>>b[i];int fl=0;for(int i=1;i<=n;i++) fl+=(a[i]!=b[i]);if(fl>m) cout<<"-1",exit(0);for(int i=1;i<=n;i++){if(a[i]==b[i]) continue;dis[i]=abs(a[i]-b[i]);q.push({1,i});}if(q.empty()) cout<<0,exit(0);m-=fl;while(m--){node now=q.top();q.pop();now.num++;q.push(now);}while(q.size()){ans=(ans+getval(dis[q.top().id],q.top().num))%mod;q.pop();}cout<<ans;
}
http://www.lryc.cn/news/224921.html

相关文章:

  • 电机应用-直流有刷电机
  • BIM、建筑机器人、隧道工程施工关键技术
  • 快速了解什么是跳跃表(skip list)
  • 【Node.js入门】1.1Node.js 简介
  • 数据库 高阶语句
  • jenkins Java heap space
  • OpenCV校准棋盘集合
  • 使用git将本地项目推送到远程仓库github
  • Mybatis-Plus使用Wrapper自定义SQL
  • 仿mudou库one thread one loop式并发服务器
  • 二十三种设计模式全面解析-组合模式与装饰器模式的结合:实现动态功能扩展
  • 智慧城市建设解决方案分享【完整】
  • unity - Blend Shape - 变形器 - 实践
  • asp.net core mvc之路由
  • 前端设计模式之【访问者模式】
  • 通过docker-compose部署elk日志系统,并使用springboot整合
  • 【NLP】特征提取: 广泛指南和 3 个操作教程 [Python、CNN、BERT]
  • 数据结构-单链表
  • 红队系列-IOT安全深入浅出
  • 亚数受邀参加“长三角G60科创走廊量子密码应用创新联盟(中心)”启动仪式
  • 直方图学习
  • Java / Android 多线程和 synchroized 锁
  • 基于51单片机的万年历-脉搏计仿真及源程序
  • 【ARFoundation学习笔记】点云与参考点
  • uni-app:js实现数组中的相关处理-数组复制
  • 8 STM32标准库函数 之 实时时钟(RTC)所有函数的介绍及使用
  • ARMday04(开发版简介、LED点灯)
  • 国际腾讯云:云服务器疑似被病毒入侵问题解决方案!!!
  • Perl语言用多线程爬取商品信息并做可视化处理
  • 认识计算机-JavaEE初阶