1.题目描述:点击打开链接
2.解题思路:本题利用矩阵快速幂+概率dp解决。根据题意可以画出来一个状态转移图,根据状态转移图不难得到一步转移概率矩阵,接下来的问题是:如何求解d步之内(包括d)均无法从其他点走到结点u的概率。
首先,既然无法到达结点u,那么出发的时候就不能选择该点。其次,为了使其他结点也无法到达结点u,可以将一步转移概率矩阵中跟结点u有关的概率全部置零。即表示u结点出发无法到达其他结点,其他结点也均无法到达u结点,这样就相当于把u结点从状态转移图中暂时删去了。然后根据马氏链的知识,假设一步转移概率矩阵为A,那么d步转移概率矩阵为A^d。再考虑到选择出发点也构成一个矩阵B,那么最终d步时候的概率矩阵为B=B*(A^d)。最终得到的B矩阵中的B[i][j]即为i经过d步走到j的概率。
这里有一个小技巧,为了便于计算B矩阵,可以在初始化的时候只将第0行的每个结点都置为1/n(结点i仍然置零),表示结点j被选为出发点的概率,这样,最终计算出的B矩阵也只在第0行有结果,表示其他点到达结点j的概率,最后,只需要把这些概率求和,即可得到到达除结点i之外其他结点的概率,即不到达结点i的概率。
本题的时间复杂度为O(N^4logd)。
3.代码:
#define _CRT_SECURE_NO_WARNINGS #include<iostream> #include<algorithm> #include<string> #include<sstream> #include<set> #include<vector> #include<stack> #include<map> #include<queue> #include<deque> #include<cstdlib> #include<cstdio> #include<cstring> #include<cmath> #include<ctime> #include<cctype> #include<functional> using namespace std; #define me(s) memset(s,0,sizeof(s)) #define pb push_back typedef long long ll; typedef unsigned int uint; typedef unsigned long long ull; typedef pair <int, int> P; const int N = 10000 + 10; int n, m, d; struct Node { double v[55][55]; void init() { memset(v, 0, sizeof(v)); } Node operator*(Node r) { Node ans; ans.init(); for (int i = 0; i < n; i++) for (int j = 0; j < n; j++) for (int k = 0; k < n; k++) ans.v[i][j] += v[i][k] * r.v[k][j]; return ans; } }a; Node operator^(Node m, int d) { Node ans; ans.init(); for(int i=0;i<n;i++)ans.v[i][i]=1; while(d>0) { if(d&1)ans=ans*m; m=m*m; d>>=1; } return ans; } int main() { int T; scanf("%d", &T); while (T--) { scanf("%d%d%d", &n, &m, &d); int u, v; vector<int>g[N]; for (int i = 0; i<m; i++) { scanf("%d%d", &u, &v); u--;v--; g[u].push_back(v); g[v].push_back(u); } a.init(); for (int i = 0; i < n; i++) if (!g[i].empty()) { int len = g[i].size(); for (int j = 0; j<len; j++) { int id = g[i][j]; a.v[i][id] = (double)1.0 / len;//求一步转移概率矩阵 } } for(int i=0;i<n;i++) { Node tmp=a; for(int j=0;j<n;j++) tmp.v[i][j]=tmp.v[j][i]=0;//将与结点i有关的概率暂时置零 Node ans; ans.init(); for(int j=0;j<n;j++) ans.v[0][j]=1.0/n; ans.v[0][i]=0; //不从结点i出发 ans=ans*(tmp^d); double res=0; for(int j=0;j<n;j++) res+=ans.v[0][j];//求和,得到d步之内均不经过结点i的概率 printf("%.10lf\n",res); } } }
版权声明:本文为博主原创文章,未经博主允许不得转载。
时间: 2024-10-10 11:03:49