如果将$x$和$y$都离散,那么删除的点的$x_{i}$和$y_{i}$必然都组成了一个完整的区间(包括过程中)

将所有点按$x$排序,再令$f[i][j][0/1]$表示当删除完区间$[i,j]$且位于点$i$/点$j$时答案(若无法删除记为$\infty$),转移考虑下一次删除的点($i-1$或$j+1$)是否可行

(这里的答案指在这个状态后删除尽量多的点所需最短时间,因此最终答案为$f[i][i][0/1]$,初始状态不易确定,可以记忆化搜索来做,当无法转移即为0)

根据上面的结论,记$mn=\min_{i\le k\le j}y_{k}$、$mx=\max_{i\le k\le j}y_{k}$,假设下一次选了$i-1$,那么即要求$y_{i-1}$等于$mn-1$或$mx+1$(选$j+1$同理),由此判定时间复杂度为$o(n^{2})$

贪心,对于存在区间$[i,j]$满足$\forall i\le k<j,|y'_{k}-y'_{k+1}|=1$,当我们删除了$i$接下来必然删除$i+1,...,j$(删除$j$同理),因此不妨将其缩为1个点(仅是视为一个点),但此时复杂度貌似仍然是$o(n^{2})$的

定义一个区间合法当且仅当其在记忆化搜索过程中会被得到,时间复杂度即为合法区间数

一个更有意义的判定方法,区间$[i,j]$合法当且仅当$mx-mn=j-i$且可以通过不断删除两端的最小值或最大值使其为空(即若$y_{i}=mn或mx$即可删除$i$,$j$同理)

对于删除,我们可以贪心,可以发现删除后一定“更容易”删除,即若一个区间合法,任意合法的删除顺序都可以将其删除

定义$S_{l,r}$表示区间$[l,r]$的子合法区间数,$Sl_{l,r}$表示区间$[l,r]$以$l$为左端点的子合法区间数,$Sr_{l,r}$类似(即以$r$为右端点)

对一个合法区间$[l,r]$,考虑如何转移求$S_{l,r}$,分类讨论:

1.$l$和$r$都可以删除,$S_{l,r}=Sl_{l,r-1}+Sr_{l+1,r}+S_{l-1,r-1}+1$

2.$l$和$r$中只有一个可以删除(假设为$l$),$S_{l,r}=S_{l+1,r}+1$

来考虑第一种情况中$Sl_{l,r-1}$的级别,不妨假设$l$为$[l,r]$区间最小值,则$r$必然为区间最大值,然后对于$a_{l}+1$的位置分类讨论:

1.若$a_{l}+1$位于$r-1$,即$a_{r-1}=a_{l}+1$,那么$Sl_{l,r-1}\le 2$($[l,l]$,$[l,r-1]$)

2.若$a_{l}+1$位于$l+1$,即$a_{l+1}=a_{l}+1$,则一定被缩点了,不存在该情况;

3.否则$a_{l+1}$和$a_{r-1}$必然都不是$[l+1,r-1]$区间最小值,同时若$a_{r-1}$为区间最大值必然满足$a_{r-1}=a_{r}-1$,会被缩点,因此只能$a_{l+1}=a_{r}-1$,那么为了保证$[a_{l},a_{l+1}]$中的数都出现,$Sl_{l,r-1}\le 2$

因此$S_{l,r}$为$o(r-l)$的级别,总合法区间数为$o(n)$级别

另外,由于要记忆化搜索记录,需要map维护,最终时间复杂度为$o(n\log_{2}n)$,可以通过

(还有一个细节,可以通过八个方向走到给定点,但平时只能走到相邻的4个方向,相当于可以减少$r-l$步,这个可以在无法转移时记录)

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 200005
4 #define ll long long
5 struct ji{
6 int x,y,id;
7 }a[N];
8 vector<int>v;
9 map<int,ll>f[2][N];
10 int n,ls[N],rs[N];
11 ll ans[N];
12 bool cmp(ji x,ji y){
13 return x.x<y.x;
14 }
15 int dis(ji x,ji y){
16 return abs(x.x-y.x)+abs(v[x.y]-v[y.y]);
17 }
18 ll dp(int l,int r,int mn,int mx,int p){
19 if (f[p][l][r])return f[p][l][r];
20 f[p][l][r]=1e15;
21 ji o=a[l+p*(r-l)];
22 if ((l>1)&&((a[l-1].y==mn-1)||(a[l-1].y==mx+1))){
23 int k=ls[l-1];
24 f[p][l][r]=min(f[p][l][r],dp(k,r,min(mn,a[k].y),max(mx,a[k].y),0)+dis(o,a[k]));
25 }
26 if ((r<n)&&((a[r+1].y==mn-1)||(a[r+1].y==mx+1))){
27 int k=rs[r+1];
28 f[p][l][r]=min(f[p][l][r],dp(l,k,min(mn,a[k].y),max(mx,a[k].y),1)+dis(o,a[k]));
29 }
30 if (f[p][l][r]==1e15)f[p][l][r]=-(r-l);
31 return f[p][l][r];
32 }
33 int main(){
34 scanf("%d",&n);
35 for(int i=1;i<=n;i++){
36 scanf("%d%d",&a[i].x,&a[i].y);
37 a[i].id=i;
38 v.push_back(a[i].y);
39 }
40 sort(v.begin(),v.end());
41 for(int i=1;i<=n;i++)a[i].y=lower_bound(v.begin(),v.end(),a[i].y)-v.begin();
42 sort(a+1,a+n+1,cmp);
43 ls[1]=1,rs[n]=n;
44 for(int i=2;i<=n;i++)
45 if (abs(a[i-1].y-a[i].y)!=1)ls[i]=i;
46 else ls[i]=ls[i-1];
47 for(int i=n-1;i;i--)
48 if (abs(a[i+1].y-a[i].y)!=1)rs[i]=i;
49 else rs[i]=rs[i+1];
50 for(int i=1;i<=n;i++)ans[a[i].id]=dp(i,i,a[i].y,a[i].y,0);
51 for(int i=1;i<=n;i++)printf("%lld\n",ans[i]);
52 }

