「SNOI2019」通信
还好我没生在陕西啊
首先发现这个题不能\(dp\),数据范围不大,好像一种网络流的样子啊
哎等等,这样向后面连边不是一个\(DAG\)吗,这不是最小权路径覆盖的板子吗
于是我们套路的拆点,对于一个点\(i\)我们拆成\(i\)和\(i'\),源点向点\(i\)连费用为\(0\)容量为\(1\)的边,\(i'\)向汇点连费用为\(0\)容量为\(1\)的边
之后我们把让\(S\)向\(i'\)直接连费用为\(W\)容量为\(1\)的边,表示直接连到控制中心
对于点\(i\),我们向\(j'(j>i)\)连容量为\(1\)费用为\(|a_i-a_j|\)的边,表示点\(j\)接到了点\(i\)后面
于是现在我们一个最小费用最大流就好了
之后喜提\(TLE\)
我们发现我们好像建出了\(n^2\)级别的边
我们考虑我们连边的方式,这样向后面一个区间连边的方式让人好熟悉啊,这不是线段树优化建图吗
但是我们注意到这里的边的费用好像不是一样的,但是这个绝对值给了我们分类讨论的可能
对于一个点\(i\),我们建出一个虚点\(x'\),\(i\)向\(x'\)连费用为\(a_i\)的边,让这个虚点\(x'\)向\(j(j>i,a_j<a_i)\)连费用为\(-a_j\)的边,\(a_j>a_i\)同理,这样我们就能实现绝对值这个问题了
那么我们如何快速向一个点之后的所有大于它或小于它的点连边呢
直接主席树优化建图就好了,这样我们的点数和边数都是\(nlogn\)级别的了
但是还是\(T\),发现我们建了两万多个点和十万多条边
写一个快一点的费用流就好了,用zkw费用流和\(slf\)优化的spfa勉勉强强能跑过的样子
另外\(loj\)上的大爷们写的看起来好像\(cdq\)分治优化建图的洞西看起来好神仙啊
代码
#include<queue>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define re register
#define LL long long
#pragma GCC optimize(3)
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
inline int read() {
char c=getchar();int x=0;while(c<'0'||x>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const LL inf=999999999999;
const int fnf=999999999;
const int maxn=2e3+5;
const int M=4e5+6;
int head[M],vis[M],a[maxn],c[maxn],b[maxn];LL d[M];
int rt[maxn][2],l[M],r[M];
struct E{int v,nxt,w,f;}e[200005];
int n,num=1,S,T,W,sz,cnt;
inline void C(int x,int y,int w,int f) {
e[++num].v=y;e[num].nxt=head[x];
e[num].w=w;e[num].f=f;head[x]=num;
}
std::deque<int> q;
inline void add(int x,int y,int w,int f) {C(x,y,w,f),C(y,x,-1*w,0);}
inline int SPFA() {
for(re int i=S;i<=cnt;i++) d[i]=inf,vis[i]=0;
d[T]=0,q.push_back(T);
while(!q.empty()) {
int k=q.front();q.pop_front();vis[k]=0;
for(re int i=head[k];i;i=e[i].nxt)
if(e[i^1].f&&d[e[i].v]>d[k]+e[i^1].w) {
d[e[i].v]=d[k]+e[i^1].w;
if(!vis[e[i].v]) {
vis[e[i].v]=1;
if(q.empty()) {q.push_back(e[i].v);continue;}
if(d[e[i].v]<d[q.front()]) q.push_front(e[i].v);
else q.push_back(e[i].v);
}
}
}
return d[S]<inf;
}
int dfs(int x,int now) {
if(x==T||!now) return now;
int flow=0,ff;vis[x]=1;
for(re int i=head[x];i;i=e[i].nxt)
if(e[i].f&&!vis[e[i].v]&&d[e[i].v]==d[x]+e[i^1].w) {
ff=dfs(e[i].v,min(e[i].f,now));
if(ff<=0) continue;
flow+=ff,now-=ff,e[i].f-=ff,e[i^1].f+=ff;
if(!now) break;
}
return flow;
}
inline int find(int x) {
int lx=1,ry=sz;
while(lx<=ry) {
int mid=lx+ry>>1;
if(c[mid]==x) return mid;
if(c[mid]<x) lx=mid+1;
else ry=mid-1;
}
return 0;
}
int ins(int pre,int x,int y,int pos,int a,int b) {
int root=++cnt;
if(x==y) {
add(root,a,b*c[x],1);
return root;
}
int mid=x+y>>1;
l[root]=l[pre],r[root]=r[pre];
if(pos<=mid) l[root]=ins(l[pre],x,mid,pos,a,b);
else r[root]=ins(r[pre],mid+1,y,pos,a,b);
if(l[root]) add(root,l[root],0,fnf);
if(r[root]) add(root,r[root],0,fnf);
return root;
}
void Con(int g,int now,int x,int y,int lx,int ry) {
if(!now) return;
if(lx<=x&&ry>=y) {add(g,now,0,fnf);return;}
int mid=x+y>>1;
if(lx<=mid) Con(g,l[now],x,mid,lx,ry);
if(ry>mid) Con(g,r[now],mid+1,y,lx,ry);
}
int main() {
n=read(),W=read();S=0,T=n+n+1;cnt=T;
for(re int i=1;i<=n;i++) a[i]=read();
for(re int i=1;i<=n;i++) add(S,i+n,W,1);
for(re int i=1;i<=n;i++) add(S,i,0,1);
for(re int i=1;i<=n;i++) add(i+n,T,0,1);
for(re int i=1;i<=n;i++) c[i]=a[i];
std::sort(c+1,c+n+1);sz=std::unique(c+1,c+n+1)-c-1;
for(re int i=1;i<=n;i++) b[i]=a[i],a[i]=find(a[i]);
for(re int i=n;i>1;--i)
rt[i][0]=ins(rt[i+1][0],1,sz,a[i],i+n,-1),
rt[i][1]=ins(rt[i+1][1],1,sz,a[i],i+n,1);
for(re int i=1;i<n;i++) {
++cnt;
add(i,cnt,b[i],1);
Con(cnt,rt[i+1][0],1,sz,1,a[i]);
++cnt;
add(i,cnt,-1*b[i],1);
Con(cnt,rt[i+1][1],1,sz,a[i],sz);
}
LL ans=0;
while(SPFA()) {
vis[T]=1;
while(vis[T]) {
for(re int i=S;i<=cnt;i++) vis[i]=0;
ans+=1ll*dfs(S,fnf)*d[S];
}
}
printf("%lld\n",ans);
return 0;
}
「SNOI2019」通信的更多相关文章
- 【LOJ】#3097. 「SNOI2019」通信
LOJ#3097. 「SNOI2019」通信 费用流,有点玄妙 显然按照最小路径覆盖那题的建图思路,把一个点拆成两种点,一种是从这个点出去,标成\(x_{i}\),一种是输入到这个点,使得两条路径合成 ...
- 「SNOI2019」通信 分治优化费用流建图
题意: n 个排成一列的哨站要进行通信.第 i 个哨站的频段为 ai. 每个哨站 ii 需要选择以下二者之一: 1.直接连接到控制中心,代价为 W:2.连接到前面的某个哨站 j(j<i),代价为 ...
- 「SNOI2019」通信 分治建图
根据题意 每个点可以直接与S,T相连 也可以和前面的哨站相连 暴力建边的话 有n2条边 要用分治优化建边: 类似于归并排序 先对每一层分为左半边与右半边 对每一半都拿出来先排序去重后 直接排成一条链建 ...
- Loj #3096. 「SNOI2019」数论
Loj #3096. 「SNOI2019」数论 题目描述 给出正整数 \(P, Q, T\),大小为 \(n\) 的整数集 \(A\) 和大小为 \(m\) 的整数集 \(B\),请你求出: \[ \ ...
- 【LOJ】#3098. 「SNOI2019」纸牌
LOJ#3098. 「SNOI2019」纸牌 显然选三个以上的连续牌可以把他们拆分成三个三张相等的 于是可以压\((j,k)\)为有\(j\)个连续两个的,有\(k\)个连续一个的 如果当前有\(i\ ...
- 【LOJ】#3096. 「SNOI2019」数论
LOJ#3096. 「SNOI2019」数论 如果\(P > Q\)我们把\(P\)和\(Q\)换一下,现在默认\(P < Q\) 这个时候每个合法的\(a_i\)都可以直接落到\(Q\) ...
- 【LOJ】#3095. 「SNOI2019」字符串
LOJ#3095. 「SNOI2019」字符串 如果两个串\(i,j\)比较\(i < j\),如果离\(a_{i}\)最近的不同的数是\(a_{k}\),如果\(j < k\)那么\(i ...
- 「SNOI2019」字符串
题目 看起来非常一眼啊,我们完全可以\(std::sort\)来解决这歌问题 于是现在的问题转化成了比较函数怎么写 随便画一下就会发现前面的好几位是一样的,后面的好几位也是一样,只需要比较中间的一段子 ...
- 「SNOI2019」积木
传送门 Description 有一块\(n\)行\(m\)列的网格板, \(n,m\)都是奇数.网格上平铺着一些\(1*2\)的积木.积木可以旋转,不能重叠.网格板上只有一格的空位. 你可以做两种操 ...
随机推荐
- JavaWeb学习 (二十四)————Filter(过滤器)常见应用
一.统一全站字符编码 通过配置参数charset指明使用何种字符编码,以处理Html Form请求参数的中文问题 1 package me.gacl.web.filter; 2 3 import ja ...
- .Net Core Cors中间件解析
一.同源策略和资源跨域共享 1.同源策略 同源策略,它是由Netscape提出的一个著名的安全策略.现在所有支持JavaScript 的浏览器都会使用这个策略.所谓同源是指,域名,协议,端口相同. 1 ...
- 【Java深入研究】5、Proxy动态代理机制详解
在学习Spring的时候,我们知道Spring主要有两大思想,一个是IoC,另一个就是AOP,对于IoC,依赖注入就不用多说了,而对于Spring的核心AOP来说,我们不但要知道怎么通过AOP来满足的 ...
- c++类构造函数详解
//一. 构造函数是干什么的 /* 类对象被创建时,编译系统对象分配内存空间,并自动调用该构造函数->由构造函数完成成员的初始化工作 eg: Counter c1; 编译 ...
- 06:合法 C 标识符
06:合法 C 标识符 查看 提交 统计 提问 总时间限制: 1000ms 内存限制: 65536kB 描述 . C语言标识符要求: 1. 非保留字: 2. 只包含字母.数字及下划线(“_”). 3. ...
- React中使用百度地图API
今天我们来搞一搞如何在React中使用百度地图API好吧,最近忙的头皮发麻,感觉身体被掏空,所以很久都没来写博客了,但今天我一定要来一篇好吧 话不多说,我们直接开始好吧 特别注意:该React项目是用 ...
- 原生js实现二级联动下拉列表菜单
二级联动下拉列表菜单的难点在于对后台返回的数据进行解析,不多逼逼,直接上代码 上图是后台返回的数据 实现代码如下: var deviceNotExist = true;//防止数据重复 if(data ...
- JavaScript与正则表达式
正则表达式的定义 正则表达式与字符串对象相关的方法 相关示例 一.正则表达式(regular expression简称res) 1.定义: 一个正则表达式就是由普通字符以及特殊字符(称为元字符)组成 ...
- 聊聊setTimeout和setInterval线程
在聊setTimeout和setInterval这两个事件的前,先聊另外一个与之密切关联的知识点,那就是线程(thread).而线程有常常跟另外一个词语--“进程”一起出现.那么何为线程?何为线程呢? ...
- [Linux.NET]在CentOS 7.x中编译方式安装Nginx
Nginx是一款轻量级的Web 服务器/反向代理服务器及电子邮件(IMAP/POP3)代理服务器,并在一个BSD-like 协议下发行.由俄罗斯的程序设计师Igor Sysoev所开发,供俄罗斯大型的 ...