Maximum Random Walk


Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 505    Accepted Submission(s): 279



Problem Description


Consider the classic random walk: at each step, you have a 1/2 chance of taking a step to the left and a 1/2 chance of taking a step to the right. Your expected position after a period of time is zero; that is, the average over many such random walks is that you end up where you started. A more interesting question is what is the expected rightmost position you will attain during the walk.


 



Input


The first line of input contains a single integer P, (1 <= P <= 15), which is the number of data sets that follow. Each data set should be processed identically and independently.

Each data set consists of a single line of input consisting of four space-separated values. The first value is an integer K, which is the data set number. Next is an integer n, which is the number of steps to take (1 <= n <= 100). The final two are double precision floating-point values L and R
which are the probabilities of taking a step left or right respectively at each step (0 <= L <= 1, 0 <= R <= 1, 0 <= L+R <= 1). Note: the probably of not taking a step would be 1-L-R.


 



Output


For each data set there is a single line of output. It contains the data set number, followed by a single space which is then followed by the expected (average) rightmost position you will obtain during the walk, as a double precision floating point value to four decimal places.


 



Sample Input


3 1 1 0.5 0.5 2 4 0.5 0.5 3 10 0.5 0.4


 



Sample Output


1 0.5000 2 1.1875 3 1.4965


 


大体题意:告诉你向左走、向右走、不动的概率,和总共的移动步数,让你计算在指定步数内,到达最右端的数学期望。


分析:


比赛没有做出来,还不会概率DP,通过这个题目了解下概率DP。


DP[i][j][k]表示 第i步,在j位置处,所到达最远距离k的概率!


因为第几步计算的话,只需要前一步就够了,因此一维可以改成滚动dp。


然后分三步:


向左走,向右走,不动!


最后计算期望用概率乘以坐标数值即可

#include<cstdio>
#include<cstring>
#include<algorithm>
#define mem(x) memset(x,0,sizeof(x));
using namespace std;
const int maxn = 200 + 10;
double dp[2][maxn][maxn];
int main(){
	int T,cnt;
	scanf("%d",&T);
	while(T--){
		double l,r;
		int n,u=0,v;
		scanf("%d%d%lf%lf",&cnt,&n,&l,&r);
		int len = n;
		n*=2;//坐标整体向右移动,防止出现负坐标! 
		mem(dp[u]);
		dp[u][len][len] = 1.0;
		for (int i = 0; i < len; ++i){//控制步数! 
			v = u ^ 1;//滚动dp; 
			mem(dp[v]);
			for (int j = 1; j <= n; ++j){
				for (int k = 1; k <= n; ++k){
					dp[v][j+1][max(k,j+1)] += dp[u][j][k] * r;//往右走可能会超过k 
					dp[v][j][k] += dp[u][j][k] * (1.0-l-r);//不动! 
					dp[v][j-1][k] += dp[u][j][k] * l;//往左走! 
				}
			}
			u = v;
		}
		double ans = 0;
		for (int i = 0; i <= n; ++i)
			for (int j = len; j <= n; ++j)
				ans += dp[u][i][j] * (j-len);
		printf("%d %.4f\n",cnt,ans);
	}
	return 0;
}