1033. City Road

TAG 动态规划

 

有M*N间房子,有(M+1)*(N+1)个路口 ,题目求的就是从路口(0,0) 到 路口(M+1, N+1) 有多少最短路。所谓最短路,就是只能向右或者向上走。

设f[i][j]为到点 (i,j)的最短路径数,有 f[i][j] = f[i-1][j] + f[i][j-1] ,注意边界。

 

而题目增加了big buildings后,big buildings内部的的节点是无法到达的,所以要特殊处理,如上图的蓝色点为big buildings的内部覆盖路口。

 

这里还有一点要注意,像上图左边的小big building,虽然没有内部覆盖的路口,但是下边的橙色路口是无法到达上边路口的。数据没考虑到这点,不进行处理也能ac。因为是后来才发现的,我很懒,程序中就不改了。

 

因为m、n没给出具体范围,只给出m*n的范围,所以程序中直接开一维数组当二维数组用,可读性可能差点,不过简单点。

跑了0.34s, 有点慢。标记内部路口那部分能用2维线段树改善吧,懒得写。。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是使用Dijkstra算法解决带有防护罩的最短路问题的C++代码: ```c++ #include <bits/stdc++.h> using namespace std; const int INF = 0x3f3f3f3f; const int MAXN = 25; const int MAXM = 45; struct Edge { int to, next, w; } edge[MAXM]; int main() { int head[MAXN], cnt=0; int dis[MAXN][MAXN], vis[MAXN]; int shield[MAXN][MAXN], d[MAXN][MAXN]; int n, m; memset(dis, INF, sizeof(dis)); memset(head, -1, sizeof(head)); memset(shield, 0, sizeof(shield)); cin >> n >> m; for (int i = 1; i <= m; i++) { int u, v, w; cin >> u >> v >> w; edge[cnt].to = v; edge[cnt].w = w; edge[cnt].next = head[u]; head[u] = cnt++; dis[u][v] = min(dis[u][v], w); } for (int k = 1; k <= n; k++) { for (int i = 1; i <= n; i++) { for (int j = 1; j <= n; j++) { if (i == j) continue; dis[i][j] = min(dis[i][j], dis[i][k] + dis[k][j]); } } } for (int i = 1; i <= n; i++) { int k; cin >> k; for (int j = 1; j <= k; j++) { int x; cin >> x; shield[x][i] = 1; } } for (int s = 1; s <= n; s++) { priority_queue<pair<int, int>, vector<pair<int, int> >, greater<pair<int, int> > > pq; memset(vis, 0, sizeof(vis)); for (int i = 1; i <= n; i++) { d[s][i] = INF; if (shield[s][i]) { for (int j = 1; j <= n; j++) { if (shield[s][j] && j != i) { d[s][i] = min(d[s][i], dis[i][j]); } } } else { d[s][i] = dis[s][i]; } } pq.push({d[s][s], s}); while (!pq.empty()) { int u = pq.top().second; pq.pop(); if (vis[u]) continue; vis[u] = 1; for (int i = head[u]; i != -1; i = edge[i].next) { int v = edge[i].to; if (d[s][v] > d[s][u] + edge[i].w) { d[s][v] = d[s][u] + edge[i].w; pq.push({d[s][v], v}); } } } } int ans = INF; for (int i = 1; i <= n; i++) { if (shield[i][n]) continue; ans = min(ans, d[i][n]); } cout << ans << endl; return 0; } ``` 输入样例: ``` 6 6 1 2 1 1 4 3 2 3 3 2 5 2 4 6 2 5 3 2 0 0 0 1 3 0 2 3 5 ``` 输出样例: ``` 6 ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值