最新消息: USBMI致力于为网友们分享Windows、安卓、IOS等主流手机系统相关的资讯以及评测、同时提供相关教程、应用、软件下载等服务。

给出3个参数,N,M,K,怪兽有N滴血,等着英雄来砍自己,英雄每一次打击,都会让怪兽流失[0~M]的血量,到底流失多少?每一次在[0~M]上等概率的获取一个值,求K次打击之后,英雄把怪兽砍死的概率

IT圈 admin 29浏览 0评论

给出3个参数,N,M,K,怪兽有N滴血,等着英雄来砍自己,英雄每一次打击,都会让怪兽流失[0~M]的血量,到底流失多少?每一次在[0~M]上等概率的获取一个值,求K次打击之后,英雄把怪兽砍死的概率

问题描述:给出3个参数,N,M,K,怪兽有N滴血,等着英雄来砍自己,英雄每一次打击,都会让怪兽流失[0~M]的血量,到底流失多少?每一次在[0~M]上等概率的获取一个值,求K次打击之后,英雄把怪兽砍死的概率

例如:

int N =  6;
int M = 3;
int K =  5;

解决方案:采用暴力递归加动态规划的思想

1、暴力递归:核心代码如下

public class NanDaoKillMonster {public static void main(String[] args) {int N =  6;int M = 3;int K =  5;double ans1 = killMonster(N, M, K);double ans2 = killDp1(N, M, K);double ans3 = killDp2(N, M, K);System.out.println("ans1:"+ans1+";ans2:"+ans2+";ans3:"+ans3);}private static double killMonster(int N, int M, int K) {if(N < 1 || M < 1 || K < 1){//边界验证return 0;}long all = (long) Math.pow(M + 1,K);//理想情况下打击的次数long kill = killProcess(K,M,N);//kill怪兽的次数//两者相比就是kill的概率return (double)((double)kill/(double)all);}private static long killProcess(int times, int M, int hp) {if(times == 0){//还剩0次时,验证怪兽是否打死return hp <= 0? 1:0;}if(hp <= 0){//如果怪兽已经没血了,直接返回此种情况kill的次数return (long)Math.pow(M + 1,times);}int ways = 0;for(int i = 0;i <= M;i++){//每次次数减 1,血滴数减 i,进行暴力递归ways +=killProcess(times -1,M,hp - i);}return ways;}
}

2、动态规划思想:

2.1、二维数组如图:

这是血量大于0的情况图,小于0的情况大家可以脑补一下,总之,思路是一样的。 

2.2、核心代码如下:

public class NanDaoKillMonster {public static void main(String[] args) {int N =  6;int M = 3;int K =  5;double ans1 = killMonster(N, M, K);double ans2 = killDp1(N, M, K);double ans3 = killDp2(N, M, K);System.out.println("ans1:"+ans1+";ans2:"+ans2+";ans3:"+ans3);}private static double killDp1(int N, int M, int K) {if(N < 1 || M < 1 || K < 1){return 0;}long all = (long)Math.pow(M + 1,K);//理想情况下打击的次数long[][] dp = new long[K+ 1][N + 1];dp[0][0] = 1;//初始化次数for(int times = 1;times <= K;times++){//如果怪兽已经没血了,直接返回此种情况kill的次数dp[times][0] = (long)Math.pow(M + 1,times);for(int hp = 1;hp <= N;hp++){//遍历怪兽的血滴long ways = 0;/*** [0~M]比如 M = 3,* dp[5][10] 依赖dp[4][10],dp[4][9],dp[4][8],dp[4][7],* 因此有下面的循环依赖*/for(int i = 0;i <= M;i++){if(hp - i >= 0){//如果还有血滴,继续叠加ways += dp[times - 1][hp - i];}else {//如果怪兽已经没血了,直接返回此种情况kill的次数ways += (long)Math.pow(M + 1,times - 1);}}dp[times][hp] = ways;}}long kill = dp[K][N];//kill怪兽的次数//两者相比就是kill的概率return (double)((double)kill/(double) all);}}

3、升级版动态规划:

3.1、核心代码如下:

public class NanDaoKillMonster {public static void main(String[] args) {int N =  6;int M = 3;int K =  5;double ans1 = killMonster(N, M, K);double ans2 = killDp1(N, M, K);double ans3 = killDp2(N, M, K);System.out.println("ans1:"+ans1+";ans2:"+ans2+";ans3:"+ans3);}private static double killDp2(int N, int M, int K) {if(N < 1 || M < 1 || K < 1){return 0;}long all = (long)Math.pow(M + 1,K);long[][] dp = new long[K + 1][N + 1];dp[0][0] = 1;for(int times = 1;times <= K;times++){dp[times][0] = (long)Math.pow(M + 1,times);/*** 比如:* [0~M]假设 M = 7,* dp[5][10] 依赖 dp[4][10 ~ 3]八个位置的数值* dp[5][11] 依赖 dp[4][11 ~ 4]八个位置的数值 ,又等于dp[5][10] + dp[4][11] - dp[4][3]* 因此有下面的循环依赖* 总结通用公式:*      dp[i][j-1] = dp[i-1][j-1 ~ j-1-M]*      dp[i][j] = dp[i-1][j ~ j-M]*      dp[i][j] = dp[i][j-1] + dp[i-1][j] - dp[i-1][j-1-M]*      前提是血量 j>0,如果 j < 0,此种次数(M + 1)的 i即times 次方*/for(int hp = 1;hp <= N;hp++){dp[times][hp] = dp[times - 1][hp] + dp[times][hp - 1];if(hp - 1 - M >= 0){dp[times][hp] -= dp[times -1][hp - 1 -M];}else {dp[times][hp] -= Math.pow(M + 1,times - 1);}}}long kill = dp[K][N];//两者相比就是kill的概率return (double)((double)(kill)/(double)(all));}}

4、三种方案执行结果如下:

 

到此,该算法分享完毕,大家一定要多练习勤于思考,定会进步很快!

给出3个参数,N,M,K,怪兽有N滴血,等着英雄来砍自己,英雄每一次打击,都会让怪兽流失[0~M]的血量,到底流失多少?每一次在[0~M]上等概率的获取一个值,求K次打击之后,英雄把怪兽砍死的概率

问题描述:给出3个参数,N,M,K,怪兽有N滴血,等着英雄来砍自己,英雄每一次打击,都会让怪兽流失[0~M]的血量,到底流失多少?每一次在[0~M]上等概率的获取一个值,求K次打击之后,英雄把怪兽砍死的概率

例如:

int N =  6;
int M = 3;
int K =  5;

解决方案:采用暴力递归加动态规划的思想

1、暴力递归:核心代码如下

public class NanDaoKillMonster {public static void main(String[] args) {int N =  6;int M = 3;int K =  5;double ans1 = killMonster(N, M, K);double ans2 = killDp1(N, M, K);double ans3 = killDp2(N, M, K);System.out.println("ans1:"+ans1+";ans2:"+ans2+";ans3:"+ans3);}private static double killMonster(int N, int M, int K) {if(N < 1 || M < 1 || K < 1){//边界验证return 0;}long all = (long) Math.pow(M + 1,K);//理想情况下打击的次数long kill = killProcess(K,M,N);//kill怪兽的次数//两者相比就是kill的概率return (double)((double)kill/(double)all);}private static long killProcess(int times, int M, int hp) {if(times == 0){//还剩0次时,验证怪兽是否打死return hp <= 0? 1:0;}if(hp <= 0){//如果怪兽已经没血了,直接返回此种情况kill的次数return (long)Math.pow(M + 1,times);}int ways = 0;for(int i = 0;i <= M;i++){//每次次数减 1,血滴数减 i,进行暴力递归ways +=killProcess(times -1,M,hp - i);}return ways;}
}

2、动态规划思想:

2.1、二维数组如图:

这是血量大于0的情况图,小于0的情况大家可以脑补一下,总之,思路是一样的。 

2.2、核心代码如下:

public class NanDaoKillMonster {public static void main(String[] args) {int N =  6;int M = 3;int K =  5;double ans1 = killMonster(N, M, K);double ans2 = killDp1(N, M, K);double ans3 = killDp2(N, M, K);System.out.println("ans1:"+ans1+";ans2:"+ans2+";ans3:"+ans3);}private static double killDp1(int N, int M, int K) {if(N < 1 || M < 1 || K < 1){return 0;}long all = (long)Math.pow(M + 1,K);//理想情况下打击的次数long[][] dp = new long[K+ 1][N + 1];dp[0][0] = 1;//初始化次数for(int times = 1;times <= K;times++){//如果怪兽已经没血了,直接返回此种情况kill的次数dp[times][0] = (long)Math.pow(M + 1,times);for(int hp = 1;hp <= N;hp++){//遍历怪兽的血滴long ways = 0;/*** [0~M]比如 M = 3,* dp[5][10] 依赖dp[4][10],dp[4][9],dp[4][8],dp[4][7],* 因此有下面的循环依赖*/for(int i = 0;i <= M;i++){if(hp - i >= 0){//如果还有血滴,继续叠加ways += dp[times - 1][hp - i];}else {//如果怪兽已经没血了,直接返回此种情况kill的次数ways += (long)Math.pow(M + 1,times - 1);}}dp[times][hp] = ways;}}long kill = dp[K][N];//kill怪兽的次数//两者相比就是kill的概率return (double)((double)kill/(double) all);}}

3、升级版动态规划:

3.1、核心代码如下:

public class NanDaoKillMonster {public static void main(String[] args) {int N =  6;int M = 3;int K =  5;double ans1 = killMonster(N, M, K);double ans2 = killDp1(N, M, K);double ans3 = killDp2(N, M, K);System.out.println("ans1:"+ans1+";ans2:"+ans2+";ans3:"+ans3);}private static double killDp2(int N, int M, int K) {if(N < 1 || M < 1 || K < 1){return 0;}long all = (long)Math.pow(M + 1,K);long[][] dp = new long[K + 1][N + 1];dp[0][0] = 1;for(int times = 1;times <= K;times++){dp[times][0] = (long)Math.pow(M + 1,times);/*** 比如:* [0~M]假设 M = 7,* dp[5][10] 依赖 dp[4][10 ~ 3]八个位置的数值* dp[5][11] 依赖 dp[4][11 ~ 4]八个位置的数值 ,又等于dp[5][10] + dp[4][11] - dp[4][3]* 因此有下面的循环依赖* 总结通用公式:*      dp[i][j-1] = dp[i-1][j-1 ~ j-1-M]*      dp[i][j] = dp[i-1][j ~ j-M]*      dp[i][j] = dp[i][j-1] + dp[i-1][j] - dp[i-1][j-1-M]*      前提是血量 j>0,如果 j < 0,此种次数(M + 1)的 i即times 次方*/for(int hp = 1;hp <= N;hp++){dp[times][hp] = dp[times - 1][hp] + dp[times][hp - 1];if(hp - 1 - M >= 0){dp[times][hp] -= dp[times -1][hp - 1 -M];}else {dp[times][hp] -= Math.pow(M + 1,times - 1);}}}long kill = dp[K][N];//两者相比就是kill的概率return (double)((double)(kill)/(double)(all));}}

4、三种方案执行结果如下:

 

到此,该算法分享完毕,大家一定要多练习勤于思考,定会进步很快!

TypeError: array_keys(): Argument #1 ($array) must be of type array, null given in /www/wwwroot/www.usbmi.com/tmp/view_template_d8_htm_read.htm:240 Stack trace: #0 /www/wwwroot/www.usbmi.com/tmp/view_template_d8_htm_read.htm(240): array_keys() #1 /www/wwwroot/www.usbmi.com/tmp/route_read.php(204): include('...') #2 /www/wwwroot/www.usbmi.com/tmp/index.inc.php(129): include('...') #3 /www/wwwroot/www.usbmi.com/index.php(29): include('...') #4 {main}