したがって、一緒によく使用する型クラスのペアがあり、毎回両方を指定することは避けたいと考えています。基本的に、置く代わりに
:: (Ord a, Fractional a, Ord b, Fractional b, ... Ord z, Fractional z) =>
すべての型指定の最初に、
:: (OrdFractional a, OrdFractional b, ... OrdFractional z)
したがって、これを行う方法に関する私の最初のアイデアは、新しい型クラスを宣言することでした
module Example where
class (Fractional a, Ord a) => OrdFractional a
example :: (OrdFractional a, OrdFractional b) => (a,b) -> (a,b) -> (a,b) -> Bool
example (x1,y1) (x2,y2) (x3,y3) = (x1/x2 < x2/x3) && (y1/y2 < y2/y3)
しかし、これは私が望んでいたほど自動的には機能しませんでした:
% ghci
Prelude> :l Example.hs
Ok, modules loaded: Example.
Prelude Example> example (1::Float,3::Float) (2,2) (3,1)
<interactive>:1:0:
No instance for (OrdFractional Float)
arising from a use of `example' at <interactive>:1:0-39
Possible fix:
add an instance declaration for (OrdFractional Float)
In the expression: example (1 :: Float, 3 :: Float) (2, 2) (3, 1)
In the definition of `it':
it = example (1 :: Float, 3 :: Float) (2, 2) (3, 1)
インスタンスを手動で作成するのは面倒に思えるので、次に、インスタンスを自動的に作成しようと考えました。
module Example where
class OrdFractional a
instance (Fractional a, Ord a) => OrdFractional a
example :: (OrdFractional a, OrdFractional b) => (a,b) -> (a,b) -> (a,b) -> Bool
example (x1,y1) (x2,y2) (x3,y3) = (x1/x2 < x2/x3) && (y1/y2 < y2/y3)
しかし、コンパイラはそれを好まなかった:
ghc -c Example.hs
Example.hs:4:0:
Illegal instance declaration for `OrdFractional a'
(All instance types must be of the form (T a1 ... an)
where a1 ... an are type *variables*,
and each type variable appears at most once in the instance head.
Use -XFlexibleInstances if you want to disable this.)
In the instance declaration for `OrdFractional a'
これを行う方法はありますか?