BZOJ 2143

新技能:并查集优化最短路。

暴力最短路是$O(n^4)$的,然后拿个线段树优化一下连边就$O($能过$)$了。

但是这样都太慢了。

我们考虑一个点如果之前被更新过了,那么之后就不会被更新了,所以我们只要能跳过这个已经被更新过的点,直接去更新没有更新过的点就行了,刚好对应了一个并查集的路径压缩,这样子每一次跳到一个没有更新过的点就是$O(1)$的了。

每一个点拿出来的更新的时候其实是要付出它的点权,所以我们要把$dis_{x, y}  + a_{x, y}$一起丢到堆里去才能保证转移的正确性。

还是不会算时间复杂度,但是非常优秀。

Code:

#include <cstdio>
#include <cstring>
#include <queue>
using namespace std;
typedef long long ll; const int N = ;
const ll inf = 0x3f3f3f3f3f3f3f3f; int n, m, b[N][N], ufs[N][N], sx[], sy[];
ll a[N][N], dis[N][N], ans[];
bool vis[N][N]; struct Node {
int x, y;
ll d; inline Node (int nowX = , int nowY = , ll nowD = 0LL) {
x = nowX, y = nowY, d = nowD;
} friend bool operator < (const Node &u, const Node &v) {
return u.d > v.d;
} };
priority_queue <Node> Q; template <typename T>
inline void read(T &X) {
X = ; char ch = ; T op = ;
for(; ch > '' || ch < ''; ch = getchar())
if(ch == '-') op = -;
for(; ch >= '' && ch <= ''; ch = getchar())
X = (X << ) + (X << ) + ch - ;
X *= op;
} inline int max(int x, int y) {
return x > y ? x : y;
} inline int min(int x, int y) {
return x > y ? y : x;
} inline int abs(int x) {
return x > ? x : -x;
} int find(int x, int y) {
return ufs[x][y] == y ? y : ufs[x][y] = find(x, ufs[x][y]);
} void dij(int st) {
for(int i = ; i <= n; i++)
for(int j = ; j <= m + ; j++) {
dis[i][j] = inf;
ufs[i][j] = j;
vis[i][j] = ;
}
dis[sx[st]][sy[st]] = 0LL;
Q.push(Node(sx[st], sy[st], a[sx[st]][sy[st]]));
ufs[sx[st]][sy[st]] = sy[st] + ;
for(; !Q.empty(); ) {
Node out = Q.top(); Q.pop();
int x = out.x, y = out.y;
if(vis[x][y]) continue;
vis[x][y] = ; int ln = max(, x - b[x][y]), rn = min(n, x + b[x][y]);
for(int i = ln; i <= rn; i++) {
int stp = b[x][y] - abs(i - x);
int lm = max(, y - stp), rm = min(m, y + stp);
for(int j = find(i, lm); j <= rm; j = find(i, j)) {
if(dis[i][j] > dis[x][y] + a[x][y]) {
dis[i][j] = dis[x][y] + a[x][y];
Q.push(Node(i, j, dis[i][j] + a[i][j]));
}
ufs[i][j] = j + ;
}
}
}
} int main() {
freopen("4.in", "r", stdin);
// freopen("my.out", "w", stdout); read(n), read(m);
for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++)
read(b[i][j]);
for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++)
read(a[i][j]); for(int i = ; i <= ; i++)
read(sx[i]), read(sy[i]); for(int i = ; i <= ; i++) {
dij(i);
for(int j = ; j <= ; j++) ans[j] += dis[sx[j]][sy[j]];
} ll res = inf, pos = ;
for(int i = ; i <= ; i++)
if(ans[i] < res) res = ans[i], pos = i; if(res >= inf) {
puts("NO");
return ;
} if(pos == ) puts("X");
if(pos == ) puts("Y");
if(pos == ) puts("Z"); printf("%lld\n", res);
return ;
}

最新文章

  1. Androidannotations框架
  2. 在Navicat premium上创建的SQL Server数据库,实现用PHP连接(即php连接微软MSSQL)
  3. 《DSP using MATLAB》示例Example5.1
  4. Hibernate 常见异常
  5. C# 中的sealed修饰符学习
  6. 在类库中无法使用ConfigurationManager
  7. 关于python写GUI桌面应用的一些研究结果
  8. json文件报expected name at 1 1错误
  9. Jsp分页的简单制作
  10. python学习笔记5-字典
  11. 华为Qinq的配置
  12. 如何在cmd中执行python文件
  13. Convert Binary Search Tree to Doubly Linked List
  14. Mvc4_ @RenderBody、@RenderPage、@RenderSection用法
  15. 【转】Linux 移动或重命名文件/目录-mv 的10个实用例子
  16. Tcp连接的七次握手浅析
  17. iscroll 4 下拉 上拉 加载
  18. C# 中 DataTable转换成IList
  19. 【API】注册表编程基础-RegCreateKeyEx、RegSetValueEx
  20. rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted

热门文章

  1. 51nod 2006 飞行员配对
  2. JavaScript define
  3. (转)android - anim translate中 fromXDelta、toXDelta、fromYDelta、toXDelta属性
  4. Java 数组的定义和遍历
  5. 获取微信用户openid
  6. HDOJ5877(dfs序+离散化+树状数组)
  7. TCP之一:传输控制协议(Transmission Control Protocol, TCP)
  8. STM32使用printf丢失第一个字母的问题
  9. NSString 转换
  10. 工程添加EF框架的方法