2023NOIP A层联测20 T3 点餐
2023NOIP A层联测20 点餐
题目很好,可惜考试没想到。
思路
可以按照
我们令
由于分治每一层最多跑
CODE
#include<bits/stdc++.h>
using namespace std;
#define inf 2e9
#define int long long
const int maxn=2e5+5;
struct node
{
int ls,rs,sz,sum;
}tree[maxn*50];
struct node1
{
int a,b;
}food[maxn];
int n,tot;
int ans[maxn],rt[maxn];
bool cmp(node1 a,node1 b){return a.b<b.b;}
void insert(int &p,int x,int l,int r)
{
tree[++tot]=tree[p];
p=tot;
if(l==r)
{
tree[p].sum+=x-1;
tree[p].sz++;
return ;
}
int mid=l+r>>1;
if(x<=mid) insert(tree[p].ls,x,l,mid);
else insert(tree[p].rs,x,mid+1,r);
tree[p].sum=tree[ tree[p].ls ].sum+tree[ tree[p].rs ].sum;
tree[p].sz=tree[ tree[p].ls ].sz+tree[ tree[p].rs ].sz;
}
int getsum(int p,int l,int r,int k)
{
if(l==r) return (l-1)*k;
int mid=l+r>>1;
if(tree[tree[p].ls].sz>=k) return getsum(tree[p].ls,l,mid,k);
else return getsum(tree[p].rs,mid+1,r,k-tree[tree[p].ls].sz)+tree[tree[p].ls].sum;
}
void solve(int l,int r,int lp,int rp)//[l,r] 的个数区间,对于 [lp,rp] 决策区间的点
{
if(r<l) return ;
int mid=l+r>>1,pos=0;
for(int i=max(lp,mid);i<=rp;i++)
{
int now=food[i].b+food[i].a+getsum(rt[i-1],1,inf,mid-1);//求 w(mid,i)
if(ans[mid]>=now)
{
ans[mid]=now;
pos=i;
}
}
solve(l,mid-1,lp,pos);//分析 f 的分布可以得出
solve(mid+1,r,pos,rp);
}
signed main()
{
scanf("%lld",&n);
for(int i=1;i<=n;i++) scanf("%lld%lld",&food[i].a,&food[i].b);
sort(food+1,food+n+1,cmp);
for(int i=1;i<=n;i++)
insert(rt[i]=rt[i-1],food[i].a+1,1,inf);
memset(ans,0x5f,sizeof(ans));
solve(1,n,1,n);
for(int i=1;i<=n;i++) printf("%lld\n",ans[i]);
}
分类:
多校联测
, 数据结构 / 线段树 / 主席树/可持久化线段树
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现