464. Can I Win

 public boolean canIWin(int maxChoosableInteger, int desiredTotal) {
        //两种特殊情况,一是目标比最大值小,二是把所有可能取值加起来都没有目标大
        if (desiredTotal <= maxChoosableInteger)
            return true;
        if (maxChoosableInteger * (maxChoosableInteger+1)/2 < desiredTotal)
            return false;
        //map存的是如果以当前目标为开始值,先手能不能赢,map不是存的哪些数字用过,是存的初始值状态
        Map<String,Boolean> map = new HashMap<>();
        //为了处理方便,这里下标从1开始,正好和数字对应
        int[] state = new int[maxChoosableInteger+1];
        return dp(map,state,desiredTotal,maxChoosableInteger);
    }
    private boolean dp(Map<String,Boolean> map,int[] state,int total,int max)
    {
        String str = Arrays.toString(state);
        //为了节省时间,如果前边有这种状态,那么直接从map中取出来
        if (map.containsKey(str))
            return map.get(str);
        //遍历所有当前可以取的数
        for (int i = 1;i <= max;i++)
        {
            if (state[i] == 0)
            {
                //记录这个数用过了
                state[i] = 1;
                //判断当前这个目标能不能赢,如果能,如果是最外层,说明能赢,直接返回。如果不是最外层,则判断的上一步走的对不对,赢了就是走错了
                if (total <= i || !dp(map,state,total-i,max))
                {
                    //记录下如果初始值是这个状态的话,先手是可以胜利的
                    map.put(str,true);
                    //返回之前一定把当前这个数重置,因为上一次的时候这个数还没用,还原上次的初始值状态,数字用的情况代表着当时的目标状态
                    state[i] = 0;
                    return true;
                }
                //能进行到这里说明如果走当前这一步,说明会输,不走这一步继续尝试,所以重置状态
                state[i] = 0;
            }
        }
        //走到这里说明遍历完了都没返回成功,说明必输,记录状态返回
        map.put(str,false);
        return false;
    }

 

posted @ 2017-10-19 19:46  stAr_1  阅读(431)  评论(0编辑  收藏  举报