Solution -「ROI 2019」「LOJ #3192」课桌
\(\mathcal{Description}\)
Link.
原题意足够简洁啦。(
\(\mathcal{Solution}\)
乍一看比较棘手,但可以从座位的安排方式入手,有结论:
一个班的学生按身高排序后,相邻的两两坐在一桌。
证明略,比较显。
第二个结论:
设按上述方案分桌,从左至右将每桌编号为 \(1\sim n\)。则每个班级的第 \(i\) 号桌都坐在同一个位子。
考虑交换两桌不能使答案变优即证。
考试的时候结论都看出来了结果写假了你敢信 qwq。
再来考虑桌子,如果一张桌子的区间被另一桌子的区间覆盖,则这张桌子一定不需要。所以剩下的区间按左端点升序排列后,右端点亦为升序。则若第 \(i_1\) 桌选用 \(j_1\) 号桌子,第 \(i_2\) 桌选用 \(j_2\) 号桌子,就会有 \(i_1<i_2\Leftrightarrow j_1\le j_2\) 成立。所以直接决策单调性分治优化求解即可。复杂度 \(\mathcal O(nm+k\log k\log n)\)。
\(\mathcal{Code}\)
/* Clearink */
#include <cstdio>
#include <vector>
#include <cassert>
#include <algorithm>
#define rep( i, l, r ) for ( int i = l, repEnd##i = r; i <= repEnd##i; ++i )
#define per( i, r, l ) for ( int i = r, repEnd##i = l; i >= repEnd##i; --i )
typedef long long LL;
typedef std::pair<int, int> PII;
#define fi first
#define se second
inline int rint() {
int x = 0, s = getchar();
for ( ; s < '0' || '9' < s; s = getchar() );
for ( ; '0' <= s && s <= '9'; s = getchar() ) x = x * 10 + ( s ^ '0' );
return x;
}
const int MAXN = 2e5;
int n, m, c, h[MAXN * 2 + 5];
LL pre[MAXN * 2 + 5], suf[MAXN * 2 + 5];
PII desk[MAXN + 5];
std::vector<int> group[MAXN + 5];
inline LL solve( const int gl, const int gr, const int dl, const int dr ) {
if ( gl > gr ) return 0;
int gm = gl + gr >> 1, dm = -1, sz = ( m << 1 ) - 1;
std::vector<int>& curG = group[gm];
std::sort( curG.begin(), curG.end() );
rep ( i, 0, sz ) pre[i] = ( i ? pre[i - 1] : 0 ) + curG[i];
per ( i, sz, 0 ) suf[i] = suf[i + 1] + curG[i];
LL res = 1ll << 60;
rep ( i, dl, dr ) {
int l = desk[i].fi, r = desk[i].se;
int cl = std::lower_bound( curG.begin(),
curG.end(), l ) - curG.begin();
int cr = std::upper_bound( curG.begin(),
curG.end(), r ) - curG.begin() - 1;
LL cost = 1ll * l * cl - ( cl ? pre[cl - 1] : 0 )
+ suf[cr + 1] - 1ll * r * ( sz - cr );
if ( cost < res ) res = cost, dm = i;
}
return res + solve( gl, gm - 1, dl, dm ) + solve( gm + 1, gr, dm, dr );
}
int main() {
// freopen( "desk.in", "r", stdin );
// freopen( "desk.out", "w", stdout );
m = rint(), n = rint(), c = rint();
rep ( i, 1, c ) desk[i].fi = rint(), desk[i].se = rint();
std::sort( desk + 1, desk + c + 1 );
int idx = 0;
rep ( i, 1, c ) {
desk[idx += desk[i].fi != desk[i - 1].fi] = desk[i];
}
c = idx;
rep ( i, 1, m ) {
rep ( j, 1, n << 1 ) h[j] = rint();
std::sort( h + 1, h + ( n << 1 | 1 ) );
rep ( j, 1, n << 1 ) group[j + 1 >> 1].push_back( h[j] );
}
printf( "%lld\n", solve( 1, n, 1, c ) );
return 0;
}
Solution -「ROI 2019」「LOJ #3192」课桌的更多相关文章
- LOJ#3054. 「HNOI 2019」鱼
LOJ#3054. 「HNOI 2019」鱼 https://loj.ac/problem/3054 题意 平面上有n个点,问能组成几个六个点的鱼.(n<=1000) 分析 鱼题,劲啊. 容易想 ...
- 【LOJ】#3036. 「JOISC 2019 Day3」指定城市
LOJ#3036. 「JOISC 2019 Day3」指定城市 一个点的可以dp出来 两个点也可以dp出来 后面的就是在两个点的情况下选一条最长的链加进去,用线段树维护即可 #include < ...
- 【LOJ】#3034. 「JOISC 2019 Day2」两道料理
LOJ#3034. 「JOISC 2019 Day2」两道料理 找出最大的\(y_{i}\)使得\(sumA_{i} + sumB_{y_i} \leq S_{i}\) 和最大的\(x_{j}\)使得 ...
- 【LOJ】#3032. 「JOISC 2019 Day1」馕
LOJ#3032. 「JOISC 2019 Day1」馕 处理出每个人把馕切成N段,每一段快乐度相同,我们选择第一个排在最前的人分给他的第一段,然后再在未选取的的人中选一个第二个排在最前的切一下,并把 ...
- 【LOJ】#3033. 「JOISC 2019 Day2」两个天线
LOJ#3033. 「JOISC 2019 Day2」两个天线 用后面的天线更新前面的天线,线段树上存历史版本的最大值 也就是线段树需要维护历史版本的最大值,后面的天线的标记中最大的那个和最小的那个, ...
- 【LOJ】#3031. 「JOISC 2019 Day1」聚会
LOJ#3031. 「JOISC 2019 Day1」聚会 听说随机可过? 我想了很久想了一个不会被卡的做法,建出前\(u - 1\)个点的虚树,然后找第\(u\)个点的插入位置,就是每次找一条最长链 ...
- 【LOJ】#3030. 「JOISC 2019 Day1」考试
LOJ#3030. 「JOISC 2019 Day1」考试 看起来求一个奇怪图形(两条和坐标轴平行的线被切掉了一个角)内包括的点个数 too naive! 首先熟练的转化求不被这个图形包含的个数 -- ...
- 【LOJ】#3014. 「JOI 2019 Final」独特的城市(长链剖分)
LOJ#3014. 「JOI 2019 Final」独特的城市(长链剖分) 显然我们画一条直径,容易发现被统计的只可能是直径某个距离较远的端点到这个点的路径上的值 用一个栈统计可以被统计的点,然后我们 ...
- Solution -「CTS 2019」「洛谷 P5404」氪金手游
\(\mathcal{Description}\) Link. 有 \(n\) 张卡牌,第 \(i\) 张的权值 \(w_i\in\{1,2,3\}\),且取值为 \(k\) 的概率正比于 \ ...
随机推荐
- spring boot 解决 跨域 的两种方法 -- 前后端分离
1.前言 以前做项目 ,基本上是使用 MVC 模式 ,使得视图与模型绑定 ,前后端地址与端口都一样 , 但是现在有些需求 ,需要暴露给外网访问 ,那么这就出现了个跨域问题 ,与同源原则冲突, 造成访问 ...
- 解读与部署(三):基于 Kubernetes 的微服务部署即代码
在基于 Kubernetes 的基础设施即代码一文中,我概要地介绍了基于 Kubernetes 的 .NET Core 微服务和 CI/CD 动手实践工作坊使用的基础设施是如何使用代码描述的,以及它的 ...
- 《剑指offer》面试题28. 对称的二叉树
问题描述 请实现一个函数,用来判断一棵二叉树是不是对称的.如果一棵二叉树和它的镜像一样,那么它是对称的. 例如,二叉树 [1,2,2,3,4,4,3] 是对称的. 1 / \ 2 ...
- [Keil 学习] printf, scanf函数的用法
C语言库函数中有一批"标准输入输出函数",它是以标准的输入输出设备(一般为终端设备)为输入输出对象的,其中用得比较多的是printf和scanf函数了. 在嵌入式设备中加入C语言的 ...
- C# 计算三角形和长方形 周长面积
编写一个控制台应用程序,输入三角形或者长方形边长,计算其周长和面积并输出. 代码如下: using System; using System.Collections.Generic; using Sy ...
- 【经验总结-markdown】markdown字体和颜色设置
字体设置 关键词为face <font face = "黑体">我是黑体</font> <font face = "宋体"> ...
- Android 12(S) 图形显示系统 - 应用建立和SurfaceFlinger的沟通桥梁(三)
1 前言 上一篇文章中我们已经创建了一个Native示例应用,从使用者的角度了解了图形显示系统API的基本使用,从这篇文章开始我们将基于这个示例应用深入图形显示系统API的内部实现逻辑,分析运作流程. ...
- golang中的pair
package main import "fmt" type Reader interface { ReadBook() } type Writer interface { Wri ...
- insert插入日期
7.5.insert插入日期 数字格式化:format select ename,sal from emp: 格式化数字:fromat(数字,'格式') select ename,format(sal ...
- nmap 查看主机上开放的端口
作用: 检测网络上的主机检测主机上开放的端口检测操作系统,硬件地址,以及软件版本检测脆弱性的漏洞(Nmap的脚本) 扫描方式: 1. -sS Tcp SYN Scan 不需要三次握手,速度快 ...