我能赢吗
问题描述:
在 “100 game” 这个游戏中,两名玩家轮流选择从 1 到 10 的任意整数,累计整数和,先使得累计整数和达到 100 的玩家,即为胜者。
如果我们将游戏规则改为 “玩家不能重复使用整数” 呢?
例如,两个玩家可以轮流从公共整数池中抽取从 1 到 15 的整数(不放回),直到累计整数和 >= 100。
给定一个整数 maxChoosableInteger (整数池中可选择的最大数)和另一个整数 desiredTotal(累计和),判断先出手的玩家是否能稳赢(假设两位玩家游戏时都表现最佳)?
你可以假设 maxChoosableInteger 不会大于 20, desiredTotal 不会大于 300。
示例:
输入:
maxChoosableInteger = 10
desiredTotal = 11
输出:
false
解释:
无论第一个玩家选择哪个整数,他都会失败。
第一个玩家可以选择从 1 到 10 的整数。
如果第一个玩家选择 1,那么第二个玩家只能选择从 2 到 10 的整数。
第二个玩家可以通过选择整数 10(那么累积和为 11 >= desiredTotal),从而取得胜利.
同样地,第一个玩家选择任意其他整数,第二个玩家都会赢。
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/can-i-win
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解决方案:
对于博弈类问题,一般做法都是定于fun()为先手能否获胜。fun计算过程中,枚举出先手方的所有可能,如果存在后手方不能获胜,则此时先手方采取该决策即可获胜,若对于先手方的所有选择后手方均可获胜,则此时先手方不能获胜返回false。
对于该问题,由于题目要求不放回的拿元素,因此需定义一布尔型的visted数组用于存储该元素是否被拿。其dfs实现代码如下:
class Solution {
int total = 0;
public boolean canIWin(int maxChoosableInteger, int desiredTotal) {
this.total = desiredTotal;
boolean[] visted = new boolean[maxChoosableInteger + 1];
return step(visted, 0);
}
// sum 为当前的和
public boolean step(boolean[] visted, int sum){
for(int i = visted.length - 1; i > 0; i--){
if(visted[i]){
continue;
}
if(sum + i >= total){
return true;
}
visted[i] = true;
if(!step(visted, sum + i)){
visted[i] = false;
return true;
}
visted[i] = false;
}
return false;
}
}
上述解法的时间复杂度为O(n!),其中n为能拿元素的最大值。
由于该过程转移顺序不能确定,因此采用dfs + 记忆集的方式求解。
由于此时的状态为一布尔型的数组,难以存储,因此使用状态压缩的方式,由于题目说道maxChoosableInteger 总是小于20的,因此可以使用一整型变量status,当前位为0表示可用,为1表示不可用。由于是从1开始的,因此i值在status中对应的是i - 1位。由于通过status可以唯一的确定状态,因此不需要加入sum(其实可以通过status得出此时的sum)。
代码如下:
class Solution {
int total = 0;
int maxInte = 0;
public boolean canIWin(int maxChoosableInteger, int desiredTotal) {
if(maxChoosableInteger * (maxChoosableInteger + 1) / 2 < desiredTotal){
return false;
}
this.total = desiredTotal;
this.maxInte = maxChoosableInteger;
int status = 0; // 当前位为0表示未取 1表示已取
Map<Integer, Boolean> map = new HashMap<>();
return step(status, map, 0);
}
public boolean step(int status, Map<Integer, Boolean> map, int sum){
if(map.containsKey(status)){
return map.get(status);
}
boolean ans = false;
for(int i = 0; i < maxInte; i++){
if((status >> i & 1) == 1){
continue;
}
if(sum + i + 1 >= total){
ans = true;
break;
}
if(!step(status | (1 << i), map, sum + i + 1)){
ans = true;
break;
}
}
map.put(status, ans);
return ans;
}
}
其时间复杂度降为O(2^N)
- RepeatMasker安装和使用
- 漏洞追踪:如何检查并修复GHOST(幽灵)漏洞
- 锁的实现原理解锁的实现加锁的实现
- 幽灵漏洞(GHOST)影响大量Linux操作系统及其发行版(更新修复方案)
- openfire中mysql的前期设置
- android中最先被执行的activity
- Spring Boot微服务架构入门
- 测序文章数据上传找哪里
- Volatile实现原理实现原子性happens-before关系从happends-before规则分析可见性编译器层面实现可见性处理器层面实现可见性
- java中的toString方法
- 简单可视化-送你一双发现美的眼睛
- 享元模式
- 揭秘:针对中国移动用户的强大网银木马剖析
- 从源代码到Runtime发生的重排序编译器重排序指令重排序内存系统重排序阻止重排序
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- lettuce和hbase中对netty的使用你都了解吗?
- SAS-新规试行下诞生的程序
- AIOHTTP的实战(一)
- JMeter二次开发插件实例
- 「R」Obtain RNAseq Values for a Specific Gene in Xena Database
- 使用JMeter测试TCP协议
- SQL使用(一):如何使用SQL语句去查询第二高的值
- OpenCV加载图片显示对应类型(位深度)方法
- 小程序快速入门教程(登录注册、开发工具、文件及结构介绍)
- 面试题-统计字符出现最大次数
- pytorch中文语言模型bert预训练代码
- React环境搭建
- leetcode - 旋转数组的最小数字
- 笔记101 | 文件的压缩与解压笔记
- 诸葛亮vs司马懿,排序算法大战谁能笑到最后?