datactl/eigenaux.h
datactl/eigenaux.h
Namespaces
Name |
---|
Syntalos |
Source code
/*
* Copyright (C) 2019-2024 Matthias Klumpp <matthias@tenstral.net>
*
* Licensed under the GNU Lesser General Public License Version 3
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the license, or
* (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this software. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include <Eigen/Dense>
#include <algorithm>
#include <QDataStream>
namespace Syntalos
{
typedef Eigen::Matrix<qint32, Eigen::Dynamic, 1> VectorXsi;
typedef Eigen::Matrix<quint64, Eigen::Dynamic, 1> VectorXul;
typedef Eigen::Matrix<qint64, Eigen::Dynamic, 1> VectorXsl;
typedef Eigen::Matrix<double, Eigen::Dynamic, 1> VectorXd;
typedef Eigen::Matrix<qint32, Eigen::Dynamic, Eigen::Dynamic> MatrixXsi;
typedef Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic> MatrixXd;
template<typename T>
double vectorMedian(const Eigen::Matrix<T, Eigen::Dynamic, 1> &vec)
{
const auto size = vec.size();
if (size == 0)
return nan(""); // what is the median of an empty vector?
std::vector<T> vecSorted(vec.size());
std::partial_sort_copy(vec.data(), vec.data() + vec.size(), std::begin(vecSorted), std::end(vecSorted));
if (size % 2 == 0)
return ((long long)vecSorted[size / 2 - 1] + (long long)vecSorted[size / 2]) / 2.0;
else
return vecSorted[size / 2];
}
template<typename T>
double vectorMedianInplace(Eigen::Matrix<T, Eigen::Dynamic, 1> &vec)
{
const auto size = vec.size();
if (size == 0)
return nan(""); // what is the median of an empty vector?
// TODO: This is dangerous code - Eigen 3.4.x brings support for STL iterators,
// which will make this much nicer when we can use it. Currently, this code
// would break for vectors with a stride != 1
std::sort(vec.data(), vec.data() + vec.size());
if (size % 2 == 0)
return ((long long)vec[size / 2 - 1] + (long long)vec[size / 2]) / 2.0;
else
return vec[size / 2];
}
template<typename T>
double vectorVariance(const Eigen::Matrix<T, Eigen::Dynamic, 1> &vec, const double &mean, bool unbiased = true)
{
if (unbiased)
return (vec.template cast<double>().array() - mean).pow(2).sum() / (vec.rows() - 1);
else
return (vec.template cast<double>().array() - mean).pow(2).sum() / vec.rows();
}
template<typename T>
double vectorVariance(const Eigen::Matrix<T, Eigen::Dynamic, 1> &vec, bool unbiased = true)
{
return vectorVariance(vec, vec.mean(), unbiased);
}
template<typename EigenType>
void serializeEigen(QDataStream &stream, const EigenType &matrix)
{
const auto rows = static_cast<quint64>(matrix.rows());
const auto cols = static_cast<quint64>(matrix.cols());
stream << rows << cols;
for (quint64 i = 0; i < rows; ++i) {
for (quint64 j = 0; j < cols; ++j) {
typename EigenType::Scalar value = matrix(i, j);
stream << value;
}
}
}
template<typename EigenType>
EigenType deserializeEigen(QDataStream &stream)
{
EigenType matrix;
quint64 rows, cols;
stream >> rows >> cols;
matrix.resize(rows, cols);
for (quint64 i = 0; i < rows; ++i) {
for (quint64 j = 0; j < cols; ++j) {
typename EigenType::Scalar value;
stream >> value;
matrix(i, j) = value;
}
}
return matrix;
}
} // namespace Syntalos
Updated on 2024-12-04 at 20:48:34 +0000