forked from M-Labs/nalgebra
Add mean and covariance computation for DMat.
This also fixes the transposition for rectangular DMat, and add scalar addition/subtraction/multiplication/division for DMat.
This commit is contained in:
parent
d12d23d2c7
commit
4ffe274b03
242
src/dmat.rs
242
src/dmat.rs
@ -1,5 +1,7 @@
|
|||||||
//! Matrix with dimensions unknown at compile-time.
|
//! Matrix with dimensions unknown at compile-time.
|
||||||
|
|
||||||
|
#[doc(hidden)]; // we hide doc to not have to document the $trhs double dispatch trait.
|
||||||
|
|
||||||
use std::rand::Rand;
|
use std::rand::Rand;
|
||||||
use std::rand;
|
use std::rand;
|
||||||
use std::num::{One, Zero};
|
use std::num::{One, Zero};
|
||||||
@ -7,28 +9,28 @@ use std::vec;
|
|||||||
use std::cmp::ApproxEq;
|
use std::cmp::ApproxEq;
|
||||||
use std::util;
|
use std::util;
|
||||||
use dvec::{DVec, DVecMulRhs};
|
use dvec::{DVec, DVecMulRhs};
|
||||||
use traits::operations::{Inv, Transpose};
|
use traits::operations::{Inv, Transpose, Mean, Cov};
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
|
mod metal;
|
||||||
|
|
||||||
/// Matrix with dimensions unknown at compile-time.
|
/// Matrix with dimensions unknown at compile-time.
|
||||||
#[deriving(Eq, ToStr, Clone)]
|
#[deriving(Eq, Clone)]
|
||||||
pub struct DMat<N> {
|
pub struct DMat<N> {
|
||||||
priv nrows: uint,
|
priv nrows: uint,
|
||||||
priv ncols: uint,
|
priv ncols: uint,
|
||||||
priv mij: ~[N]
|
priv mij: ~[N]
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Trait of object `o` which can be multiplied by a `DMat` `d`: `d * o`.
|
double_dispatch_binop_decl_trait!(DMat, DMatMulRhs)
|
||||||
pub trait DMatMulRhs<N, Res> {
|
double_dispatch_binop_decl_trait!(DMat, DMatDivRhs)
|
||||||
/// Multiplies a `DMat` by `Self`.
|
double_dispatch_binop_decl_trait!(DMat, DMatAddRhs)
|
||||||
fn binop(left: &DMat<N>, right: &Self) -> Res;
|
double_dispatch_binop_decl_trait!(DMat, DMatSubRhs)
|
||||||
}
|
|
||||||
|
|
||||||
impl<N, Rhs: DMatMulRhs<N, Res>, Res> Mul<Rhs, Res> for DMat<N> {
|
mul_redispatch_impl!(DMat, DMatMulRhs)
|
||||||
#[inline(always)]
|
div_redispatch_impl!(DMat, DMatDivRhs)
|
||||||
fn mul(&self, other: &Rhs) -> Res {
|
add_redispatch_impl!(DMat, DMatAddRhs)
|
||||||
DMatMulRhs::binop(self, other)
|
sub_redispatch_impl!(DMat, DMatSubRhs)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<N> DMat<N> {
|
impl<N> DMat<N> {
|
||||||
/// Creates an uninitialized matrix.
|
/// Creates an uninitialized matrix.
|
||||||
@ -89,6 +91,20 @@ impl<N: Clone> DMat<N> {
|
|||||||
mij: vec::from_elem(nrows * ncols, val)
|
mij: vec::from_elem(nrows * ncols, val)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Builds a matrix filled with the components provided by a vector.
|
||||||
|
///
|
||||||
|
/// The vector must have at least `nrows * ncols` elements.
|
||||||
|
#[inline]
|
||||||
|
pub fn from_vec(nrows: uint, ncols: uint, vec: &[N]) -> DMat<N> {
|
||||||
|
assert!(nrows * ncols <= vec.len());
|
||||||
|
|
||||||
|
DMat {
|
||||||
|
nrows: nrows,
|
||||||
|
ncols: ncols,
|
||||||
|
mij: vec.slice_to(nrows * ncols).to_owned()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<N> DMat<N> {
|
impl<N> DMat<N> {
|
||||||
@ -116,7 +132,7 @@ impl<N> DMat<N> {
|
|||||||
|
|
||||||
/// Transforms this matrix into an array. This consumes the matrix and is O(1).
|
/// Transforms this matrix into an array. This consumes the matrix and is O(1).
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn to_array(self) -> ~[N] {
|
pub fn to_vec(self) -> ~[N] {
|
||||||
self.mij
|
self.mij
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -350,14 +366,31 @@ Inv for DMat<N> {
|
|||||||
impl<N: Clone> Transpose for DMat<N> {
|
impl<N: Clone> Transpose for DMat<N> {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn transposed(&self) -> DMat<N> {
|
fn transposed(&self) -> DMat<N> {
|
||||||
|
if self.nrows == self.ncols {
|
||||||
let mut res = self.clone();
|
let mut res = self.clone();
|
||||||
|
|
||||||
res.transpose();
|
res.transpose();
|
||||||
|
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
let mut res = unsafe { DMat::new_uninitialized(self.ncols, self.nrows) };
|
||||||
|
|
||||||
|
for i in range(0u, self.nrows) {
|
||||||
|
for j in range(0u, self.ncols) {
|
||||||
|
unsafe {
|
||||||
|
res.set_fast(j, i, self.at_fast(i, j))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
res
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
fn transpose(&mut self) {
|
fn transpose(&mut self) {
|
||||||
|
if self.nrows == self.ncols {
|
||||||
for i in range(1u, self.nrows) {
|
for i in range(1u, self.nrows) {
|
||||||
for j in range(0u, self.ncols - 1) {
|
for j in range(0u, self.ncols - 1) {
|
||||||
let off_i_j = self.offset(i, j);
|
let off_i_j = self.offset(i, j);
|
||||||
@ -369,6 +402,53 @@ impl<N: Clone> Transpose for DMat<N> {
|
|||||||
|
|
||||||
util::swap(&mut self.nrows, &mut self.ncols);
|
util::swap(&mut self.nrows, &mut self.ncols);
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
// FIXME: implement a better algorithm which does that in-place.
|
||||||
|
*self = self.transposed();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<N: Num + NumCast + Clone> Mean<DVec<N>> for DMat<N> {
|
||||||
|
fn mean(&self) -> DVec<N> {
|
||||||
|
let mut res: DVec<N> = DVec::new_zeros(self.ncols);
|
||||||
|
let normalizer: N = NumCast::from(1.0f64 / NumCast::from(self.nrows));
|
||||||
|
|
||||||
|
for i in range(0u, self.nrows) {
|
||||||
|
for j in range(0u, self.ncols) {
|
||||||
|
unsafe {
|
||||||
|
let acc = res.at_fast(j) + self.at_fast(i, j) * normalizer;
|
||||||
|
res.set_fast(j, acc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
res
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<N: Clone + Num + NumCast + DMatDivRhs<N, DMat<N>> + ToStr > Cov<DMat<N>> for DMat<N> {
|
||||||
|
// FIXME: this could be heavily optimized, removing all temporaries by merging loops.
|
||||||
|
fn cov(&self) -> DMat<N> {
|
||||||
|
assert!(self.nrows > 1);
|
||||||
|
|
||||||
|
let mut centered = unsafe { DMat::new_uninitialized(self.nrows, self.ncols) };
|
||||||
|
let mean = self.mean();
|
||||||
|
|
||||||
|
// FIXME: use the rows iterator when available
|
||||||
|
for i in range(0u, self.nrows) {
|
||||||
|
for j in range(0u, self.ncols) {
|
||||||
|
unsafe {
|
||||||
|
centered.set_fast(i, j, self.at_fast(i, j) - mean.at_fast(j));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: return a triangular matrix?
|
||||||
|
let normalizer: N = NumCast::from(self.nrows() - 1);
|
||||||
|
// FIXME: this will do 2 allocations for temporaries!
|
||||||
|
(centered.transposed() * centered) / normalizer
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<N: ApproxEq<N>> ApproxEq<N> for DMat<N> {
|
impl<N: ApproxEq<N>> ApproxEq<N> for DMat<N> {
|
||||||
@ -398,3 +478,137 @@ impl<N: ApproxEq<N>> ApproxEq<N> for DMat<N> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
macro_rules! scalar_mul_impl (
|
||||||
|
($n: ident) => (
|
||||||
|
impl DMatMulRhs<$n, DMat<$n>> for $n {
|
||||||
|
#[inline]
|
||||||
|
fn binop(left: &DMat<$n>, right: &$n) -> DMat<$n> {
|
||||||
|
DMat {
|
||||||
|
nrows: left.nrows,
|
||||||
|
ncols: left.ncols,
|
||||||
|
mij: left.mij.iter().map(|a| a * *right).collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
macro_rules! scalar_div_impl (
|
||||||
|
($n: ident) => (
|
||||||
|
impl DMatDivRhs<$n, DMat<$n>> for $n {
|
||||||
|
#[inline]
|
||||||
|
fn binop(left: &DMat<$n>, right: &$n) -> DMat<$n> {
|
||||||
|
DMat {
|
||||||
|
nrows: left.nrows,
|
||||||
|
ncols: left.ncols,
|
||||||
|
mij: left.mij.iter().map(|a| a / *right).collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
macro_rules! scalar_add_impl (
|
||||||
|
($n: ident) => (
|
||||||
|
impl DMatAddRhs<$n, DMat<$n>> for $n {
|
||||||
|
#[inline]
|
||||||
|
fn binop(left: &DMat<$n>, right: &$n) -> DMat<$n> {
|
||||||
|
DMat {
|
||||||
|
nrows: left.nrows,
|
||||||
|
ncols: left.ncols,
|
||||||
|
mij: left.mij.iter().map(|a| a + *right).collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
macro_rules! scalar_sub_impl (
|
||||||
|
($n: ident) => (
|
||||||
|
impl DMatSubRhs<$n, DMat<$n>> for $n {
|
||||||
|
#[inline]
|
||||||
|
fn binop(left: &DMat<$n>, right: &$n) -> DMat<$n> {
|
||||||
|
DMat {
|
||||||
|
nrows: left.nrows,
|
||||||
|
ncols: left.ncols,
|
||||||
|
mij: left.mij.iter().map(|a| a - *right).collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
scalar_mul_impl!(f64)
|
||||||
|
scalar_mul_impl!(f32)
|
||||||
|
scalar_mul_impl!(u64)
|
||||||
|
scalar_mul_impl!(u32)
|
||||||
|
scalar_mul_impl!(u16)
|
||||||
|
scalar_mul_impl!(u8)
|
||||||
|
scalar_mul_impl!(i64)
|
||||||
|
scalar_mul_impl!(i32)
|
||||||
|
scalar_mul_impl!(i16)
|
||||||
|
scalar_mul_impl!(i8)
|
||||||
|
scalar_mul_impl!(float)
|
||||||
|
scalar_mul_impl!(uint)
|
||||||
|
scalar_mul_impl!(int)
|
||||||
|
|
||||||
|
scalar_div_impl!(f64)
|
||||||
|
scalar_div_impl!(f32)
|
||||||
|
scalar_div_impl!(u64)
|
||||||
|
scalar_div_impl!(u32)
|
||||||
|
scalar_div_impl!(u16)
|
||||||
|
scalar_div_impl!(u8)
|
||||||
|
scalar_div_impl!(i64)
|
||||||
|
scalar_div_impl!(i32)
|
||||||
|
scalar_div_impl!(i16)
|
||||||
|
scalar_div_impl!(i8)
|
||||||
|
scalar_div_impl!(float)
|
||||||
|
scalar_div_impl!(uint)
|
||||||
|
scalar_div_impl!(int)
|
||||||
|
|
||||||
|
scalar_add_impl!(f64)
|
||||||
|
scalar_add_impl!(f32)
|
||||||
|
scalar_add_impl!(u64)
|
||||||
|
scalar_add_impl!(u32)
|
||||||
|
scalar_add_impl!(u16)
|
||||||
|
scalar_add_impl!(u8)
|
||||||
|
scalar_add_impl!(i64)
|
||||||
|
scalar_add_impl!(i32)
|
||||||
|
scalar_add_impl!(i16)
|
||||||
|
scalar_add_impl!(i8)
|
||||||
|
scalar_add_impl!(float)
|
||||||
|
scalar_add_impl!(uint)
|
||||||
|
scalar_add_impl!(int)
|
||||||
|
|
||||||
|
scalar_sub_impl!(f64)
|
||||||
|
scalar_sub_impl!(f32)
|
||||||
|
scalar_sub_impl!(u64)
|
||||||
|
scalar_sub_impl!(u32)
|
||||||
|
scalar_sub_impl!(u16)
|
||||||
|
scalar_sub_impl!(u8)
|
||||||
|
scalar_sub_impl!(i64)
|
||||||
|
scalar_sub_impl!(i32)
|
||||||
|
scalar_sub_impl!(i16)
|
||||||
|
scalar_sub_impl!(i8)
|
||||||
|
scalar_sub_impl!(float)
|
||||||
|
scalar_sub_impl!(uint)
|
||||||
|
scalar_sub_impl!(int)
|
||||||
|
|
||||||
|
impl<N: ToStr + Clone> ToStr for DMat<N> {
|
||||||
|
fn to_str(&self) -> ~str {
|
||||||
|
let mut res = ~"DMat ";
|
||||||
|
res = res + self.nrows.to_str() + " " + self.ncols.to_str() + " {\n";
|
||||||
|
|
||||||
|
for i in range(0u, self.nrows) {
|
||||||
|
for j in range(0u, self.ncols) {
|
||||||
|
res = res + " " + unsafe { self.at_fast(i, j).to_str() };
|
||||||
|
}
|
||||||
|
|
||||||
|
res = res + "\n";
|
||||||
|
}
|
||||||
|
res = res + "}";
|
||||||
|
|
||||||
|
res
|
||||||
|
}
|
||||||
|
}
|
||||||
|
23
src/dvec.rs
23
src/dvec.rs
@ -12,6 +12,7 @@ use std::iter::FromIterator;
|
|||||||
use traits::geometry::{Dot, Norm, Translation};
|
use traits::geometry::{Dot, Norm, Translation};
|
||||||
use traits::structure::{Iterable, IterableMut};
|
use traits::structure::{Iterable, IterableMut};
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
mod metal;
|
mod metal;
|
||||||
|
|
||||||
/// Vector with a dimension unknown at compile-time.
|
/// Vector with a dimension unknown at compile-time.
|
||||||
@ -85,6 +86,16 @@ impl<N> DVec<N> {
|
|||||||
at: vec
|
at: vec
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub unsafe fn set_fast(&mut self, i: uint, val: N) {
|
||||||
|
*self.at.unsafe_mut_ref(i) = val
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn to_vec(self) -> ~[N] {
|
||||||
|
self.at
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<N: Clone> DVec<N> {
|
impl<N: Clone> DVec<N> {
|
||||||
@ -93,6 +104,18 @@ impl<N: Clone> DVec<N> {
|
|||||||
pub fn from_elem(dim: uint, elem: N) -> DVec<N> {
|
pub fn from_elem(dim: uint, elem: N) -> DVec<N> {
|
||||||
DVec { at: vec::from_elem(dim, elem) }
|
DVec { at: vec::from_elem(dim, elem) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Builds a vector filled with the components provided by a vector.
|
||||||
|
///
|
||||||
|
/// The vector must have at least `dim` elements.
|
||||||
|
#[inline]
|
||||||
|
pub fn from_vec(dim: uint, vec: &[N]) -> DVec<N> {
|
||||||
|
assert!(dim <= vec.len());
|
||||||
|
|
||||||
|
DVec {
|
||||||
|
at: vec.slice_to(dim).to_owned()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<N> DVec<N> {
|
impl<N> DVec<N> {
|
||||||
|
@ -10,7 +10,7 @@ use vec::*;
|
|||||||
|
|
||||||
// traits
|
// traits
|
||||||
pub use traits::structure::{Mat, Dim, Indexable, Iterable, IterableMut, MatCast, Row, Col};
|
pub use traits::structure::{Mat, Dim, Indexable, Iterable, IterableMut, MatCast, Row, Col};
|
||||||
pub use traits::operations::{Absolute, ScalarSub, ScalarAdd, Inv, RMul, Transpose};
|
pub use traits::operations::{Absolute, ScalarSub, ScalarAdd, Inv, RMul, Transpose, Mean, Cov};
|
||||||
pub use traits::geometry::{Rotation, RotationMatrix, Rotate, Transformation, Transform,
|
pub use traits::geometry::{Rotation, RotationMatrix, Rotate, Transformation, Transform,
|
||||||
Translation, Translate, ToHomogeneous, FromHomogeneous,
|
Translation, Translate, ToHomogeneous, FromHomogeneous,
|
||||||
RotationWithTranslation, AbsoluteRotate};
|
RotationWithTranslation, AbsoluteRotate};
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
#[macro_escape];
|
#[macro_escape];
|
||||||
|
|
||||||
|
#[doc(hidden)]; // we hide doc to not have to document the $trhs double dispatch trait.
|
||||||
|
|
||||||
// Create the traits needed to do fancy operator oveloading.
|
// Create the traits needed to do fancy operator oveloading.
|
||||||
// This is a meta version of
|
// This is a meta version of
|
||||||
// http://smallcultfollowing.com/babysteps/blog/2012/10/04/refining-traits-slash-impls/
|
// http://smallcultfollowing.com/babysteps/blog/2012/10/04/refining-traits-slash-impls/
|
||||||
|
@ -113,3 +113,45 @@ fn test_inv_rotation3() {
|
|||||||
assert!((rot.transposed() * rot).approx_eq(&One::one()));
|
assert!((rot.transposed() * rot).approx_eq(&One::one()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_mean_dmat() {
|
||||||
|
let mat = DMat::from_vec(
|
||||||
|
3,
|
||||||
|
3,
|
||||||
|
[
|
||||||
|
1.0f64, 2.0, 3.0,
|
||||||
|
4.0f64, 5.0, 6.0,
|
||||||
|
7.0f64, 8.0, 9.0,
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(mat.mean().approx_eq(&DVec::from_vec(3, [4.0f64, 5.0, 6.0])));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_cov_dmat() {
|
||||||
|
let mat = DMat::from_vec(
|
||||||
|
5,
|
||||||
|
3,
|
||||||
|
[
|
||||||
|
4.0, 2.0, 0.60,
|
||||||
|
4.2, 2.1, 0.59,
|
||||||
|
3.9, 2.0, 0.58,
|
||||||
|
4.3, 2.1, 0.62,
|
||||||
|
4.1, 2.2, 0.63
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = DMat::from_vec(
|
||||||
|
3,
|
||||||
|
3,
|
||||||
|
[
|
||||||
|
0.025, 0.0075, 0.00175,
|
||||||
|
0.0075, 0.007, 0.00135,
|
||||||
|
0.00175, 0.00135, 0.00043
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(mat.cov().approx_eq(&expected));
|
||||||
|
}
|
||||||
|
@ -31,6 +31,22 @@ pub trait Outer<V, M> {
|
|||||||
fn outer(&self, other: &V) -> M;
|
fn outer(&self, other: &V) -> M;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Trait for computing the covariance of a set of data.
|
||||||
|
pub trait Cov<M> {
|
||||||
|
/// Computes the covariance of the obsevations stored by `self`:
|
||||||
|
/// * for matrices, observations are stored in its rows.
|
||||||
|
/// * for vectors, observations are stored in its components (thus are 1-dimensional).
|
||||||
|
fn cov(&self) -> M;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Trait for computing the covariance of a set of data.
|
||||||
|
pub trait Mean<N> {
|
||||||
|
/// Computes the mean of the observations stored by `self`.
|
||||||
|
/// * for matrices, observations are stored in its rows.
|
||||||
|
/// * for vectors, observations are stored in its components (thus are 1-dimensional).
|
||||||
|
fn mean(&self) -> N;
|
||||||
|
}
|
||||||
|
|
||||||
// XXX: those two traits should not exist since there is generalized operator overloading of Add
|
// XXX: those two traits should not exist since there is generalized operator overloading of Add
|
||||||
// and Sub.
|
// and Sub.
|
||||||
// However, using the same trait multiple time as a trait bound (ex: impl<T: Add<N, V> + Add<V, V>)
|
// However, using the same trait multiple time as a trait bound (ex: impl<T: Add<N, V> + Add<V, V>)
|
||||||
|
@ -105,16 +105,24 @@ pub trait Row<R> {
|
|||||||
fn row(&self, i: uint) -> R;
|
fn row(&self, i: uint) -> R;
|
||||||
/// Writes the `i`-th row of `self`.
|
/// Writes the `i`-th row of `self`.
|
||||||
fn set_row(&mut self, i: uint, R);
|
fn set_row(&mut self, i: uint, R);
|
||||||
|
|
||||||
|
// FIXME: add iterators on rows: this could be a very good way to generalize _and_ optimize
|
||||||
|
// a lot of operations.
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Trait to access columns of a matrix or vector.
|
/// Trait to access columns of a matrix or vector.
|
||||||
pub trait Col<C> {
|
pub trait Col<C> {
|
||||||
/// The number of column of this matrix or vector.
|
/// The number of column of this matrix or vector.
|
||||||
fn num_cols(&self) -> uint;
|
fn num_cols(&self) -> uint;
|
||||||
|
|
||||||
/// Reads the `i`-th column of `self`.
|
/// Reads the `i`-th column of `self`.
|
||||||
fn col(&self, i: uint) -> C;
|
fn col(&self, i: uint) -> C;
|
||||||
|
|
||||||
/// Writes the `i`-th column of `self`.
|
/// Writes the `i`-th column of `self`.
|
||||||
fn set_col(&mut self, i: uint, C);
|
fn set_col(&mut self, i: uint, C);
|
||||||
|
|
||||||
|
// FIXME: add iterators on columns: this could be a very good way to generalize _and_ optimize
|
||||||
|
// a lot of operations.
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Trait of objects having a spacial dimension known at compile time.
|
/// Trait of objects having a spacial dimension known at compile time.
|
||||||
|
Loading…
Reference in New Issue
Block a user