专栏文章
题解:P13504 [OOI 2024] More Gifts
P13504题解参与者 1已保存评论 0
文章操作
快速查看文章及其快照的属性,并进行相关操作。
- 当前评论
- 0 条
- 当前快照
- 1 份
- 快照标识符
- @mimzwzbj
- 此快照首次捕获于
- 2025/12/01 18:17 3 个月前
- 此快照最后确认于
- 2025/12/01 18:17 3 个月前
OOI 超级无敌强强题。
不难发现,如果初始序列中的元素种类本身就只有 种,那么答案就是 。
如果原序列中有大于 中元素,显然一个人能拿走的长度一定小于 。
我们可以把两个序列 拼接起来,就可以计算一个人从某一种元素出发最多能拿多少长度。
注意到 ,值域很大,而本题中我们只需要关心元素之间是否相等,所以先将 离散化一下。
我们考虑维护 表示从第 个元素开始一个人最多能拿多少个礼物。
这个东西可以用滑动窗口维护。
计算出 后就可以模拟我们可爱的小学奥数《周期问题》 的解题过程计算答案,详细过程见代码。
进入循环节前最多走 步,走出循环节之后也最多走 步,因此这部分的时间复杂度是 的。
整个算法的瓶颈在于离散化,总体时间复杂度 ,常数有点大但是通过 没什么问题。
代码:
CPPvoid solve(){
cin>>n>>k>>t;
for(int i=1;i<=n;i++) cin>>a[i];
// 离散化 a
vector<ll> vec;
for(int i=1;i<=n;i++) vec.push_back(a[i]);
sort(vec.begin(),vec.end());
vec.erase(unique(vec.begin(),vec.end()),vec.end());
for(int i=1;i<=n;i++) a[i]=lower_bound(vec.begin(),vec.end(),a[i])-vec.begin()+1;
int m=vec.size(); // 种类
if(vec.size()<=t) return cout<<1<<'\n',void();
for(int i=1;i<=n;i++) a[n+i]=a[i];
for(int i=0;i<=m;i++) cnt[i]=0;
int cur=0,r=1;
for(int l=1;l<=n;l++){
while(r<=2*n){
if(cnt[a[r]]==0){
if(cur==t) break;
cur++;
}
cnt[a[r]]++,r++;
}
jump[l]=r-l;
cnt[a[l]]--;
if(cnt[a[l]]==0) cur--;
}
ll tot=n*k,cur_dis=0;
vector<pii> vis(n+1,{-1,-1});
ll ans=0,u=1;
while(cur_dis<tot){
if(vis[u].fi!=-1){
ll cyc_lenseg=ans-vis[u].fi,cyc_lendis=cur_dis-vis[u].se;
ll rem=tot-cur_dis,loops=rem/cyc_lendis;
if(loops>0){
ans+=loops*cyc_lenseg;
cur_dis+=loops*cyc_lendis;
fill(vis.begin(),vis.end(),make_pair(-1,-1));
continue;
}
}
vis[u]={ans,cur_dis};
ll step=jump[u];
cur_dis+=step;
ans++;
u=(u+step-1)%n+1;
}
return cout<<ans<<'\n',void();
}
相关推荐
评论
共 0 条评论,欢迎与作者交流。
正在加载评论...