Problem 113

http://projecteuler.net/index.php?section=problems&id=113

とりあえず、再帰式を考えて、DPにした。
しかし、かなりきれいな再帰式なので手計算できそうではある。が、いいや。

import Data.Array.IArray

n = 100
mkArray :: (Ix i)=> (i -> Integer)-> (i,i)->Array i Integer
mkArray f b = listArray b. map f . range $ b

descend = mkArray dsc ((1,0),(n,8))
dsc (1,m) = m+1
dsc (d'+1,m) = sum [descend!(d',k)|k<-[0..m]]

ascend = mkArray asc ((1,2),(n,10))
asc (1,m) = 10 - m
asc (d'+1,m) = sum [ascend!(d',k)|k<-[m..9]]

notBouncy = mkArray b (1,n)
    where b 1 = 10
          b (d+1) = sum [descend!(k,l-1) + ascend!(k,l+1) | k<-[1..d],l<-[1..9]] + 9 + notBouncy! d

main = print$ notBouncy ! n - 1
追記
notBouncy n = descend + ascend - 10
    where descend = product [n+1..n+9] `div` product [1..9] -- use [0..9]
          ascend = product [n+1..n+8] `div` product [1..8] -- use [1..9] , no leading 0
p113 n = sum.map notBouncy $ [1..n]

重複組み合わせでした.