zoukankan      html  css  js  c++  java
  • BZOJ3143 [Hnoi2013]游走 【高斯消元】

    题目

    一个无向连通图,顶点从1编号到N,边从1编号到M。
    小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这条边的编号的分数。当小Z 到达N号顶点时游走结束,总分为所有获得的分数之和。
    现在,请你对这M条边进行编号,使得小Z获得的总分的期望值最小。

    输入格式

    第一行是正整数N和M,分别表示该图的顶点数 和边数,接下来M行每行是整数u,v(1≤u,v≤N),表示顶点u与顶点v之间存在一条边。 输入保证30%的数据满足N≤10,100%的数据满足2≤N≤500且是一个无向简单连通图。

    输出格式

    仅包含一个实数,表示最小的期望值,保留3位小数。

    输入样例

    3 3

    2 3

    1 2

    1 3

    输出样例

    3.333

    解释

    边(1,2)编号为1,边(1,3)编号2,边(2,3)编号为3。

    题解

    我们算出每个边的期望经过次数wi,按大到小分别赋1到M就是最优解
    wi=xudegree[u]+xvdegree[v]
    xi指每个点的经过次数
    我们只要算出xi就可以了
    对于每个点i,我们可以列出一个方程:
    xi=j>ijxjdegree[j]
    N个方程,N个未知数,高斯消元可求解

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<cmath>
    #define eps 1e-9
    #define LL long long int
    #define REP(i,n) for (int i = 1; i <= (n); i++)
    #define Redge(u) for (int k = h[u]; k != -1; k = ed[k].nxt)
    using namespace std;
    const int maxn = 505,maxm = 500005,INF = 1000000000;
    inline int RD(){
        int out = 0,flag = 1; char c = getchar();
        while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
        while (c >= 48 && c <= 57) {out = (out << 1) + (out << 3) + c - '0'; c = getchar();}
        return out * flag;
    }
    int G[maxn][maxn],N,M,de[maxn];
    double A[maxn][maxn],X[maxn];
    struct EDGE{int a,b; double w;}e[maxm];
    inline bool operator <(const EDGE& a,const EDGE& b){return a.w > b.w;}
    void gauss(){
        for (int i = 1; i <= N; i++){
            double t = A[i][i];
            if (fabs(t) < eps) continue; 
            for (int j = i; j <= N + 1; j++) A[i][j] /= t;
            for (int j = i + 1; j <= N; j++){
                t = A[j][i];
                for (int k = i; k <= N + 1; k++)
                    A[j][k] -= A[i][k] * t;
            }
        }
        for (int i = N; i > 0; i--){
            for (int j = i + 1; j <= N; j++)
                A[i][N + 1] -= X[j] * A[i][j];
            X[i] = A[i][N + 1] / A[i][i];
        }
    }
    int main(){
        N = RD(); M = RD();
        REP(i,M){
            de[e[i].a = RD()]++; de[e[i].b = RD()]++;
            G[e[i].a][e[i].b] = G[e[i].b][e[i].a] = true;
        }
        for (int i = 1; i < N; i++) A[i][i] = -1;
        REP(i,M){
            A[e[i].a][e[i].b] += 1.0 / de[e[i].b];
            A[e[i].b][e[i].a] += 1.0 / de[e[i].a];
        }
        for (int i = 1; i <= N; i++) A[i][N + 1] = 0;
        A[1][N + 1] = -1; A[N][N + 1] = 1; A[N][N] = 1;
        gauss();
        REP(i,M) e[i].w = X[e[i].a] / de[e[i].a] + X[e[i].b] / de[e[i].b];
        sort(e + 1,e + 1 + M);
        double ans = 0;
        REP(i,M) ans += e[i].w * i;
        printf("%.3lf",ans);
        return 0;
    }
    
  • 相关阅读:
    android 数据绑定(6)自定义绑定方法、双向数据绑定
    android apk瘦身(2) R8编译器:压缩代码、压缩资源、优化代码
    Kotlin 泛型
    android 数据绑定(5) kotlin 的binding bug
    android 数据绑定(4)实用特性及疑惑:使用控件、格式化@string/xxx、对象传递、双向数据绑定
    android apk瘦身(1) 使用矢量图 和 webp,去掉多余cpu架构的库文件
    android 数据绑定(3)自动更新UI
    android 数据绑定(2)绑定表达式
    android 数据绑定(1)Ativity、Fragment、Item绑定数据源
    react.js中render的return的坑
  • 原文地址:https://www.cnblogs.com/Mychael/p/8282731.html
Copyright © 2011-2022 走看看