+
+
+-- | Compute the i,jth cofactor of the given @matrix@. This simply
+-- premultiplues the i,jth minor by (-1)^(i+j).
+cofactor :: (Arity m, Determined (Mat m m) a)
+ => Mat (S m) (S m) a
+ -> Int
+ -> Int
+ -> a
+cofactor matrix i j =
+ (-1)^(toInteger i + toInteger j) NP.* (minor matrix i j)
+
+
+-- | Compute the inverse of a matrix using cofactor expansion
+-- (generalized Cramer's rule).
+--
+-- Examples:
+--
+-- >>> let m1 = fromList [[37,22],[17,54]] :: Mat2 Double
+-- >>> let e1 = [54/1624, -22/1624] :: [Double]
+-- >>> let e2 = [-17/1624, 37/1624] :: [Double]
+-- >>> let expected = fromList [e1, e2] :: Mat2 Double
+-- >>> let actual = inverse m1
+-- >>> frobenius_norm (actual - expected) < 1e-12
+-- True
+--
+inverse :: (Arity m,
+ Determined (Mat (S m) (S m)) a,
+ Determined (Mat m m) a,
+ Field.C a)
+ => Mat (S m) (S m) a
+ -> Mat (S m) (S m) a
+inverse matrix =
+ (1 / (determinant matrix)) *> (transpose $ construct lambda)
+ where
+ lambda i j = cofactor matrix i j
+