Прикладной экземпляр для наборов (вложенных списков)

#haskell #applicative #category-theory

#haskell #прикладной #теория категорий

Вопрос:

В настоящее время я работаю над личным проектом для своего класса по математике и пытаюсь формализовать теорию множеств в Haskell. Набор, определенный в нашем классе, представляет собой произвольную вложенность элементов определенного юниверса. Я решил представить это как стандартный вложенный список де-факто:

 data Set a where
  Empty :: Set a
  Elem  ::   a   -> Set a -> Set a
  Set   :: Set a -> Set a -> Set a
  

Как ленивый программист на Haskell, я хочу написать экземпляры для всех стандартных классов типов.

Functor Экземпляр тривиален:

 instance Functor Set where
  fmap _ Empty        = Empty
  fmap f (Elem x set) = Elem (f x) set
  fmap f (Set s set)  = Set (fmap f s) $ fmap f set
  

Foldable и Traversable также относительно просты в реализации.

Я не застрял Applicative . pure также просто:

 instance Applicative Set where
  pure x = Elem x Empty
  

Тем не менее, я застрял на определении ap для вложенных списков.

 -- set has a monoid instance
(<*>) :: Set (a -> b) -> Set a -> Set b
Elem fx  fxs  <*> x = fmap fx x `mappend` (fxs <*> x)
Set  fxs fxss <*> x = Set ???
  

Для обычного, не вложенного списка прикладной экземпляр принимает декартово произведение каждой функции с каждым элементом и применяет его:

 fx <*> xs = [f x | f <- fx, x <- xs]
  

Каким-то образом вложенный список должен сохранять свою базовую структуру. Какой правильный экземпляр?

Ответ №1:

Ваш экземпляр почти правильный, еще несколько предложений:

 instance Applicative Set where
  pure x = Elem x Empty
  -- the cartesian product of the empty set and x is empty
  Empty         <*> x = Empty
  -- the cartesian product of x and the empty set is empty
  x             <*> Empty = Empty
  -- if you encounter a function, apply it to the entire list
  -- and append the result of the recursive call to the rest.
  Elem fx  fxs  <*> x = fmap fx x `mappend` (fxs <*> x)
  -- If you reach a level of nesting, go into the nested list
  -- and prepend that to the rest.
  Set  fxs fxss <*> x = Set (fxs <*> x) (fxss <*> x)
  

Этот экземпляр удовлетворяет всем прикладным законам:

 pure id  <*> x      = x
pure f   <*> pure x = pure $ f x
pure (.) <*> pure u <*> pure v <*> pure w = u <*> (v <*> w)
u        <*> pure y = pure ($ y) <*> u