[atAGC047F]Rooks的更多相关文章

  1. UVA - 11134 Fabled Rooks[贪心 问题分解]

    UVA - 11134 Fabled Rooks We would like to place n rooks, 1 ≤ n ≤ 5000, on a n × n board subject to t ...

  2. (light OJ 1005) Rooks dp

    http://www.lightoj.com/volume_showproblem.php?problem=1005        PDF (English) Statistics Forum Tim ...

  3. 01_传说中的车(Fabled Rooks UVa 11134 贪心问题)

    问题来源:刘汝佳<算法竞赛入门经典--训练指南> P81: 问题描述:你的任务是在n*n(1<=n<=5000)的棋盘上放n辆车,使得任意两辆车不相互攻击,且第i辆车在一个给定 ...

  4. uva 11134 fabled rooks (贪心)——yhx

    We would like to place n rooks, 1 n 5000, on a n nboard subject to the following restrictions• The i ...

  5. L - Fabled Rooks(中途相遇法和贪心)

    Problem F: Fabled Rooks We would like to place n rooks, 1 ≤ n ≤ 5000, on a n×n board subject to the ...

  6. UVA 11134 - Fabled Rooks(贪心+优先队列)

    We would like to place  n  rooks, 1 ≤  n  ≤ 5000, on a  n×n  board subject to the following restrict ...

  7. 贪心 uvaoj 11134 Fabled Rooks

    Problem F: Fabled Rooks We would like to place n rooks, 1 ≤ n ≤ 5000, on a n×n board subject to the ...

  8. uva 11134 - Fabled Rooks(问题转换+优先队列)

    题目链接:uva 11134 - Fabled Rooks 题目大意:给出n,表示要在n*n的矩阵上放置n个车,并且保证第i辆车在第i个区间上,每个区间给出左上角和右小角的坐标.另要求任意两个车之间不 ...

  9. 1005 - Rooks(规律)

    1005 - Rooks   PDF (English) Statistics Forum Time Limit: 1 second(s) Memory Limit: 32 MB A rook is ...

随机推荐

  1. 从零入门 Serverless | 教你使用 IDE/Maven 快速部署 Serverless 应用

    作者 | 许成铭(竞霄) 阿里云开发工程师 SAE 应用部署方式 1. SAE 概述 首先,简单介绍一下 SAE.SAE 是一款面向应用的 Serverless PaaS 平台,支持 Spring C ...

  2. python中的load、loads实现反序列化

    load与loads 简介: 在python自动化中,我们传递一些参数是需要从文件中读取过来的,读取过来的字典并非python对象数据类型而是string类型. 这样在我们传递参数的时候就会出现格式不 ...

  3. git 修改最后一次提交

    git 修改最后一次提交 有时候我们提交完了才发现漏掉了几个文件没有添加,或者提交信息写错了. 此时,可以运行带有 --amend 选项的提交命令来重新提交:git commit --amend -m ...

  4. Java内存分析--栈--堆

    Java内存分析--栈--堆 JVM的内存分析: 1.栈内存 1.连续的存储空间,遵循后进先出的原则. 2.每个线程包含一个栈区,栈区只保存基础数据类型的对象和自定义对象的引用. 3.每个栈中的数据都 ...

  5. Python代码阅读(第21篇):将变量名称转换为蛇式命名风格

    Python 代码阅读合集介绍:为什么不推荐Python初学者直接看项目源码 本篇阅读的代码实现将变量名称转换为蛇式命名风格(snake case)的功能. 本篇阅读的代码片段来自于30-second ...

  6. Java正则中"\\\\"表示普通反斜杠

    Java中"\"用于转义字符,"\\"表示普通无转义功能的反斜杠. 如果将字符串当做正则表达式来解析,那么"\\"也有了特殊意义,它与其后的 ...

  7. 嵌入式STM32的GPIO口工作模式的介绍

    一.输入模式 1. 浮空输入 浮空输入模式下,上拉和下拉两个开关断开,高或低电平通过施密特触发器到达输入数据寄存器,CPU可以通过读取输入数据寄存器从而读取到外部输入的高低电平值. 2. 输入上拉模式 ...

  8. 洛谷 P3195 [HNOI2008] 玩具装箱

    链接: P3195 题意: 给出 \(n\) 个物品及其权值 \(c\),连续的物品可以放进一个容器,如果将 \(i\sim j\) 的物品放进一个容器,产生的费用是 \(\left(j-i+\sum ...

  9. Mac搭建以太坊私有链

    记录过程与问题 一.安装 以go版本的ethereum进行安装 brew tap ethereum/ethereum brew install ethereum # 如果希望基于ethereum的de ...

  10. 使用vsftpd 搭建ftp服务

    ftp 基础服务器基础知识 ftp有三种登录方式.匿名登录(所有用户).本地用户.虚拟用户(guest). FTP工作模式 主动模式:服务端从20端口主动向客户端发起链接. 控制端口21:数据传输端口 ...