Atcoder 题面传送门 & 洛谷题面传送门

神仙调整+乱搞题。

首先某些人(including me)一看到最大值最小就二分答案,事实上二分答案对这题正解没有任何启发。

首先将 \(a_i\) 从小到大排序。我们考虑将分配的点对看作一条条线,对于 \(a_x+a_y<M\) 的点对 \((x,y)\) 我们在 \(x,y\) 之间连一条蓝线,对于 \(a_x+a_y\ge M\) 的点对我们在 \(x,y\) 之间连一条红线。

先抛结论,再给证明:如在最优分配方式中,我们的连线方式肯定是长这样的:

证明:使用调整法,证明上述命题,等价于证明对于以下 \(7\) 种情况,左边的情况都可以被调整为右边的情况且答案不会更劣(这里借用了粉兔题解中的图):

我们考虑一一对其进行证明,为了表述方便我们统一假设从左到右四个点分别为 \(a_p,a_q,a_r,a_s\),则显然 \(a_p\le a_q\le a_r\le a_s\):

  1. 对于左边第一个的情况,左边的最大值为 \(\max(a_p+a_q,a_r+a_s)=a_r+a_s\),右边的最大值为 \(\max(a_p+a_s,a_q+a_r)\),而由于 \(a_p+a_s\le a_r+a_s,a_q+a_r\le a_r+a_s\),故右边答案不会比左边更劣。
  2. 对于右边第一个的情况,左边的最大值为 \(\max(a_p+a_r,a_q+a_s-M)\),而由于 \(a_s-M<0\),故 \(a_q+a_s-M<a_q<a_p+a_r\),故左边的最大值实际上是 \(a_p+a_r\),右边的最大值为 \(\max(a_p+a_q,a_r+a_s-M)\),而显然 \(a_p+a_q\le a_p+a_r,a_r+a_s-M<a_r\le a_p+a_r\),故右边答案不会比左边更劣,同时又因为 \(a_p+a_q\le a_p+a_r<M,a_r+a_s\ge a_q+a_s\ge M\),故 \(a_p,a_q\) 之间连的依旧是蓝线,\(a_r,a_s\) 之间连的依旧是红线。
  3. 对于左边第二个的情况,左边的最大值为 \(a_q+a_s\),右边的最大值为 \(\max(a_p+a_s,a_q+a_r)\),而 \(a_p+a_s\le a_q+a_s,a_q+a_r\le a_q+a_s\),故右边答案不会比左边更劣。
  4. 对于右边第二个的情况,左边的最大值为 \(\max(a_p+a_s,a_q+a_r-M)=a_p+a_s\),右边的最大值为 \(\max(a_p+a_q,a_r+a_s-M)\),又由于 \(a_p+a_q\le a_p+a_s,a_r+a_s-M<a_r\le a_p+a_s\),故右边答案不会比左边更劣。
  5. 对于左边第三、四个的情况,证明方法同左边第一、二个,只不过需要整体减个 \(M\)。
  6. 对于右边第三个的情况,左边最大值为 \(\max(a_q+a_r,a_p+a_s-M)=a_q+a_r\),右边最大值为 \(\max(a_p+a_q,a_r+a_s-M)\),又由于 \(a_p+a_q\le a_q+a_r,a_r+a_s-M<a_r\le a_q+a_r\),故右边答案不会比左边更劣。

综上,只要出现线相交或者不同颜色的线出现包含关系的情况,都可以被调整,证毕。

