原题传送门

同步赛上我一开始想了个看似正确却漏洞百出的贪心:按\(a_i+b_i\)的和从大向小贪心

随便想想发现是假的,然后就写了个28pts的暴力dp

杜神后半程说这题就是个贪心,但我没时间写了 (实际是没想明白)

我们来说这道题的正解:

我们先珂以满足和最大,再满足并集大小大于等于\(l\)。所以我们先将\(a\)序列和\(b\)序列排序,取出两个序列的前\(k\)大

如果并集大小大于等于\(l\)就直接统计答案

否则我们要凑满\(l\)个都包含的,在凑的过程中动态更新答案

我们在两个序列中都选中前\(k-l\)个不含有并集的数,并加入答案中,易知这些数一定含在答案中(这个珂以在草稿纸上推一推)。接下来考虑如何凑并集元素,一次凑一组:

1.两个都没被选中的情况下的最大值,并将它们选中

2.一个被选中的情况下的最大值。假设\(a[rk[i]]\)选中了,但\(b[rk[i]]\)没选中,我们就要找到一个最小的j使得\(a[rk[j]]\)没被选中,我们就珂以选中\(b[rk[i]\)和\(a[rk[j]]\),使得a中不成对的还是\(k-l\)个。a,b反之亦然

我们求出1、2两种情况贡献最大值并更新选中状态(都用堆维护,具体细节见代码),重复\(l\)次即可求出答案

注意:两次2也许就会将两个都没选中的变成都选中,所以要及时舍掉不合法的

#include <bits/stdc++.h>
#define N 200005
#define ll long long
#define getchar nc
using namespace std;
inline char nc(){
static char buf[100000],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++;
}
inline int read()
{
register int x=0,f=1;register char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9')x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return x*f;
}
inline void write(register ll x)
{
if(!x)putchar('0');if(x<0)x=-x,putchar('-');
static int sta[20];register int tot=0;
while(x)sta[tot++]=x%10,x/=10;
while(tot)putchar(sta[--tot]+48);
}
int T,n,k,l,a[N],b[N],ar[N],br[N],va[N],vb[N],vis[N];
ll ans;
inline bool cmpa(register int x,register int y)
{
return a[x]>a[y];
}
inline bool cmpb(register int x,register int y)
{
return b[x]>b[y];
}
struct node{
int pos,val;
inline bool operator < (const node &it) const {
return val<it.val;
}
};
inline bool chkmax(register int &a,register int b)
{
return a<b?a=b,1:0;
}
int main()
{
T=read();
while(T--)
{
n=read(),k=read(),l=read();
for(register int i=1;i<=n;++i)
a[i]=read(),ar[i]=i,va[i]=vb[i]=vis[i]=0;
for(register int i=1;i<=n;++i)
b[i]=read(),br[i]=i;
sort(ar+1,ar+1+n,cmpa);
sort(br+1,br+1+n,cmpb);
for(register int i=1;i<=k;++i)
va[ar[i]]=1,vb[br[i]]=1;
ans=0;
priority_queue<int> pa,pb;
priority_queue<node> qa,qb;
for(register int i=1;i<=n;++i)
if(va[i]&&vb[i])
qb.push((node){i,a[i]+b[i]}),vis[i]=1;
if(qb.size()>=l)
{
for(register int i=1;i<=n;++i)
{
if(va[i])
ans+=a[i];
if(vb[i])
ans+=b[i];
}
}
else
{
for(register int i=1,tot=0;i<=k;++i)
if(!vb[ar[i]])
{
if(tot<k-l)
ans+=a[ar[i]],vis[ar[i]]=1,pa.push(b[ar[i]]);
else
qa.push((node){ar[i],a[ar[i]]+b[ar[i]]});
++tot;
}
for(register int i=1,tot=0;i<=n;++i)
if(!va[br[i]])
{
if(tot<k-l)
ans+=b[br[i]],vis[br[i]]=1,pb.push(a[br[i]]);
else
qb.push((node){br[i],a[br[i]]+b[br[i]]});
++tot;
}
int af=1,bf=1;
while(l--)
{
while(af<=k&&vis[ar[af]])
++af;
while(bf<=k&&vis[br[bf]])
++bf;
while(!qa.empty()&&vis[qa.top().pos]&&!(va[qa.top().pos]&&vb[qa.top().pos]))
qa.pop();
while(!qb.empty()&&vis[qb.top().pos]&&!(va[qb.top().pos]&&vb[qb.top().pos]))
qb.pop();
int maxx=0,typ=-1;
if(!qa.empty())
maxx=qa.top().val,typ=0;
if(!qb.empty()&&chkmax(maxx,qb.top().val))
typ=1;
if(!pa.empty()&&af<=k&&chkmax(maxx,pa.top()+a[ar[af]]))
typ=2;
if(!pb.empty()&&bf<=k&&chkmax(maxx,pb.top()+b[br[bf]]))
typ=3;
ans+=maxx;
if(typ==0)
vis[qa.top().pos]=1,qa.pop();
else if(typ==1)
vis[qb.top().pos]=1,qb.pop();
else if(typ==2)
vis[ar[af]]=1,pa.pop(),pa.push(b[ar[af]]);
else
vis[br[bf]]=1,pb.pop(),pb.push(a[br[bf]]);
}
}
write(ans),puts("");
}
return 0;
}

【题解】Luogu P5470 [NOI2019]序列的更多相关文章

  1. luogu P5470 [NOI2019]序列 dp 贪心 费用流 模拟费用流

    LINK:序列 考虑前20分 容易想到爆搜. 考虑dp 容易设\(f_{i,j,k,l}\)表示前i个位置 选了j对 且此时A选择了k个 B选择了l个的最大值.期望得分28. code //#incl ...

  2. Luogu P5470 [NOI2019]序列

    题目 可以直接贪心,但是用模拟费用流推的话会更轻松. 首先有一个显然的建图方式: \(S\)到\(0\)流量为\(k\),费用为\(0\). \(0\)到\(a_i\)流量为\(1\),费用为\(-a ...

  3. 洛谷 P5470 - [NOI2019] 序列(反悔贪心)

    洛谷题面传送门 好几天没写题解了,写篇题解意思一下(大雾 考虑反悔贪心,首先我们考虑取出 \(a,b\) 序列中最大的 \(k\) 个数,但这样并不一定满足交集 \(\ge L\) 的限制,因此我们需 ...

  4. [luogu P3648] [APIO2014]序列分割

    [luogu P3648] [APIO2014]序列分割 题目描述 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序 ...

  5. [题解] Luogu P5446 [THUPC2018]绿绿和串串

    [题解] Luogu P5446 [THUPC2018]绿绿和串串 ·题目大意 定义一个翻转操作\(f(S_n)\),表示对于一个字符串\(S_n\), 有\(f(S)= \{S_1,S_2,..., ...

  6. 【题解】Luogu P2572 [SCOI2010]序列操作

    原题传送门:P2572 [SCOI2010]序列操作 这题好弱智啊 裸的珂朵莉树 前置芝士:珂朵莉树 窝博客里对珂朵莉树的介绍 没什么好说的自己看看吧 操作1:把区间内所有数推平成0,珂朵莉树基本操作 ...

  7. 【题解】Luogu P5294 [HNOI2019]序列

    原题传送门 题意:给你一个长度为\(n\)的序列\(A\),每次询问修改一个元素(只对当前询问有效),然后让你找到一个不下降序列\(B\),使得这两个序列相应位置之差的平方和最小,并输出这个最小平方和 ...

  8. 【题解】Luogu P5471 [NOI2019]弹跳

    原题传送门 先考虑部分分做法: subtask1: 暴力\(O(nm)\)枚举,跑最短路 subtask2: 吧一行的点压到vector中并排序,二分查找每一个弹跳装置珂以到达的城市,跑最短路 sub ...

  9. 【题解】Luogu P5468 [NOI2019]回家路线

    原题传送门 前置芝士:斜率优化 不会的可以去杜神博客学 这道题我考场上只会拆点跑最短路的70pts做法 后来回家后发现错误的爆搜都能拿满分(刀片) 还有很多人\(O(mt)\)过的,还是要坚持写正解好 ...

随机推荐

  1. .NET Core 下调用WebAPI

    前言 今天我们介绍多种客户端调用WebApi的方式,可以是原生写的,也可以借助.NET 框架下的其他HTTP库.我们一起来看看它们之间的一些异同吧- RestSharp 首先要介绍的就是这款REST ...

  2. 华为企业级AS111-S,比较垃圾的地方

    今天换了一个华为企业级AS111-S 路由器,比较垃圾的地方: 1. 网页管理界面是https,却用一个无效的证书,chrome直接不能访问,IE可以访问,但第一次登陆改密码的时候就出错了. 然后怎么 ...

  3. Mac Brew 安装及配置

    mac 终端下,执行以下命令,即可安装brew: /usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homeb ...

  4. spoon数据转换中文乱码(kettle)

    (1) 查看mysql数据库是否为utf8(status) (2) 设置spoon (3) 文本打开spoon.bat,找到set OPT=%OPT% %PENTAHO_DI_JAVA_OPTIONS ...

  5. pod install/update失败:Failed to connect to 127.0.0.1 port 1080: Connection refused

    出现这类错误,通常是因为代理发生的,取消代理即可! 1.查看有无相关代理: git config --global http.proxy git config --global https.proxy ...

  6. node 淘宝镜像

    永久使用 打开终端执行 npm config set registry https://registry.npm.taobao.org 临时使用 npm --registry https://regi ...

  7. 初学者git的用法

    初学者github的用法 1.在github上创建一个自己的工程 2.按着上面的要求执行你的命令行 3.将你的代码放到这个已经创建了.git的文件夹中,执行git add . 系统出现如下错误:war ...

  8. 在centos下安装pycrypto报错

    yum -y install python36-devel libevent-devel libjpeg-devel zlib-devel

  9. docker学习-qiang

    docker优势 轻量级.资源损耗少.启动快.性能高.只能运行linux系统 容器技术发展历程 1.chroot技术 新建一个子系统(拥有自己完整的系统文件) ldd /bin/ls(查看ls命令依赖 ...

  10. JMeter5.1开发SMTP协议接口脚本

    jmeter可以测试发邮件和读取邮件. 发送邮件 上图部分解释: Server:邮件发送服务 Port:发邮件端口,不加密25,加密465,如果是465端口,Security settings 需要选 ...