诀窍是使用类型类。在 的情况下printf
,关键是PrintfType
类型类。它不公开任何方法,但重要的部分还是在类型中。
class PrintfType r
printf :: PrintfType r => String -> r
所以printf
有一个重载的返回类型。在普通情况下,我们没有额外的参数,所以我们需要能够实例化r
到IO ()
. 为此,我们有实例
instance PrintfType (IO ())
接下来,为了支持可变数量的参数,我们需要在实例级别使用递归。特别是我们需要一个实例,以便如果r
是 a PrintfType
,函数类型x -> r
也是 a PrintfType
。
-- instance PrintfType r => PrintfType (x -> r)
当然,我们只想支持实际可以格式化的参数。这就是第二种类型类的PrintfArg
用武之地。所以实际的实例是
instance (PrintfArg x, PrintfType r) => PrintfType (x -> r)
这是一个简化版本,它在类中接受任意数量的参数Show
并打印它们:
{-# LANGUAGE FlexibleInstances #-}
foo :: FooType a => a
foo = bar (return ())
class FooType a where
bar :: IO () -> a
instance FooType (IO ()) where
bar = id
instance (Show x, FooType r) => FooType (x -> r) where
bar s x = bar (s >> print x)
在这里,bar
采用递归构建的 IO 操作,直到没有更多参数为止,此时我们只需执行它。
*Main> foo 3 :: IO ()
3
*Main> foo 3 "hello" :: IO ()
3
"hello"
*Main> foo 3 "hello" True :: IO ()
3
"hello"
True
QuickCheck 也使用相同的技术,其中Testable
类具有基本情况的实例Bool
,以及在类中接受参数的函数的递归实例Arbitrary
。
class Testable a
instance Testable Bool
instance (Arbitrary x, Testable r) => Testable (x -> r)