接下来考虑怎样求答案,暴力枚举分割点显然是不可行的,不过注意到对于两个不同且均合法的分割点 \(p\) 和 \(p'\),如果 \(p<p'\),那么以 \(p\) 为分割点的每条线的权值都小于以 \(p'\) 为分割点的每条线的权值,因此我们肯定希望分割点越靠左越好,而如果我们分割点太左了(yyq:政治学得很好嘛),那就会出现右侧有的线不是红线的情况,因此我们可以二分找出合法的且最靠左的分割点 \(p\),然后求出答案即可。

时间复杂度线性对数。

namespace fastio{
#define FILE_SIZE 1<<23
char rbuf[FILE_SIZE],*p1=rbuf,*p2=rbuf,wbuf[FILE_SIZE],*p3=wbuf;
inline char getc(){return p1==p2&&(p2=(p1=rbuf)+fread(rbuf,1,FILE_SIZE,stdin),p1==p2)?-1:*p1++;}
inline void putc(char x){(*p3++=x);}
template<typename T> void read(T &x){
x=0;char c=getchar();T neg=0;
while(!isdigit(c)) neg|=!(c^'-'),c=getchar();
while(isdigit(c)) x=(x<<3)+(x<<1)+(c^48),c=getchar();
if(neg) x=(~x)+1;
}
template<typename T> void recursive_print(T x){if(!x) return;recursive_print(x/10);putc(x%10^48);}
template<typename T> void print(T x){if(!x) putc('0');if(x<0) putc('-'),x=~x+1;recursive_print(x);}
template<typename T> void print(T x,char c){if(!x) putc('0');if(x<0) putc('-'),x=~x+1;recursive_print(x);putc(c);}
void print_final(){fwrite(wbuf,1,p3-wbuf,stdout);}
}
using namespace fastio;
const int MAXN=1e5;
int n,m,a[MAXN*2+5];
int add(int x,int y){return (x+y<m)?(x+y):(x+y-m);}
bool check(int mid){bool flg=1;for(int i=(mid<<1)+1;i<=n<<1;i++) flg&=(a[i]+a[(n<<1)+(mid<<1)+1-i]>=m);return flg;}
int main(){
read(n);read(m);
for(int i=1;i<=n<<1;i++) read(a[i]);
sort(a+1,a+(n<<1)+1);int l=0,r=n,p=-1;
while(l<=r){
int mid=l+r>>1;
if(check(mid)) p=mid,r=mid-1;
else l=mid+1;
} int mx=0;
// printf("%d\n",p);
for(int i=1;i<=p<<1;i++) chkmax(mx,add(a[i],a[(p<<1)+1-i]));
for(int i=(p<<1)+1;i<=n<<1;i++) chkmax(mx,add(a[i],a[(n<<1)+(p<<1)+1-i]));
printf("%d\n",mx);
return 0;
}

Atcoder Grand Contest 032 E - Modulo Pairing(乱搞+二分)的更多相关文章

  1. Atcoder Grand Contest 032

    打的第一场Atcoder,已经知道会被虐得很惨,但没有想到竟然只做出一题-- 思维急需提升. A - Limited Insertion 这题还是很签到的. 感觉正着做不好做,我们反着来,把加数变为删 ...

  2. AtCoder Grand Contest 032 A - Limited Insertion( 思维)

    Time Limit: 2 sec / Memory Limit: 1024 MB Score : 400400 points Problem Statement Snuke has an empty ...

  3. AtCoder Grand Contest 032 B - Balanced Neighbors——构造

    题意 B - Balanced Neighbors 给定一个整数 $N$($3\leq N \leq 100$),构造一个顶点编号为 $1...N$ 的无向图,需满足如下两个条件: 简单图且连通 存在 ...

  4. Atcoder Grand Contest 006 D - Median Pyramid Hard(二分+思维)

    Atcoder 题面传送门 & 洛谷题面传送门 u1s1 Atcoder 不少思维题是真的想不出来,尽管在 Atcoder 上难度并不高 二分答案(这我倒是想到了),检验最上面一层的数是否 \ ...

  5. 【AtCoder Grand Contest 007E】Shik and Travel [Dfs][二分答案]

    Shik and Travel Time Limit: 50 Sec  Memory Limit: 512 MB Description 给定一棵n个点的树,保证一个点出度为2/0. 遍历一遍,要求每 ...

  6. AtCoder Grand Contest 010

    AtCoder Grand Contest 010 A - Addition 翻译 黑板上写了\(n\)个正整数,每次会擦去两个奇偶性相同的数,然后把他们的和写会到黑板上,问最终能否只剩下一个数. 题 ...

  7. AtCoder Grand Contest 012

    AtCoder Grand Contest 012 A - AtCoder Group Contest 翻译 有\(3n\)个人,每一个人有一个强大值(看我的假翻译),每三个人可以分成一组,一组的强大 ...

  8. AtCoder Grand Contest 011

    AtCoder Grand Contest 011 upd:这篇咕了好久,前面几题是三周以前写的... AtCoder Grand Contest 011 A - Airport Bus 翻译 有\( ...

  9. AtCoder Grand Contest 031 简要题解

    AtCoder Grand Contest 031 Atcoder A - Colorful Subsequence description 求\(s\)中本质不同子序列的个数模\(10^9+7\). ...

随机推荐

  1. python常用功能

    1. 获取昨天日期 引入datetime模块 import datetime def getYesterday(): today = datetime.date.today() #返回当前本地日期 # ...

  2. ES2020新特性记录

    1.可选链操作符 // oldlet ret = obj && obj.first && obj.first.second// newlet ret = obj?.fi ...

  3. Scrum Meeting 0507

    零.说明 日期:2021-5-7 任务:简要汇报两日内已完成任务,计划后两日完成任务 一.进度情况 组员 负责 两日内已完成的任务 后两日计划完成的任务 qsy PM&前端 测试 测试 cyy ...

  4. 【二食堂】Alpha - 项目展示

    项目展示 1. 团队介绍 二食堂很难排队 姓名 介绍 职务 刘享 热爱游戏,尤其是RPG和metrovinia类的游戏. 会C/C++, python, java. 后端 左正 一个普通的大学生,Py ...

  5. 热身训练4 Eighty seven

    Eighty seven 简要题意: n个卡片,其中第i个卡片的数值为$a[i]$.一共q次询问,每次询问将删去其中3个卡片(可能删除若干相同的卡片)后,问能否选出10个卡片,数值之和等于87. n≤ ...

  6. yum history使用详解(某次为解决误卸载软件的回退实验)

    [root@localhost ~]# yum history list #查看历史 Loaded plugins: fastestmirror ID | Command line | Date an ...

  7. Qt char * 转QString

    1.QString转char * 先将QString转换为QByteArray,再将QByteArray转换为char *. 注意:不能用char *mm = str.toLatin1().data( ...

  8. 链表中倒数第K个结点 牛客网 剑指Offer

    链表中倒数第K个结点 牛客网 剑指Offer 题目描述 输入一个链表,输出该链表中倒数第k个结点. # class ListNode: # def __init__(self, x): # self. ...

  9. ELK集群之filebeat(6)

    filebeat工作原理 ilebeat是本地文件的日志数据采集器. 作为服务器上的代理安装,Filebeat监视日志目录或特定日志文件,tail file,并将它们转发给Elasticsearch或 ...

  10. v-html | 数据内容包含元素标签或者样式

    问题 如果我们展示的数据包含元素标签或者样式,我们想展示标签或样式所定义的属性作用,该怎么进行渲染 插值表达式{{}}和v-text指令被直接解析为了字符串元素. <body> <d ...