चर्च-एन्कोडेड सूचियों का उपयोग करने के लिए आपको DList
के बजाय एक विकल्प पर विचार करना चाहिए। विचार यह है कि आप एक सूची को एक अपारदर्शी मान के रूप में दर्शाते हैं जो जानता है कि सूची में foldr
निष्पादित करने के तरीके को कैसे। यह RankNTypes
एक्सटेंशन का उपयोग कर की आवश्यकता है:
{-# LANGUAGE RankNTypes #-}
import Prelude
import Control.Applicative
import Data.Foldable (Foldable)
import qualified Data.Foldable as F
import Data.Traversable (Traversable)
import qualified Data.Traversable as T
-- | Laws:
--
-- > runList xs cons nil == xs
-- > runList (fromList xs) f z == foldr f z xs
-- > foldr f z (toList xs) == runList xs f z
newtype ChurchList a =
ChurchList { runList :: forall r. (a -> r -> r) -> r -> r }
-- | Make a 'ChurchList' out of a regular list.
fromList :: [a] -> ChurchList a
fromList xs = ChurchList $ \k z -> foldr k z xs
-- | Turn a 'ChurchList' into a regular list.
toList :: ChurchList a -> [a]
toList xs = runList xs (:) []
-- | We can construct an empty 'ChurchList' without using a @[]@.
nil :: ChurchList a
nil = ChurchList $ \_ z -> z
-- | The 'ChurchList' counterpart to '(:)'. Unlike 'DList', whose
-- implementation uses the regular list type, 'ChurchList' doesn't
-- rely on it at all.
cons :: a -> ChurchList a -> ChurchList a
cons x xs = ChurchList $ \k z -> k x (runList xs k z)
-- | Append two 'ChurchList's. This runs in O(1) time. Note that
-- there is no need to materialize the lists as @[a]@.
append :: ChurchList a -> ChurchList a -> ChurchList a
append xs ys = ChurchList $ \k z -> runList xs k (runList ys k z)
-- | Map over a 'ChurchList'. No need to materialize the list.
instance Functor ChurchList where
fmap f xs = ChurchList $ \k z -> runList xs (\x xs' -> k (f x) xs') z
-- | The 'Foldable' instance is trivial, given the 'ChurchList' law.
instance Foldable ChurchList where
foldr f z xs = runList xs f z
instance Traversable ChurchList where
traverse f xs = runList xs step (pure nil)
where step x rest = cons <$> f x <*> rest
इस के लिए नकारात्मक पक्ष यह एक ChurchList
-folding एक ChurchList
सस्ती है के लिए कोई कुशल tail
आपरेशन वहाँ है, लेकिन बार-बार पूंछ लेने महंगा है कि है ...
इसके अलावा
स्रोत
2013-03-23 23:38:56
को Sjoerd का जवाब, एक डीएलआईस्ट ** निर्माण ** के लिए केवल कुशल है - यदि आपने अपनी सूची बनाई है और इसे संसाधित करना चाहते हैं, तो आपको इसे 'toList' के साथ गुप्त करना चाहिए और फिर नियमित सूची को संसाधित करना चाहिए। –
तो हम बस 'फोल्ड (डीएल एफ) = गुना (एफ [])' क्यों परिभाषित नहीं करते हैं? हम इस बारे में भूल सकते हैं कि कैसे 'डीएलआईस्टी लागू किए जाते हैं और उन्हें तत्वों के अनुक्रम के कुछ प्रतिनिधित्व के रूप में देखते हैं, और फिर' फोल्डबल 'को लागू करने का अर्थ समझ में आता है। इस तरह 'फेंक्टर' और 'ट्रैवर्सबल' को लागू करने में शायद कुछ नुकसान होंगे, लेकिन 'फोल्डबल' काफी उचित लगता है। –
हाँ, 'Foldable' बहुत खराब नहीं हो सकता है, पैकेज पहले से ही' foldr' है और यह सब के बाद पर्याप्त है। मुझे लगता है कि इसे लागू नहीं किया गया कारण है क्योंकि आखिरी अपडेट 200 9 में था, जब 'Foldable' अभी तक एक प्रसिद्ध प्रकार की श्रेणी नहीं थी। –