NKOJ卡常卡不过QAQ

description

给两个A,B序列,让你分别在A,B中各选k个数,其中至少有L对下标相等。

Solution

把问题转化为至多选n-K对下标不同的对。

配对问题就用费用流……

同坐标A,B两两连边。

然后有一个虚拟点,所有点A连向它,它又连向所有点B。代表不同下标的点配对。这个点还要拆成两个虚拟点(中间容量为n-K)。

接着模拟费用流

(注意这次的模拟费用流的反悔功能是利用贪心实现的)

过程中记录\(flow\)为剩下的流量(还能选不同下标的对数)

1.直接选两个下标相等的点\(a_i\)和\(b_i\)

2.直接选两个下标不等的点\(a_i\)和\(b_j\) flow--

令\(a'_i\)表示\(a_i\)没选\(b_i\)选了,同理\(b'_i\)表示\(b_i\)没选\(a_i\)选了(总之就是相对应下标选了)

3.选\(a'_i\)和\(b_j\),配对后flow--,但是发现可以做到\(a'_i\)配\(b_i\),然后\(b_j\)替补原来\(b_i\)配对的点。flow++(相当于flow不变化)

4.(3的a,b颠倒)选\(b'_i\)和\(a_j\),同3

5.选\(a'_i\)和\(b'_j\),断掉\(b_i\)和\(a_j\)的联系,发现i,j都可以分别满足a,b配对。flow++

注意在优先选值最大的以外,优先选5,最后选2(flow相当于第二关键字)

过程用5个堆维护就好了,分别维护\(a_i\) , \(b_i\) , \(a_i+b_i\) , \(a'_i\) , \(b'_i\)

中间需要删除用标记就好。

然后\(a'_i\)和\(b'_i\)都是对立点标记后,才加进去的。

我还想过一个贪心优化,就是a,b先从大到小排序,然后分别找前k大,看看是否存在下标相同的a,b点对,存在就直接取了,不过没什么用,因为排序费时。

最后为什么我跑的这么慢,呜呜……

code

点击查看代码
#include<bits/stdc++.h>
using namespace std;
const int N=1e6+5;
typedef long long ll;
char buf[1<<23],*p1=buf,*p2=buf,obuf[1<<23],*O=obuf;
#define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
inline int read() {
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)){if(ch=='-') f=-1;ch=getchar();}
while(isdigit(ch)) x=x*10+(ch^48),ch=getchar();
return x*f;
}
void print(ll x) {
if(x>9) print(x/10);
*O++=x%10+'0';
}
int n,K,L,a[N],b[N],flow;
ll inf=1e18;
bool ua[N],ub[N];
struct node {
int p;ll w;
bool operator<(const node &u) const{return w<u.w;}
};
struct nd {
int p;ll w;
bool operator<(const nd &u) const{return w>u.w;}
}A[N],B[N];
priority_queue<node> Q1,Q2,Q3,Q4,Q5;
void _clear() {
while(Q1.size()>1&&ua[Q1.top().p]) Q1.pop();
while(Q2.size()>1&&ub[Q2.top().p]) Q2.pop();
while(Q3.size()>1&&(ua[Q3.top().p]||ub[Q3.top().p]))Q3.pop();
}
void Clear() {
while(Q1.size()>1) Q1.pop();
while(Q2.size()>1) Q2.pop();
while(Q3.size()>1) Q3.pop();
while(Q4.size()>1) Q4.pop();
while(Q5.size()>1) Q5.pop();
for(int i=1;i<=n;i++) ua[i]=ub[i]=0;
}
#define u1 Q1.top().w
#define u2 Q2.top().w
#define u3 Q3.top().w
#define u4 Q4.top().w
#define u5 Q5.top().w
ll sum;
bool flag=0;
//void init() { //没用的贪心优化
// for(int i=1;i<=n;i++) A[i]=(nd){i,a[i]},B[i]=(nd){i,b[i]};
// sort(A+1,A+1+n);
// sort(B+1,B+1+n);
// for(int i=1;i<=L;i++) ua[A[i].p]=ub[B[i].p]=1;
// for(int i=1;i<=n&&L;i++) {
// if(ua[i]&&ub[i]) {sum+=a[i]+b[i];L--;K--;}
// else {
// ua[i]=ub[i]=0;Q1.push((node){i,a[i]});Q2.push((node){i,b[i]});Q3.push((node){i,a[i]+b[i]});
// }
// }
// if(!L) {
// for(int i=1,j=1;i<=n&&j<=K;i++) if(!ua[A[i].p])sum+=A[i].w;
// for(int i=1,j=1;i<=n&&j<=K;i++) if(!ub[B[i].p])sum+=B[i].w;
// flag=1;print(sum);*O++='\n';
// }
//}
void solve() {
for(int i=1;i<=n;i++) {Q1.push((node){i,a[i]});Q2.push((node){i,b[i]});Q3.push((node){i,a[i]+b[i]});}
int flow=L<=0?1e9:K-L;
for(int i=1;i<=K;i++) {
_clear();
ll w1=flow?u1+u2:-inf;
ll w2=u3;
ll w3=u4+u2;
ll w4=u5+u1;
ll w5=u4+u5;
ll mx=max(max(w1,w2),max(w3,w4));
if(w5>=mx) {flow++;sum+=w5;ua[Q4.top().p]=ub[Q5.top().p]=1;Q4.pop();Q5.pop();}
else if(w4==mx) {
sum+=w4;int x=Q1.top().p;ua[x]=ub[Q5.top().p]=1;
if(!ub[x])Q5.push((node){x,b[x]});
Q1.pop();Q5.pop();
}
else if(w3==mx) {
sum+=w3;int y=Q2.top().p;ua[Q4.top().p]=ub[y]=1;
if(!ua[y])Q4.push((node){y,a[y]});
Q4.pop();Q2.pop();
}
else if(w2==mx) {
sum+=w2;ua[Q3.top().p]=ub[Q3.top().p]=1;
Q3.pop();
}
else {
int x=Q1.top().p,y=Q2.top().p;
sum+=w1;flow--;ua[x]=ub[y]=1;
if(!ub[x]) Q5.push((node){x,b[x]});
if(!ua[y]) Q4.push((node){y,a[y]});
Q1.pop();Q2.pop();
}
}
print(sum);*O++='\n';
}
int main() {
Q1.push((node){-1,-inf}),Q2.push((node){-1,-inf}),Q3.push((node){-1,-inf}),Q4.push((node){-1,-inf}),Q5.push((node){-1,-inf});
int T=read();
while(T--) {
n=read(),K=read(),L=read();
for(int i=1;i<=n;i++) {a[i]=read();}
for(int i=1;i<=n;i++) {b[i]=read();}
sum=flag=0;
// if(L>=n/2)init();
if(!flag)solve();
Clear();
}
fwrite(obuf,O-obuf,1,stdout);
return 0;
}

