makeplex salon: 麻雀の問題
あなたのスキルで飯は食えるか? 史上最大のコーディングスキル判定 (1/2) - ITmedia エンタープライズ
をやってみた。
言語はHaskell。
たぶん、あってる。根拠はない。
とりあえず、各牌が無限にあると思うと、以下のシンプルなコードで良いと思う。
デカルトさんは言いました「困難は分割せよ」と。
もし、「頭」も「待ち」もなければ、牌は順子と刻子の組に分ければ良いだけ。
というわけで、「頭」と「待ち」を決める部分と、順子、刻子に分割する部分に分けて実装。
import Data.List mpart [] = [[]] mpart hs@(a:b:c:ds) = ms1 ++ ms2 where ms1 | a == b && b == c = map ([a,b,c]:) $ mpart ds | otherwise = [] ms2 | null $ [a+1,a+2] \\ hs = map ([a,a+1,a+2]:).mpart $ hs \\ [a,a+1,a+2] | otherwise = [] machi n hs os | null $ os \\ hs = concat [map ([os,a]++).mpart $ hs' \\ a | a <- as, null $ a \\ hs'] | otherwise = [] where as | length os == 2 = map (replicate 2) [1..n] | otherwise = [[]] hs' = hs \\ os machiAll n hs = concatMap (machi n hs) $ map return [1..n] ++ [[a,b]| a <- [1..n], b <- [a,a+1,a+2], b <= n] main :: IO () main = mapM_ print $ machiAll 9 [1,1,1,2,3,4,5,6,7,8,9,9,9]
ただ、実際は各牌は高々4枚だった記憶がある。つまり、[1,1,1,1,…] という牌を持っている状況では(11)を頭にして、[11]を待ちにするのは
ダメ。1が5枚あれば良いのだが。
というわけで、結構面倒な場合分けが必要な気がする。
その、各牌が4枚しかないことを考慮したコード。
import Data.List ((\\), group, nub) -- 順子、刻子のみの組合せに分割 mpart :: [Int] -> [[[Int]]] mpart [] = [[]] mpart hs@(a:b:c:ds) = ms1 ++ ms2 where ms1 | a == b && b == c = map ([a,b,c]:) $ mpart ds | otherwise = [] -- 刻子無し ms2 | null $ [a+1,a+2] \\ hs = map ([a,a+1,a+2]:).mpart $ hs \\ [a,a+1,a+2] | otherwise = [] -- 順子無し -- 「待ち」と頭の候補の列挙 candidates :: Int -> [Int] -> [([Int], [Int], [Int])] candidates n hs = ts ++ ks ++ ss1 ++ ss2 where ts = [([a], [], hs \\ [a]) | (a:_) <- filter ((4>).length).group $ hs] -- 単騎待ち ks = [([a,a], [b,b], hs \\ [a,a,b,b]) | a <- xs, b <- xs \\ [a]] -- 刻子での待ち where xs = map head.filter ((1<).length).group $ hs ss1 = [([a,a+1], [b,b], hs \\ [a,a+1,b,b]) | a <- nub hs, elem (a+1) hs, b <- nub hs, null $ [a,a+1,b,b] \\ hs, ((4>).length.filter (a+2==) $ hs) || ((4>).length.filter (a-1==) $ hs)] -- 順子での待ち、その1 ss2 = [([a,a+2], [b,b], hs \\ [a,a+2,b,b]) | a <- nub hs, elem (a+2) hs, b <- nub hs, null $ [a,a+2,b,b] \\ hs, (4>).length.filter (a+1==) $ hs] -- 順子での待ち、その2 machi :: Int -> [Int] -> [[[Int]]] machi n hs = filter (not.null).map f $ candidates n hs where f (os,as,hs) = concatMap ([os,as]++).mpart $ hs main :: IO () main = mapM_ print $ machi 9 [1,1,1,2,3,4,5,6,7,8,9,9,9]
場合わけの問題か。リストの内包表記がヒドイことになっている。効率?入力長は固定ですよ。