CF677D Vanya and Treasure

有一个 \(n\times m\) 的矩阵 \(a(1\le a_{i,j}\le p)\),求从起点 \((1,1)\) 出发依次遍历值为 \(1\to p\) 的矩阵单元的最短路径曼哈顿距离。保证满足 \(a_{i,j}=p\) 的 \((i,j)\) 唯一。

数据范围:\(1\le n,m\le 300\),\(1\le p\le n\cdot m\)。


先记录 \(\tt vector\) 数组 \(w\),\(w_t\) 表示 \(a_{i,j}=t\) 的位置集合。

\(w_t\) 的每个元素有三个属性:\(x,y,g\)。\(x\) 和 \(y\) 是位置坐标,\(g\) 是出发遍历矩阵值 \(1\to t\) 后到 \((x,y)\) 的最短路径长度。


暴力做法:从 \(w_{i-1}\) 的所有 \(g\) 值递推 \(w_i\) 的所有 \(g\) 值:

\[u.g=\min_{v\in w_{i-1}}\{v.g+{\rm abs}(u.x-v.x)+{\rm abs}(u.y-v.y)\}(u\in w_i)
\]

时间复杂度 \(\Theta\left(\sum_{i\in[2,p]}|w_{i-1}|\cdot |w_i|\right)\le\Theta(n^2m^2)\)

  • 怎么卡到 \(\Theta(n^2m^2)\) 的?

比如 \(n=300,m=300,p=2\),矩阵一半是 \(1\) 一半是 \(2\)。


这题的优化是真的巧,反正我比赛时没想到。

考虑以下情况:

\[\forall i\in[2,p]:|w_{i-1}|\cdot|w_i|\le n\cdot m
\]

总的时间复杂度是:

\[\Theta\left(\sum_{i\in[2,p]}|w_{i-1}|\cdot |w_i|\right)
\]

同时满足 \(\sum_{i=1}^p |w_i|=n\cdot m\),根据柯西不等式:

\[\begin{split}
&\left(\sum_{i=2}^p|w_{i-1}|\cdot |w_i|\right)^2\\
\le&\sum_{i=1}^{p-1}|w_i|^2\sum_{i=2}^{p}|w_i|^2\\
\le&\left(\sum_{i=1}^{p}|w_i|^2\right)^2\\
\le&\left(\sqrt{n\cdot m}\times\left(\sqrt{n\cdot m}\right)^2\right)^2
\end{split}
\]

所以 \(\sum_{i=2}^p|w_{i-1}|\cdot |w_i|\le n\cdot m\times\sqrt{n\cdot m}\)。

复杂为 \(\Theta(n\cdot m\sqrt{n\cdot m})\) 可以通过。


但是如果 \(\exists i\in[2,p]:|w_{i-1}|\cdot|w_i|>n\cdot m\) 怎么办呢?

可以套个 \(\Theta(V)\) 的多源无向无权图最短路模板 \(\tt Bfs\)。

所以此时单次递推的时间复杂度也是 \(\Theta(n\cdot m)\)。

这样的单次递推与 \(|w_{i-1}|\cdot|w_i|=n\cdot m\) 相比:

  1. 一次递推时间复杂度相等。

  2. 由于对于这个 \(i\) 的 \(|w_{i-1}|\cdot|w_i|\) 大,所以对于其他 \(i\) 的 \(|w_{i-1}|\cdot|w_i|\) 较小。所以总时间复杂度小。

所以这样优化后总时间复杂度 \(\le \Theta(n\cdot m\sqrt{n\cdot m})\)。可以通过。


  • 代码:
//Data
const int N=3e2;
int n,m,k,a[N+7][N+7];
struct node{
int x,y,g;
node(int X=0,int Y=0,int G=0){x=X,y=Y,g=G;}
};
vector<node> w[N*N+7]; //Bfs
int d[N+7][N+7];
int tx[]={0,0,-1,1},ty[]={-1,1,0,0};
int ok(int x,int y){return 1<=x&&x<=n&&1<=y&&y<=m;}
void Bfs(vector<node>&s){ //多源无向无权图最短路模板 Bfs。
vector<node> q;
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) d[i][j]=inf;
int sc=-1;
q.pb(s[++sc]);
for(int i=0;i<sz(q);i++){
node v=q[i];
while(sc+1<sz(s)&&s[sc+1].g<=v.g) q.pb(s[++sc]);
for(int t=0;t<4;t++){
node u=node(v.x+tx[t],v.y+ty[t]);
if(ok(u.x,u.y)&&v.g+1<d[u.x][u.y]) d[u.x][u.y]=u.g=v.g+1,q.pb(u);
}
}
} //Main
int main(){
scanf("%d%d%d",&n,&m,&k);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++){
scanf("%d",&a[i][j]);
if(a[i][j]==1) w[a[i][j]].pb(node(i,j,i+j-2));
else w[a[i][j]].pb(node(i,j,inf));
}
for(int key=2;key<=k;key++){
if(sz(w[key-1])*sz(w[key])<=n*m){
for(node&u:w[key]) for(node v:w[key-1])
u.g=min(u.g,v.g+abs(u.x-v.x)+abs(u.y-v.y));
} else {
vector<node> s;
for(node v:w[key-1]) s.pb(v);
Bfs(s);
for(node&u:w[key]) u.g=d[u.x][u.y];
}
}
printf("%d\n",w[k][0].g);
return 0;
}

