-- | -- Module: Math.NumberTheory.Recurrences.Pentagonal -- Copyright: (c) 2018 Alexandre Rodrigues Baldé -- Licence: MIT -- Maintainer: Alexandre Rodrigues Baldé -- -- Values of . -- {-# LANGUAGE TypeApplications #-} module Math.NumberTheory.Recurrences.Pentagonal ( partition ) where import qualified Data.Chimera as Ch import Data.Vector (Vector) import Numeric.Natural (Natural) -- | Infinite list of generalized pentagonal numbers. -- Example: -- -- >>> take 10 pents -- [0,1,2,5,7,12,15,22,26,35] pents :: (Enum a, Num a) => [a] pents = interleave (scanl (\acc n -> acc + 3 * n - 1) 0 [1..]) (scanl (\acc n -> acc + 3 * n - 2) 1 [2..]) where interleave :: [a] -> [a] -> [a] interleave (n : ns) (m : ms) = n : m : interleave ns ms interleave _ _ = [] -- | @p(n) = p(n-1) + p(n-2) - p(n-5) - p(n-7) + p(n-11) + ...@, where @p(0) = 1@ -- and @p(k) = 0@ for a negative integer @k@. Uses a @Chimera@ from the -- @chimera@ package to memoize previous results. partitionF :: Num a => (Word -> a) -> Word -> a partitionF _ 0 = 1 partitionF f n = sum $ zipWith (*) (cycle [1, 1, -1, -1]) $ map (f . (n -)) $ takeWhile (<= n) $ tail pents -- | Infinite zero-based table of . -- -- >>> take 10 partition -- [1,1,2,3,5,7,11,15,22,30] -- -- >>> :set -XDataKinds -- >>> import Data.Mod -- >>> partition !! 1000 :: Mod 1000 -- (991 `modulo` 1000) partition :: Num a => [a] partition = Ch.toList $ Ch.tabulateFix @Vector partitionF {-# SPECIALIZE partition :: [Int] #-} {-# SPECIALIZE partition :: [Word] #-} {-# SPECIALIZE partition :: [Integer] #-} {-# SPECIALIZE partition :: [Natural] #-}