我发现自己需要的数据结构可能包含A
,可能是B
,而且肯定是其中之一。如果我要破解这个东西的一般数据类型,它可能看起来像:
data OneOrBoth a b = A a | B b | AB a b
maybeA :: OneOrBoth a b -> Maybe a
maybeB :: OneOrBoth a b -> Maybe b
eitherL :: OneOrBoth a b -> Either a b -- Prefers a
eitherR :: OneOrBoth a b -> Either a b -- Prefers b
hasBoth, hasExactlyOne, hasA, hasB :: OneOrBoth a b -> Bool
此数据结构是否有名称?是否有规范的方法来处理Haskell中的一个或两个结构?
答案 0 :(得分:43)
这可以用来表示两个值的组合,其中 如果任一输入是,则定义组合。代数,类型
These A B
代表(A + B + AB),它不易考虑 总和和产品 - 像Either A (B, Maybe A)
这样的类型不清楚 难以使用。
答案 1 :(得分:7)
Data.These已被提及,可能是最好的选择,但如果我要自己动手,我会这样做:
import Control.Applicative ((<$>), (<*>))
type These a b = Either (Either a b) (a, b)
maybeA :: These a b -> Maybe a
maybeA (Left (Left a)) = Just a
maybeA (Right (a, _)) = Just a
maybeA _ = Nothing
maybeB :: These a b -> Maybe b
maybeB (Left (Right b)) = Just b
maybeB (Right (_, b)) = Just b
maybeB _ = Nothing
eitherA :: These a b -> Either a b
eitherA (Left (Left a)) = Left a
eitherA (Right (a, _)) = Left a
eitherA (Left (Right b)) = Right b
eitherB :: These a b -> Either a b
eitherB (Left (Right b)) = Right b
eitherB (Right (_, b)) = Right b
eitherB (Left (Left a)) = Left a
hasBoth, hasJustA, hasJustB, hasA, hasB :: These a b -> Bool
hasBoth (Right _) = True
hasBoth _ = False
hasJustA (Left (Left _)) = True
hasJustA _ = False
hasJustB (Left (Right _)) = True
hasJustB _ = False
hasA = (||) <$> hasBoth <*> hasJustA
hasB = (||) <$> hasBoth <*> hasJustB
答案 2 :(得分:5)
如果您想要“零,一或两者”,您可以1 + A + B + A*B = (1 + A) * (1 + B)
或(Maybe A, Maybe B)
。
您可以A + B + A*B = (1+A)*(1+B)-1
将(Maybe A, Maybe B)
包裹在newtype
中并使用智能构造函数删除(Nothing,Nothing)
:
module Some (
Some(),
this, that, those, some,
oror, orro, roro, roor,
swap
) where
import Control.Applicative ((<|>))
newtype Some a b = Some (Maybe a, Maybe b) deriving (Show, Eq)
-- smart constructors
this :: a -> Some a b
this a = Some (Just a,Nothing)
that :: b -> Some a b
that b = Some (Nothing, Just b)
those :: a -> b -> Some a b
those a b = Some (Just a, Just b)
-- catamorphism/smart deconstructor
some :: (a -> r) -> (b -> r) -> (a -> b -> r) -> Some a b -> r
some f _ _ (Some (Just a, Nothing)) = f a
some _ g _ (Some (Nothing, Just b)) = g b
some _ _ h (Some (Just a, Just b)) = h a b
some _ _ _ _ = error "this case should be unreachable due to smart constructors"
swap :: Some a b -> Some b a
swap ~(Some ~(ma,mb)) = Some (mb,ma)
-- combining operators
oror, orro, roro, roor :: Some a b -> Some a b -> Some a b
-- prefer the leftmost A and the leftmost B
oror (Some (ma,mb)) (Some (ma',mb')) = Some (ma <|> ma', mb <|> mb')
-- prefer the leftmost A and the rightmost B
orro (Some (ma,mb)) (Some (ma',mb')) = Some (ma <|> ma', mb' <|> mb)
-- prefer the rightmost A and the rightmost B
roro = flip oror
-- prefer the rightmost A and the leftmost B
roor = flip orro
组合操作员很有趣:
λ this "red" `oror` that "blue" `oror` those "beige" "yellow"
Some (Just "red",Just "blue")
λ this "red" `orro` that "blue" `orro` those "beige" "yellow"
Some (Just "red",Just "yellow")
λ this "red" `roor` that "blue" `roor` those "beige" "yellow"
Some (Just "beige",Just "blue")
λ this "red" `roro` that "blue" `roro` those "beige" "yellow"
Some (Just "beige",Just "yellow")