LG-P8858 折线 Solution

更好的阅读体验戳此进入

题面

从从 (0,0)(10100,10100) 的矩形里有正偶数 n 个整点,需构造一条从 (0,0)(10100,10100) 的折线,要求其每部分都平行于坐标轴,不能经过给定的整点,需要将整块区域分为包含给定整点数量相等的两块,要最小化其整点。输出合法的折线的整点数,保证一定存在如下直线。

Solution

提供一个 O(Tnlogn) 的线段树上二分做法。

首先我们可以考虑观察一下样例和大样例,不难发现所有答案均在 [2,4] 之间,以此可猜想答案一定在此区间中,可以尝试感性证明一下:

首先一个折点的话一定无法将矩形分为两块,所以不合法。

两个折点的话即为通过一条直线将矩形分为两半,这条直线可以水平也可以竖直,所以对于这种情况,我们只需要对 x 坐标和 y 坐标分别做一个前缀和,然后分别遍历一遍,如果存在一个点 i 使 sumi=n2 那么显然合法,答案为 2

三个折点的话随便画一下就会发现,最终的情况一定是隔离起来一个左上角或者右下角,如图:

Oops! The image is blocked! Please visit my own website to observe the image! 图片被墙了,请通过文章头的跳转链接访问!

Oops! The image is blocked! Please visit my own website to observe the image! 图片被墙了,请通过文章头的跳转链接访问!

考虑如何维护,显然可以把这个东西按照类似二位偏序或者说二维数点来写,先按照 x 排序,然后把每一段相同的 x 的所有 y 都插到权值线段树里,然后在线段树上二分查找是否存在一个前缀刚好等于 n2。然后再把整个顺序反过来插反过来查,找是否存在一个后缀恰好等于 n2,如果能找到那么显然可以通过隔离出来一段左上角或右下角的角落构造合法折线,答案即为 3

如果以上的判断都不合法的话那么显然最终答案即为 4,这个通过我们最开始 “面向数据编程” 得到的性质直接得到,也可以考虑画一下,如图:

Oops! The image is blocked! Please visit my own website to observe the image! 图片被墙了,请通过文章头的跳转链接访问!

显然这个时候我们是可以隔离出来任意数量的整点了,比较好理解,考虑一下如果想更多地包含新的点,将中间那块凸起略移动一下 xy 即可,感性理解一下即可。

至此我们便以 O(Tnlogn) 的复杂度解决了这道题,还算比较直观,作为 T1 难度挺合理。

赛时Code

UPD

update-2022_11_22 初稿

update-2022_11_22 修改了一处细节错误