Codeforces 677D - Vanya and Treasure - [DP+优先队列BFS]
题目链接:http://codeforces.com/problemset/problem/677/D
题意:
有 $n \times m$ 的网格,每个网格上有一个棋子,棋子种类为 $t[i][j]$,棋子的种类数为 $p$。
现在出发点为 $(1,1)$,必须按照种类 $1 \sim p$ 进行移动,即从种类 $x$ 的棋子出发,下一个目标必须是 $x+1$ 才行,直到走到种类为 $p$ 的棋子就终止。求最短路径。
题解:
我们先把棋子按照种类分组,分成 $p$ 组。
$dp[i][j]$ 表示到达目前这个棋子的最短路,那么转移方程为 $dp[i][j] = min(dp[i][j],dp[x][y]+|x-i|+|y-j|)$,其中 $(x,y)$ 为上一组中所有棋子的坐标。
然后要是直接暴力的状态转移的话,是要TLE的,考虑进行优化。
考虑一个界限 $K$,假设当前组为 $T[i]$,上一组为 $T[i-1]$,
那么当 $T[i].size \le K$ 时,我们就用继续用上面的暴力动态转移,那么对于所有的“上一组”点数加起来不会差过 $nm$,因此总时间复杂度 $O(K \cdot nm)$;
如果 $T[i].size > K$,我们在网格上进行多源点的优先队列BFS(或者说优先队列dijkstra),源点是所有的 $T[i-1]$ 组内的点,搜出到所有 $T[i]$ 组内的点的最短距离,这样BFS最多跑一遍所有网格,时间复杂度 $O(nm \log(nm))$;由于这样的组数目不会超过 $\frac{nm}{K}$ 个,所以总时间复杂度为 $O(\frac{nm}{K} nm \log(nm) )$。
这样一来,两种加起来的总时间复杂度就是 $O(Knm+\frac{nm}{K}nm\log(nm) ) = O( nm (K + \frac{nm\log(nm)}{K}) )$,由此可知取 $K=\sqrt{nm \log(nm) }$ 时,时间复杂度最小,为 $O(nm\sqrt{nm\log(nm)})$。
AC代码:
- #include<bits/stdc++.h>
- #define idx(x,y) ((x-1)*m+y)
- #define mp(x,y) make_pair(x,y)
- #define fi first
- #define se second
- using namespace std;
- typedef pair<int,int> P;
- const int INF=0x3f3f3f3f;
- const int maxn=;
- int n,m,p,ed,K;
- P pos[maxn*maxn];
- vector<int> T[maxn*maxn];
- int dp[maxn*maxn];
- int dist(const P& u,const P& v) {
- return abs(u.fi-v.fi)+abs(u.se-v.se);
- }
- int dx[]={,,-,};
- int dy[]={,,,-};
- int d[maxn*maxn]; bool vis[maxn*maxn];
- priority_queue< P, vector<P>, greater<P> > Q;
- int main()
- {
- ios::sync_with_stdio();
- cin.tie(), cout.tie();
- cin>>n>>m>>p;
- K=sqrt(n*m*log2(n*m));
- for(int i=,tp;i<=n;i++)
- {
- for(int j=;j<=m;j++)
- {
- cin>>tp;
- T[tp].push_back(idx(i,j));
- pos[idx(i,j)]=mp(i,j);
- if(tp==p) ed=idx(i,j);
- }
- }
- memset(dp,0x3f,sizeof(dp));
- for(auto v:T[]) dp[v]=dist(mp(,),pos[v]);
- for(int i=;i<=p;i++)
- {
- if(T[i].size()<=K)
- {
- for(auto v:T[i])
- for(auto u:T[i-])
- dp[v]=min(dp[v],dp[u]+dist(pos[u],pos[v]));
- }
- else
- {
- memset(d,0x3f,sizeof(d));
- memset(vis,,sizeof(vis));
- for(auto u:T[i-]) d[u]=dp[u], Q.push(mp(d[u],u));
- while(Q.size())
- {
- int u=Q.top().se; Q.pop();
- if(vis[u]) continue;
- vis[u]=;
- for(int k=;k<;k++)
- {
- if(pos[u].fi+dx[k]< || pos[u].fi+dx[k]>n) continue;
- if(pos[u].se+dy[k]< || pos[u].se+dy[k]>n) continue;
- int v=idx(pos[u].fi+dx[k],pos[u].se+dy[k]);
- if(vis[v]) continue;
- if(d[v]>d[u]+) d[v]=d[u]+, Q.push(mp(d[v],v));
- }
- }
- for(auto v:T[i]) dp[v]=d[v];
- }
- }
- cout<<dp[ed]<<endl;
- }
Codeforces 677D - Vanya and Treasure - [DP+优先队列BFS]的更多相关文章
- Codeforces 677D Vanya and Treasure 暴力+BFS
链接 Codeforces 677D Vanya and Treasure 题意 n*m中有p个type,经过了任意一个 type=i 的各自才能打开 type=i+1 的钥匙,最初有type=1的钥 ...
- CodeForces 677D. Vanya and Treasure 枚举行列
677D. Vanya and Treasure 题意: 给定一张n*m的图,图上每个点标有1~p的值,你初始在(1,1)点,你必须按照V:1,2,3...p的顺序走图上的点,问你如何走时间最少. 思 ...
- CodeForces 677D Vanya and Treasure
$dp$,树状数组. 很明显这是一个$DAG$上的$dp$,由于边太多,暴力$dp$会超时,需要优化. 例如计算$dp[x][y]$,可以将区域分成四块,$dp[x][y]$取四块中的最小值,每一块用 ...
- Codeforces Round #355 (Div. 2) D. Vanya and Treasure dp+分块
题目链接: http://codeforces.com/contest/677/problem/D 题意: 让你求最短的从start->...->1->...->2->. ...
- codeforces 677D D. Vanya and Treasure(二维线段树)
题目链接: D. Vanya and Treasure time limit per test 1.5 seconds memory limit per test 256 megabytes inpu ...
- 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 ...
- codeforces 677D(分层图dp)
Codeforces 677D 传送门:https://codeforces.com/contest/677/problem/D 题意: 给你一个n*m的方格图,每个点有一个权值val,现在要求你从坐 ...
- Codeforces 981 共同点路径覆盖树构造 BFS/DP书架&最大值
A /*Huyyt*/ #include<bits/stdc++.h> #define mem(a,b) memset(a,b,sizeof(a)) #define pb push_bac ...
- 题解-CF677D Vanya and Treasure
CF677D Vanya and Treasure 有一个 \(n\times m\) 的矩阵 \(a(1\le a_{i,j}\le p)\),求从起点 \((1,1)\) 出发依次遍历值为 \(1 ...
随机推荐
- linux popen()与system()的区别
linux popen()与system()的区别 popen() 可以在调用程序和POSIX shell /usr/bin/sh 要执行的命令之间创建一个管道(请参阅sh-posix(1) ). p ...
- 11g新特性-自动sql调优(Automatic SQL Tuning)
11g新特性-自动sql调优(Automatic SQL Tuning) 在Oracle 10g中,引进了自动sql调优特性.此外,ADDM也会监控捕获高负载的sql语句. 在Oracle 11g中, ...
- 训练集测试集划分 train_test_split(X, y, stratify=y)
from sklearn.model_selecting import train_test_spilt() 参数stratify: 依据标签y,按原数据y中各类比例,分配给train和test,使得 ...
- easy_install与pip 区别
作为Python爱好者,如果不知道easy_install或者pip中的任何一个的话,那么...... easy_insall的作用和perl中的cpan,ruby中的gem类似,都提供了在线一键 ...
- MXNET:权重衰减-gluon实现
构建数据集 # -*- coding: utf-8 -*- from mxnet import init from mxnet import ndarray as nd from mxnet.gluo ...
- 施工测量中Cad一些非常有用的插件
经常会遇到坐标在cad中批量展点.从cad中批量保存坐标点.导入cad中的坐标怎么才能有点号,怎么快速标注cad里的坐标点··· ··· 这一切都是可以程序化的,cad是可以二次开发的,我经常用易语言 ...
- C#获取起始位置以及添加全局资源字典
获取起始位置 Path.Combine(AppDomain.CurrentDomain.BaseDirectory); 添加全局资源 string temp = "this is a str ...
- 用panels 制作drupal首页
1.下载zen主题 路径:https://www.drupal.org/project/zen2.“启用并设为默认”3.基于zen制作子主题 把zen目录下STARTERKIT文件夹,复制到sites ...
- vue与自定义元素的关系
你可能已经注意到 Vue.js 组件非常类似于自定义元素--它是 Web 组件规范的一部分.实际上 Vue.js 的组件语法参考了该规范.例如 Vue 组件实现了 Slot API 与 is 特性.但 ...
- Java知多少(25)再谈Java包
在Java中,为了组织代码的方便,可以将功能相似的类放到一个文件夹内,这个文件夹,就叫做包. 包不但可以包含类,还可以包含接口和其他的包. 目录以"\"来表示层级关系,例如 E:\ ...