Alice has a hand
of cards, given as an array of integers.html
Now she wants to rearrange the cards into groups so that each group is size W
, and consists of W
consecutive cards.git
Return true
if and only if she can.github
Example 1:算法
Input: hand = [1,2,3,6,2,3,4,7,8], W = 3 Output: true Explanation: Alice's can be rearranged as hand[1,2,3],[2,3,4],[6,7,8]
Example 2:spa
Input: hand = [1,2,3,4,5], W = 4 Output: false Explanation: Alice's can't be rearranged into groups of hand4
Note:code
1 <= hand.length <= 10000
0 <= hand[i] <= 10^9
1 <= W <= hand.length
这道题说是咱们在打扑克牌,是否能将手里的牌都以顺子的形式出完。在打拐3挖坑或者斗地主的时候,顺子牌在后期的威力是蛮大的,某些人手里憋了一把牌,结果到后期找个机会上手了以后,直接一把甩完,看的手中只有一把单牌的博主是目瞪口呆。其实到了后期,你们手中都没啥牌了的时候,就算是很小的连牌,也不必定能要得起,而划单是最没前途的出法,因此要尽可能将手中的牌都组成顺子丢出去。这里给了一个W,规定了顺子的最小长度,那么咱们就拿例子1来模拟下打牌吧,首先摸到了牌以后,确定要先整牌,按从小到大的顺序排列,这里就不考虑啥3最大,4最小啥的,就统一按原始数字排列吧:htm
1 2 2 3 3 4 6 7 8blog
好,下面要来组顺子,既然这里是3张可连,那么从最小的开始连呗。其实这道题仍是简化了许多,真正打牌的时候,即使是3张起连,那么连4张5张都是能够的,能够这里限定了只能连W张,就使得题目变简单了。咱们用贪婪算法就能够了,首先从1开始,那么必定得有2和3,才能起连,若少了任何一个,均可以直接返回false,好那么取出这三张后,手里还有:排序
2 3 4 6 7 8leetcode
那么从当前手里的最小的牌2开始起连,那么手里必需要有3和4,若少了任何一个,均可以直接返回 false,好那么取出这三张后,手里还有:
6 7 8
从当前手里的最小的牌6开始起连,那么手里必需要有7和8,若少了任何一个,均可以直接返回 false,好那么取出这三张后,手里没牌了,咱们成功的连完了全部的牌。分析这个过程,不难发现,因为牌能够重复,因此要统计每张牌出现的次数,同时还要给牌按大小排序,用 TreeMap 来创建牌的大小和其出现次数之间的映射就最好不过了,利用了其能够按 key 值排序的特色。首先遍历手中牌,创建映射。而后开始 while 循环,条件是 TreeMap 不为空,而后去除最小的那张牌,而后遍历能组成顺子的W张牌,若没有直接返回 true,有的话,则映射值自减1,若映射值为0了,则从 TreeMap 中移除该映射对儿便可,while 循环退出后返回 true,参见代码以下:
解法一:
class Solution { public: bool isNStraightHand(vector<int>& hand, int W) { map<int, int> m; for (int i : hand) ++m[i]; while (!m.empty()) { int start = m.begin()->first; for (int i = 0; i < W; ++i) { if (!m.count(start + i)) return false; if (--m[start + i] == 0) m.erase(start + i); } } return true; } };
咱们也能够不对 TreeMap 进行删除操做,而是直接修改其映射值,在创建好映射对儿以后,不用 while 循环,而是遍历全部的映射对儿,若某个映射值为0了,直接跳过。而后仍是遍历能组成顺子的W张牌,若某张牌出现的次数小于顺子起始位置的牌的个数,则直接返回 false,由于确定会有起始牌剩余,没法全组成顺子,这样还避免了上面解法中一张一张减的操做,提升了运算效率。而后映射值减去起始牌的个数,最后 for 循环退出后,返回 true,参见代码以下:
解法二:
class Solution { public: bool isNStraightHand(vector<int>& hand, int W) { map<int, int> m; for (int i : hand) ++m[i]; for (auto a : m) { if (a.second == 0) continue; for (int i = a.first; i < a.first + W; ++i) { if (m[i] < a.second) return false; m[i] = m[i] - a.second; } } return true; } };
Github 同步地址:
https://github.com/grandyang/leetcode/issues/846
参考资料:
https://leetcode.com/problems/hand-of-straights/
https://leetcode.com/problems/hand-of-straights/discuss/135700/Short-Java-solution!
https://leetcode.com/problems/hand-of-straights/discuss/135598/C%2B%2BJavaPython-O(MlogM)-Complexity