首页 > 其他分享 >dp----在一些意想不到的地方用到了dp

dp----在一些意想不到的地方用到了dp

时间:2022-10-04 21:34:47浏览次数:73  
标签:int long ---- 意想不到 abc271 序列 include dp

《题一:Subsequence Path》

原题链接:https://atcoder.jp/contests/abc271/tasks/abc271_e

原题详细题解:https://atcoder.jp/contests/abc271/editorial/4940

题目大意:

有N个城镇编号为1、…、N和M,道路编号为1…、M。

给定一个长度为K的序列E,由1到M之间的整数组成。使用道路从城镇1到城镇N的旅行方式称为好路径,如果:

按照路径中使用的顺序排列的道路编号序列是E的子序列。

请注意,序列的子序列是通过从原始序列中删除0个或多个元素,并在不更改顺序的情况下连接其余元素而获得的序列。

找出一条好路中所用道路的最小长度总和。

如果没有好的道路,请报告这一事实。

 

 我的思想思路:

首先我在想各种图论中的算法,但是失败

然后我想对E这个序列作手脚,但是最后我只能够想到对这个序列爆搜,对于序列上的每一个点看是否要选,然后能否到最终点

但是时间复杂度肯定会超时

然后我就没有想到dp

这里用dp的话dp状态转移方程十分好些:

 

 

 1 #include <iostream>
 2 #include <cstring>
 3 #include <algorithm>
 4 #include <vector>
 5 using namespace std;
 6 typedef pair<int, int> PII;
 7 const int N = 2 * 1e5 + 5;
 8 const long long INF = 1e18;
 9 int n, m, k;
10 PII roads[N];
11 int len[N];
12 long long dp[N];
13 int main()
14 {
15     cin >> n >> m >> k;
16     for (int i = 1; i <= m; i++)
17     {
18         int a, b, w;
19         scanf("%d%d%d", &a, &b, &w);
20         roads[i] = {a, b};
21         len[i] = w;
22     }
23     for (int i = 0; i <= n; i++)
24         dp[i] = INF;
25     dp[1] = 0;
26     while (k--)
27     {
28         int index;
29         scanf("%d", &index);
30         int s = roads[index].first;
31         int f = roads[index].second;
32         int w = len[index];
33         if (dp[f] > dp[s] + w)
34             dp[f] = dp[s] + w;
35     }
36     if (dp[n] >= INF)
37         cout << -1;
38     else
39         cout << dp[n];
40     return 0;
41 }

 

标签:int,long,----,意想不到,abc271,序列,include,dp
From: https://www.cnblogs.com/cilinmengye/p/16754530.html

相关文章