Program Listing for File AlgebraicSpanningEdgeCentrality.hpp

Return to documentation for file (include/networkit/algebraic/algorithms/AlgebraicSpanningEdgeCentrality.hpp)

/*
 * AlgebraicSpanningEdgeCentrality.hpp
 *
 *  Created on: Jul 12, 2016
 *      Author: Michael Wegner (michael.wegner@student.kit.edu)
 */

#ifndef NETWORKIT_ALGEBRAIC_ALGORITHMS_ALGEBRAIC_SPANNING_EDGE_CENTRALITY_HPP_
#define NETWORKIT_ALGEBRAIC_ALGORITHMS_ALGEBRAIC_SPANNING_EDGE_CENTRALITY_HPP_

#include <atomic>
#include <cmath>

#include <networkit/centrality/Centrality.hpp>
#include <networkit/numerics/LAMG/Lamg.hpp>

namespace NetworKit {

template <class Matrix>
class AlgebraicSpanningEdgeCentrality : public Centrality {
public:
    AlgebraicSpanningEdgeCentrality(const Graph &graph, double tol = 0.1)
        : Centrality(graph), tol(tol) {}

    void run() override;

    void runApproximation();

private:
    double tol;
};

template <class Matrix>
void AlgebraicSpanningEdgeCentrality<Matrix>::run() {
    const count n = G.numberOfNodes();
    const count m = G.numberOfEdges();
    scoreData.clear();
    scoreData.resize(m, 0.0);

    std::vector<Vector> rhs(m, Vector(n));
    this->G.parallelForEdges([&](node u, node v, edgeid e) {
        rhs[e][u] = +1;
        rhs[e][v] = -1;
    });

    std::vector<Vector> solutions(m, Vector(n));

    Lamg<Matrix> lamg(1e-5);
    lamg.setupConnected(Matrix::laplacianMatrix(this->G));
    lamg.parallelSolve(rhs, solutions);

    this->G.parallelForEdges([&](node u, node v, edgeid e) {
        double diff = solutions[e][u] - solutions[e][v];
        scoreData[e] = std::fabs(diff);
    });

    hasRun = true;
}

template <class Matrix>
void AlgebraicSpanningEdgeCentrality<Matrix>::runApproximation() {
    const count n = G.numberOfNodes();
    const count m = G.numberOfEdges();
    scoreData.clear();
    scoreData.resize(m, 0.0);
    double epsilon2 = tol * tol;
    const count k = std::ceil(std::log2(n)) / epsilon2;
    double randTab[2] = {1.0 / std::sqrt(k), -1.0 / std::sqrt(k)};

    const auto rhsLoader = [&](count, Vector &yRow) -> Vector & {
        yRow.fill(0);
        G.forEdges([&](node u, node v, edgeweight w, index) {
            double rand = randTab[Aux::Random::integer(1)];
            yRow[u] += w * rand;
            yRow[v] -= w * rand;
        });
        return yRow;
    };

    std::vector<std::atomic<double>> scoreDataAtom(m);
    const auto resultProcessor = [&](count, const Vector &zRow) {
        G.forEdges([&](node u, node v, edgeid e) {
            double diff = (zRow[u] - zRow[v]);
            diff *= diff;
            for (double old = scoreDataAtom[e].load(); !scoreDataAtom[e].compare_exchange_strong(
                     old, old + diff, std::memory_order_relaxed);)
                ;
        });
    };

    Lamg<Matrix> lamg(1e-5);
    lamg.setupConnected(Matrix::laplacianMatrix(this->G));
    lamg.parallelSolve(rhsLoader, resultProcessor, std::make_pair(k, n));

#pragma omp parallel for
    for (omp_index i = 0; i < static_cast<omp_index>(m); ++i) {
        scoreData[i] = scoreDataAtom[i].load(std::memory_order_relaxed);
    }

    hasRun = true;
}

} /* namespace NetworKit */

#endif // NETWORKIT_ALGEBRAIC_ALGORITHMS_ALGEBRAIC_SPANNING_EDGE_CENTRALITY_HPP_