校內模擬賽 蟲洞(by NiroBC)

題意:ios

  n個點m條邊的有向圖,每一天每條邊存在的機率都是p,在最優策略下,詢問從1到n的指望天數。git

分析:spa

  dijkstra。code

  每次必定會優先選dp最小的後繼走,若是這條邊不存在,選次小的,以此類推。blog

  dp[i]表示從i開始到n的指望天數,從後往前推,每次取出dp最小的,更新其餘點。get

代碼:string

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cctype>
#include<cmath>
#include<set>
#include<map>
#include<vector>
#include<queue>
#include<bitset>
#define pa pair<double,int>
#define fore(i, u, v) for (int i = head[u], v = e[i].to; i; i = e[i].nxt, v = e[i].to) 
using namespace std;
typedef long long LL;

inline int read() {
    int x=0,f=1;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-1;
    for(;isdigit(ch);ch=getchar())x=x*10+ch-'0';return x*f;
}

const int N = 200005;
struct Edge { int to, nxt; double w; } e[N << 1];
int head[N], En, n, m;
double P[N], dp[N], sp[N], s[N], tp[N];
bool vis[N];

inline void add_edge(int u,int v,int w) {
    ++En; e[En].to = v, e[En].w = 1.0 * w / 100.0, e[En].nxt = head[u]; head[u] = En;
}
void Dijkstra() {
    priority_queue< pa, vector< pa >, greater< pa > > q;
    for (int i = 1; i <= n; ++i) dp[i] = 1e18, tp[i] = 1.0;
    q.push(pa(0, n)); dp[n] = 0; 
    while (!q.empty()) {
        int u = q.top().second; q.pop();
        if (vis[u]) continue; vis[u] = 1;
        fore(i, u, v) {
            double sum = s[v] + dp[u] * e[i].w * tp[v];
            double sump = sp[v] + e[i].w * tp[v];
            if (dp[v] > (sum + 1.0) / sump) {
                dp[v] = (sum + 1.0) / sump;
                s[v] += dp[u] * e[i].w * tp[v];
                sp[v] += e[i].w * tp[v];
                tp[v] *= (1 - e[i].w);
                q.push(pa(dp[v], v));
            }
        }
    }
    printf("%.3lf\n", dp[1]);
}
int main() {
    n = read(), m = read();
    for (int u, v, w, i = 1; i <= m; ++i) 
        u = read(), v = read(), w = read(), add_edge(v, u, w);
    Dijkstra();
    return 0;
}
相關文章
相關標籤/搜索