「NOI2019」序列的更多相关文章

  1. LOJ 3158: 「NOI2019」序列

    题目传送门:LOJ #3158. 题意简述: 给定两个长度为 \(n\) 的正整数序列 \(a,b\),要求在每个序列中都选中 \(K\) 个下标,并且要保证同时在两个序列中都被选中的下标至少有 \( ...

  2. loj3161「NOI2019」I 君的探险(随机化,整体二分)

    loj3161「NOI2019」I 君的探险(随机化,整体二分) loj Luogu 题解时间 对于 $ N \le 500 $ 的点,毫无疑问可以直接 $ O(n^2) $ 暴力询问解决. 考虑看起 ...

  3. Loj #3059. 「HNOI2019」序列

    Loj #3059. 「HNOI2019」序列 给定一个长度为 \(n\) 的序列 \(A_1, \ldots , A_n\),以及 \(m\) 个操作,每个操作将一个 \(A_i\) 修改为 \(k ...

  4. 「HNOI2016」序列 解题报告

    「HNOI2016」序列 有一些高妙的做法,懒得看 考虑莫队,考虑莫队咋移动区间 然后你在区间内部找一个最小值的位置,假设现在从右边加 最小值左边区间显然可以\(O(1)\),最小值右边的区间是断掉的 ...

  5. AC日记——「SDOI2017」序列计数 LibreOJ 2002

    「SDOI2017」序列计数 思路: 矩阵快速幂: 代码: #include <bits/stdc++.h> using namespace std; #define mod 201704 ...

  6. loj #2051. 「HNOI2016」序列

    #2051. 「HNOI2016」序列 题目描述 给定长度为 n nn 的序列:a1,a2,⋯,an a_1, a_2, \cdots , a_na​1​​,a​2​​,⋯,a​n​​,记为 a[1: ...

  7. 「JSOI2014」序列维护

    「JSOI2014」序列维护 传送门 其实这题就是luogu的模板线段树2,之所以要发题解就是因为学到了一种比较NB的 \(\text{update}\) 的方式.(参见这题) 我们可以把修改操作统一 ...

  8. 「BZOJ1251」序列终结者 (splay 区间操作)

    题面: 1251: 序列终结者 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 5367  Solved: 2323[Submit][Status][D ...

  9. 「LuoguP1430」 序列取数(区间dp

    题目描述 给定一个长为n的整数序列(n<=1000),由A和B轮流取数(A先取).每个人可从序列的左端或右端取若干个数(至少一个),但不能两端都取.所有数都被取走后,两人分别统计所取数的和作为各 ...

随机推荐

  1. java中throws子句是怎么用的?工作原理是什么

    7.throws子句 马克-to-win:当你的方法里抛出了checked异常,如你不catch,代表你当时不处理(不想处理或没条件处理),但你必须得通过"throws那个异常"告 ...

  2. Jquery中each的3种遍历方式

    学习目标: 参考博文: https://blog.csdn.net/honey_th/article/details/7404273 一.Jquery中each的几种遍历方法 1. 选择器+遍历 &l ...

  3. Java自定义异常类的简单实现

    学习目标: 掌握自定义异常类 例题: 需求:自定义异常类,简单判断是否注册成功 代码如下: RegisterException类: /** * @author YanYang * @projectNa ...

  4. centos6的yum源更新版本

    概述 centos6系统从2020年12月1号开始不再维护,官方的yum源不再可用,同时国内的阿里云镜像和163镜像也都不再可用. 但是我们有一些老的服务器仍然在使用centos6系统版本,依赖库的安 ...

  5. python函数基础算法简介

    一.多层语法糖本质 """ 语法糖会将紧挨着的被装饰对象名字当参数自动传入装饰器函数中""" def outter(func_name): ...

  6. 初识ES6(JavaScript)

    初识ES6(JavaScript) 关于ES6: ES6即ECMAScript6,是一种规范,JavaScript遵循了这种规范. *优点:*代码比较简洁. *缺点:*浏览器的兼容性不好. 1.变量和 ...

  7. 百兆以太网(100BASE-TX)的波形和眼图

    沾了公司的光用了那台采样率吓死人的示波器看了下百兆以太网的三电平波形和眼图. 之前我也强调过百兆的三电平是不能从1状态越过0状态跳到-1状态的,从眼图上能明显看出来. 可以看出这个信号还是不错的.甚至 ...

  8. Go xmas2020 学习笔记 12、Structs, Struct tags & JSON

    12-Structs, Struct tags & JSON. Struct. Struct Gotcha. Anonymous Struct Type. Make the zero valu ...

  9. Java核心知识1:泛型机制详解

    1 理解泛型的本质 JDK 1.5开始引入Java泛型(generics)这个特性,该特性提供了编译时类型安全检测机制,允许程序员在编译时检测到非法的类型. 泛型的本质是参数化类型,即给类型指定一个参 ...

  10. OpenHarmony 3.1 Beta版本关键特性解析——探秘隐式查询

    ​(以下内容来自开发者分享,不代表 OpenHarmony 项目群工作委员会观点)​ 徐浩 隐式查询是 OpenAtom OpenHarmony(以下简称"OpenHarmony" ...