祝大家学习愉快!

题解-CF677D Vanya and Treasure的更多相关文章

  1. codeforces 677D D. Vanya and Treasure(二维线段树)

    题目链接: D. Vanya and Treasure time limit per test 1.5 seconds memory limit per test 256 megabytes inpu ...

  2. Codeforces Round #355 (Div. 2) D. Vanya and Treasure 分治暴力

    D. Vanya and Treasure 题目连接: http://www.codeforces.com/contest/677/problem/D Description Vanya is in ...

  3. Codeforces 677D Vanya and Treasure 暴力+BFS

    链接 Codeforces 677D Vanya and Treasure 题意 n*m中有p个type,经过了任意一个 type=i 的各自才能打开 type=i+1 的钥匙,最初有type=1的钥 ...

  4. CodeForces 677D. Vanya and Treasure 枚举行列

    677D. Vanya and Treasure 题意: 给定一张n*m的图,图上每个点标有1~p的值,你初始在(1,1)点,你必须按照V:1,2,3...p的顺序走图上的点,问你如何走时间最少. 思 ...

  5. 【12.78%】【codeforces 677D】Vanya and Treasure

    time limit per test1.5 seconds memory limit per test256 megabytes inputstandard input outputstandard ...

  6. 题解 CF492C Vanya and Exams

    CF492C Vanya and Exams 有了Pascal题解,来一波C++题解呀qwq.. 简单的贪心题 按b[i]从小到大排序,一个一个学科写直到达到要求即可 #include<cstd ...

  7. Codeforces Round #355 (Div. 2) D. Vanya and Treasure dp+分块

    题目链接: http://codeforces.com/contest/677/problem/D 题意: 让你求最短的从start->...->1->...->2->. ...

  8. Codeforces 677D - Vanya and Treasure - [DP+优先队列BFS]

    题目链接:http://codeforces.com/problemset/problem/677/D 题意: 有 $n \times m$ 的网格,每个网格上有一个棋子,棋子种类为 $t[i][j] ...

  9. CodeForces 677D Vanya and Treasure

    $dp$,树状数组. 很明显这是一个$DAG$上的$dp$,由于边太多,暴力$dp$会超时,需要优化. 例如计算$dp[x][y]$,可以将区域分成四块,$dp[x][y]$取四块中的最小值,每一块用 ...

随机推荐

  1. 用GitHub Pages搭建博客(五)

    本篇介绍GitHub Pages自定义域名 在用GitHub Pages搭建博客(二)中介绍到,默认的GitHub Pages域名就是仓库地址,即: 账号名.github.io 如果我们要使用自定义域 ...

  2. 异常记录-Gradle依赖掉坑之旅

    前言 最近在项目中遇到了一个问题,死活拉不下来依赖,耗费了一整天,感觉自己真是菜的抠脚. 没想到今天脑子一清醒,刷刷的问题逐个击破了. 问题描述: 项目成员添加了新的依赖,然后我这边项目拉下来,bui ...

  3. 部署Dotnet Core应用到Kubernetes(二)

    前一篇文章,概念性地介绍了K8s的一些基础组件,如Pod.部署和服务.这篇文章,我打算写写如何使用YAML清单定义和配置这些资源.   实际上,在K8s集群中创建对象有几种方式 - 命令,或声明.两种 ...

  4. Redis分布式锁的正确使用与实现原理

    模拟一个电商里面下单减库存的场景. 1.首先在redis里加入商品库存数量. 2.新建一个Spring Boot项目,在pom里面引入相关的依赖. <dependency> <gro ...

  5. webug第二关:从图片中你能找到什么?

    第二关:从图片中你能找到什么? 记事本打开发现提示 binwalk,发现压缩包rar

  6. MindManager 2021 版新增了哪些功能

    MindManager Windows 21是一款强大的可视化工具和思维导图软件,在工作应用中有出色的表现.今天就带大家来看下这个新版本增加了哪些功能? 1.新增现代主题信息样式MindManager ...

  7. yii2-imagine的使用

    <?php /** * 图片常用处理 * * 需要 yii/yii2-imagine 的支持 * php composer.phar require --prefer-dist yiisoft/ ...

  8. Redis 基础设计结构之三 hash(哈希)

    Redis 有 5 种基础数据结构,分别为:string (字符串).list (列表).set (集合).hash (哈希) 和 zset (有序集合). 今天来说一下hash(哈希),hash的数 ...

  9. 牛客练习赛71 数学考试 题解(dp)

    题目链接 题目大意 要你求出有多少个长度为n的排列满足m个限制条件 第i个限制条件 p[i]表示前 p[i]个数不能是1-p[i]的排列 题目思路 这个感觉是dp但是不知道怎么dp 首先就是要明白如果 ...

  10. Linux初学学习笔记 -----正则表达式和通配符

    简单来说通配符是用来匹配文件名和目录而正则表达式是用来匹配文本内容的 常用的通配符 *:匹配任意多个字符 下面的是以p为开头的目录里面的文件 ?:匹配任意一个字符 [-]:匹配括号内出现的任意一个字符 ...