zoukankan      html  css  js  c++  java
  • BZOJ 3143: [Hnoi2013]游走 [概率DP 高斯消元]

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

     

    输入保证30%的数据满足N≤10,100%的数据满足2≤N≤500且是一个无向简单连通图。


    做过一道类似的后感觉比较简单了

    求$f[i]$到每个点的概率

    $f[i]=sumlimits_{(i,j) in E}{f[j]*frac{1}{d[j]}}$

    $f[1]$额外加上$1$

    $f[n]=0$因为到$n$后就不走了没必要用$n$的概率

    然后就可以得到通过一条边的概率啦,贪心分配即可

    然后BZOJ数据太弱了....洛谷的数据在消元时还要判断系数$<eps$

    PS:这种题应该保证有解吧

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    #include <cmath>
    using namespace std;
    typedef long long ll;
    const int N=505;
    double eps=1e-7;
    inline int read(){
        char c=getchar();int x=0;
        while(c<'0'||c>'9'){c=getchar();}
        while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
        return x;
    }
    int n,m,u,v;
    int d[N];
    double p[N],a[N][N];
    struct edge{
        int v,ne,u;
    }e[N*N<<1];
    int h[N],cnt=0;
    inline void ins(int u,int v){
        cnt++;
        e[cnt].u=u;e[cnt].v=v;e[cnt].ne=h[u];h[u]=cnt;
        cnt++;
        e[cnt].u=v;e[cnt].v=u;e[cnt].ne=h[v];h[v]=cnt;
    }
    void buildEquation(){
        for(int i=1;i<n;i++){
            a[i][i]=1;int j;
            for(int k=h[i];k;k=e[k].ne) j=e[k].v,a[i][j]=-1.0/d[j];
        }
        a[1][n+1]=1;
        a[n][n]=1;a[n][n+1]=0;
    }
    void GaussElimination(){
        for(int i=1;i<=n;i++){
            int r=i;
            for(int j=i+1;j<=n;j++) if(abs(a[j][i])>abs(a[r][i])) r=j;
            if(r!=i) for(int k=1;k<=n+1;k++) swap(a[r][k],a[i][k]);
            for(int j=i+1;j<=n;j++) if(abs(a[j][i])>eps){
                double t=a[j][i]/a[i][i];
                for(int k=i;k<=n+1;k++) a[j][k]-=t*a[i][k];
            }
        }
        for(int i=n;i>=1;i--){
            for(int j=n;j>i;j--) a[i][n+1]-=a[i][j]*a[j][n+1];
            a[i][n+1]/=a[i][i];
            p[i]=a[i][n+1];
        }
    }
    double f[N*N];
    void solve(){
        for(int i=1;i<=m;i++){
            int u=e[i<<1].u,v=e[i<<1].v;
            f[i]=p[u]/d[u]+p[v]/d[v];
        }
        sort(f+1,f+1+m);
        double ans=0;
        for(int i=1;i<=m;i++) ans+=(m-i+1)*f[i];
        printf("%.3lf",ans);
    }
    int main(){
        freopen("in","r",stdin);
        n=read();m=read();
        for(int i=1;i<=m;i++) u=read(),v=read(),ins(u,v),d[u]++,d[v]++;
        buildEquation();
        GaussElimination();
        solve();
    }
     
     
  • 相关阅读:
    京东POP店铺使用京东物流,如何拦截订单
    京东POP店铺使用京东物流切仓操作方法
    京东评价系统更新190301
    京东考试题目答案,每次顺序都不一样,一气之下,全部复制出来,满分过
    win10 去掉资源管理器左侧的Creative Cloud Files
    java 截取字符串获取子字符串
    在jsp中如何用request中获取后台传来的数据?
    java 中怎样获取input的值
    获取${}中的值? 比如说var a=${date },无法取出date中的值
    ${}
  • 原文地址:https://www.cnblogs.com/candy99/p/6418877.html
Copyright © 2011-2022 走看看