fromList,
head,
ifoldl,
- length,
+ imap,
map,
maximum,
replicate,
toList :: Mat m n a -> [[a]]
toList (Mat rows) = map V.toList (V.toList rows)
+
-- | Create a matrix from a nested list.
fromList :: (Arity m, Arity n) => [[a]] -> Mat m n a
fromList vs = Mat (V.fromList $ map V.fromList vs)
--- | Unsafe indexing.
+-- | Unsafe indexing. Much faster than the safe indexing.
(!!!) :: (Arity m, Arity n) => Mat m n a -> (Int, Int) -> a
-(!!!) m (i, j) = (row m i) ! j
+(!!!) (Mat rows) (i, j) = (rows ! i) ! j
+
-- | Safe indexing.
-(!!?) :: Mat m n a -> (Int, Int) -> Maybe a
-(!!?) m@(Mat rows) (i, j)
- | i < 0 || j < 0 = Nothing
- | i > V.length rows = Nothing
- | otherwise = if j > V.length (row m j)
- then Nothing
- else Just $ (row m j) ! j
+--
+-- Examples:
+--
+-- >>> let m = fromList [[1,2],[3,4]] :: Mat2 Int
+-- >>> m !!? (-1,-1)
+-- Nothing
+-- >>> m !!? (-1,0)
+-- Nothing
+-- >>> m !!? (-1,1)
+-- Nothing
+-- >>> m !!? (0,-1)
+-- Nothing
+-- >>> m !!? (0,0)
+-- Just 1
+-- >>> m !!? (0,1)
+-- Just 2
+-- >>> m !!? (1,-1)
+-- Nothing
+-- >>> m !!? (1,0)
+-- Just 3
+-- >>> m !!? (1,1)
+-- Just 4
+-- >>> m !!? (2,-1)
+-- Nothing
+-- >>> m !!? (2,0)
+-- Nothing
+-- >>> m !!? (2,1)
+-- Nothing
+-- >>> m !!? (2,2)
+-- Nothing
+--
+(!!?) :: (Arity m, Arity n) => Mat m n a -> (Int, Int) -> Maybe a
+(!!?) matrix idx =
+ ifoldl2 f Nothing matrix
+ where
+ f k l found cur = if (k,l) == idx then (Just cur) else found
-- | The number of rows in the matrix.
nrows :: forall m n a. (Arity m) => Mat m n a -> Int
nrows _ = arity (undefined :: m)
+
-- | The number of columns in the first row of the
-- matrix. Implementation stolen from Data.Vector.Fixed.length.
ncols :: forall m n a. (Arity n) => Mat m n a -> Int
ncols _ = arity (undefined :: n)
--- | Return the @i@th row of @m@. Unsafe.
-row :: Mat m n a -> Int -> (Vec n a)
-row (Mat rows) i = rows ! i
-
-
-- | Return the @i@th row of @m@ as a matrix. Unsafe.
-row' :: (Arity m, Arity n) => Mat m n a -> Int -> Row n a
-row' m i =
+row :: (Arity m, Arity n) => Mat m n a -> Int -> Row n a
+row m i =
construct lambda
where
lambda _ j = m !!! (i, j)
--- | Return the @j@th column of @m@. Unsafe.
---column :: Mat m n a -> Int -> (Vec m a)
---column (Mat rows) j =
--- V.map (element j) rows
--- where
--- element = flip (!)
-
-
-- | Return the @j@th column of @m@ as a matrix. Unsafe.
column :: (Arity m, Arity n) => Mat m n a -> Int -> Col m a
column m j =
identity_matrix =
construct (\i j -> if i == j then (fromInteger 1) else (fromInteger 0))
+
-- | Given a positive-definite matrix @m@, computes the
-- upper-triangular matrix @r@ with (transpose r)*r == m and all
-- values on the diagonal of @r@ positive.
row_function rowinit idx r = V.ifoldl (g idx) rowinit r
+-- | Map a function over a matrix of any dimensions, passing the
+-- coordinates @i@ and @j@ to the function @f@.
+--
+-- Examples:
+--
+-- >>> let m = fromList [[1,2],[3,4]] :: Mat2 Int
+-- >>> imap2 (\i j _ -> i+j) m
+-- ((0,1),(1,2))
+--
+imap2 :: (Int -> Int -> a -> b) -> Mat m n a -> Mat m n b
+imap2 f (Mat rows) =
+ Mat $ V.imap g rows
+ where
+ g i = V.imap (f i)