为什么这题还不错却没人写题解啊。
考虑序列不降时有什么性质:ai−ai−1≥0,i∈(1,n]
考虑操作是区间同时加减,想到差分。
记 d1=a1,di=ai−ai−1,i∈(1,n],那么我们要让 di≥0,i∈[1,n]
每次操作即
那么对于 di>0 的,可以最多减 di 次,提供 di 次加的机会,对于 di<0 的,至少被加 |di| 次,需要 di 次被加。显然转化为二分图之后最小费用最大流。
但是假如只规定正权点向负权点提供流量是不够的,因为可能长度无法匹配导致我们对于 i,j,di>0,dj<0 需要 k 来先让 i,k 操作,再让 k,j 操作来实现。
所以我们要对于每个区间,无论左右是什么点,都连。
代码
#include <bits/stdc++.h>
using namespace std;
int rd() {
int f=1,sum=0; char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-1; ch=getchar();}
while(isdigit(ch)) {sum=(sum<<3)+(sum<<1)+ch-'0';ch=getchar();}
return sum*f;
}
#define N (int)(5e4+5)
#define inf (int)(2e9)
struct edge {
int nex,to,w,c;
}e[N<<1];
int hea[N],cnt=1;
int mic[2][N],bian[N];
int n,m,S,T,tot,a[N],b[N],ans;
void add_edge(int x,int y,int z,int c) {
e[++cnt].nex=hea[x]; e[cnt].to=y; e[cnt].w=z; e[cnt].c=c; hea[x]=cnt;
}
void add(int x,int y,int z,int c) {
add_edge(x,y,z,c); add_edge(y,x,0,-c);
}
deque<int>q;
bool vis[N];
int dis[N];
bool spfa() {
for(int i=0;i<N;i++) dis[i]=inf,vis[i]=0;
q.push_back(S); dis[S]=0;
while(!q.empty()) {
int x=q.front(); q.pop_front();
vis[x]=0;
for(int i=hea[x];i;i=e[i].nex) {
int y=e[i].to;
if(e[i].w&&dis[y]>dis[x]+e[i].c) {
dis[y]=dis[x]+e[i].c;
if(!vis[y]) {
vis[y]=1;
if(!q.empty()&&dis[y]<dis[q.front()]) q.push_front(y);
else q.push_back(y);
}
}
}
}
return dis[T]<inf;
}
int dfs(int x,int lim) {
if(x==T||!lim) return lim;
int flow=0,fl;
vis[x]=1;
for(int i=hea[x];i&&lim;i=e[i].nex) {
int y=e[i].to;
if(e[i].w&&!vis[y]&&dis[y]==dis[x]+e[i].c) {
fl=dfs(y,min(lim,e[i].w));
if(!fl) continue;
flow+=fl; lim-=fl; e[i].w-=fl; e[i^1].w+=fl;
ans+=e[i].c*fl;
}
}
if(!flow) dis[x]=inf;
vis[x]=0;
return flow;
}
void dinic() {
while(spfa()) {
memset(vis,0,sizeof(vis));
dfs(S,inf);
}
}
int main() {
n=rd(); m=rd(); S=0; T=n+2;
for(int i=1;i<=n;i++) b[i]=rd();
for(int i=1;i<=n;i++) a[i]=b[i]-b[i-1];
for(int i=0;i<=n+1;i++) mic[0][i]=mic[1][i]=inf;
char op; int x,y;
for(int i=1;i<=m;i++) {
cin>>op; x=rd(); y=rd();
if(op=='+') op=0; else op=1;
mic[op][x]=min(mic[op][x],y);
}
add(S,n+1,inf,0);
for(int i=1;i<=n;i++) {
if(a[i]>0) {
add(S,i,a[i],0);
} else {
add(i,T,-a[i],0); bian[++tot]=cnt-1;
}
}
for(int i=1;i<=n;i++) {
for(int l=1,r=l+i;r<=n+1;l++,r=l+i) {
if(mic[0][i]<inf) add(r,l,inf,mic[0][i]);
if(mic[1][i]<inf) add(l,r,inf,mic[1][i]);
}
}
dinic();
for(int i=1;i<=tot;i++) {
if(e[bian[i]].w) {
printf("-1"); return 0;
}
}
printf("%d",ans);
return 0;
}
__EOF__
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】