How do you use the Bounded typeclass in Haskell to define a type with a floating point range?

后端 未结 1 794
旧巷少年郎
旧巷少年郎 2021-01-01 18:34

I expected the following code to fail with a type error due to violation of the minBound and maxBound. But, as you can see, it goes through without flagging an error.

相关标签:
1条回答
  • 2021-01-01 18:49

    That's not what Bounded is for. Bounded a just defines the functions minBound :: a and maxBound :: a. It does not induce any special checking or anything.

    You can define a bounded type using a so-called smart constructor. That is:

    module Probability (Probability) where
    
    newtype Probability = P { getP :: Float }
        deriving (Eq,Ord,Show)
    
    mkP :: Float -> Probability
    mkP x | 0 <= x && x <= 1 = P x
          | otherwise = error $ show x ++ " is not in [0,1]"
    
    -- after this point, the Probability data constructor is not to be used
    
    instance Num Probability where
        P x + P y = mkP (x + y)
        P x * P y = mkP (x * y)
        fromIntegral = mkP . fromIntegral
        ...
    

    So the only way to make a Probability is to use the mkP function eventually (this is done for you when you use numeric operations given our Num instance), which checks that the argument is in range. Because of the module's export list, outside of this module is it not possible to construct an invalid probability.

    Probably not the two-liner you were looking for, but oh well.

    For extra composability, you could factor out this functionality by making a BoundCheck module instead of Probability. Just like above, except:

    newtype BoundCheck a = BC { getBC :: a }
        deriving (Bounded,Eq,Ord,Show)
    
    mkBC :: (Bounded a) => a -> BoundCheck a
    mkBC x | minBound <= x && x <= maxBound = BC x
           | otherwise = error "..."
    
    instance (Bounded a) => Num (BoundCheck a) where
        BC x + BC y = mkBC (x + y)
        ...
    

    Thus you can get the functionality you were wishing was built in for you when you asked the question. 

    To do this deriving stuff you may need the language extension {-# LANGUAGE GeneralizedNewtypeDeriving #-}.

    0 讨论(0)
提交回复
热议问题