Source code for graph_tool.spectral

#! /usr/bin/env python
# -*- coding: utf-8 -*-
#
# graph_tool -- a general graph manipulation python module
#
# Copyright (C) 2006-2025 Tiago de Paula Peixoto <tiago@skewed.de>
#
# 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 program 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 program. If not, see <http://www.gnu.org/licenses/>.

"""
``graph_tool.spectral``
-----------------------

This module contains several linear operators based on network structure, which
useful for spectral analysis.

Sparse matrices
+++++++++++++++

.. autosummary::
   :nosignatures:
   :toctree: autosummary

   adjacency
   laplacian
   incidence
   transition
   modularity_matrix
   hashimoto

Operator objects
++++++++++++++++

.. autosummary::
   :nosignatures:
   :toctree: autosummary
   :template: class.rst

   AdjacencyOperator
   LaplacianOperator
   IncidenceOperator
   TransitionOperator
   HashimotoOperator
   CompactHashimotoOperator

"""

from .. import _degree, _prop, Graph, GraphView, _limit_args, Vector_int64_t, \
    Vector_double, _parallel
from .. generation import label_self_loops
import numpy
import scipy.sparse
import scipy.sparse.linalg

from .. dl_import import dl_import
dl_import("from . import libgraph_tool_spectral")

__all__ = ["adjacency", "AdjacencyOperator", "laplacian", "LaplacianOperator",
           "incidence", "IncidenceOperator", "transition", "TransitionOperator",
           "modularity_matrix", "hashimoto", "HashimotoOperator",
           "CompactHashimotoOperator"]

def _operator(f):
    text = """.. admonition:: :class:`~scipy.sparse.linalg.LinearOperator` vs. sparse matrices

        For many linear algebra computations it is more efficient to pass
        ``operator=True`` to this function. In this case, it will return a
        :class:`scipy.sparse.linalg.LinearOperator` subclass, which implements
        matrix-vector and matrix-matrix multiplication, and is sufficient for
        the sparse linear algebra operations available in the scipy module
        :mod:`scipy.sparse.linalg`. This avoids copying the whole graph as a
        sparse matrix, and performs the multiplication operations in parallel
        (if enabled during compilation) --- see note below.

    @parallel@

        (The above is only applicable if ``operator == True``, and when the
        object returned is used to perform matrix-vector or matrix-matrix
        multiplications.)
    """
    f.__doc__ = f.__doc__.replace("@operator@", text)
    return f

[docs] @_parallel @_operator def adjacency(g, weight=None, vindex=None, operator=False, csr=True): r"""Return the adjacency matrix of the graph. Parameters ---------- g : :class:`~graph_tool.Graph` Graph to be used. weight : :class:`~graph_tool.EdgePropertyMap` (optional, default: ``None``) Edge property map with the edge weights. vindex : :class:`~graph_tool.VertexPropertyMap` (optional, default: ``None``) Vertex property map specifying the row/column indices. If not provided, the internal vertex index is used. operator : ``bool`` (optional, default: ``False``) If ``True``, a :class:`scipy.sparse.linalg.LinearOperator` subclass is returned, instead of a sparse matrix. csr : ``bool`` (optional, default: ``True``) If ``True``, and ``operator`` is ``False``, a :class:`scipy.sparse.csr_matrix` sparse matrix is returned, otherwise a :class:`scipy.sparse.coo_matrix` is returned instead. Returns ------- A : :class:`~scipy.sparse.csr_matrix` or :class:`AdjacencyOperator` The (sparse) adjacency matrix. Notes ----- For undirected graphs, the adjacency matrix is defined as .. math:: A_{ij} = \begin{cases} 1 & \text{if } (j, i) \in E, \\ 2 & \text{if } i = j \text{ and } (i, i) \in E, \\ 0 & \text{otherwise}, \end{cases} where :math:`E` is the edge set. For directed graphs, we have instead simply .. math:: A_{ij} = \begin{cases} 1 & \text{if } (j, i) \in E, \\ 0 & \text{otherwise}. \end{cases} In the case of weighted edges, the entry values are multiplied by the weight of the respective edge. In the case of networks with parallel edges, the entries in the matrix become simply the edge multiplicities (or twice them for the diagonal, for undirected graphs). .. note:: For directed graphs the definition above means that the entry :math:`A_{ij}` corresponds to the directed edge :math:`j\to i`. Although this is a typical definition in network and graph theory literature, many also use the transpose of this matrix. @operator@ Examples -------- .. testsetup:: import scipy.linalg from pylab import * >>> g = gt.collection.data["polblogs"] >>> A = gt.adjacency(g, operator=True) >>> N = g.num_vertices() >>> ew1 = scipy.sparse.linalg.eigs(A, k=N//2, which="LR", return_eigenvectors=False) >>> ew2 = scipy.sparse.linalg.eigs(A, k=N-N//2, which="SR", return_eigenvectors=False) >>> ew = np.concatenate((ew1, ew2)) >>> figure(figsize=(8, 2)) <...> >>> scatter(real(ew), imag(ew), c=sqrt(abs(ew)), linewidths=0, alpha=0.6) <...> >>> xlabel(r"$\operatorname{Re}(\lambda)$") Text(...) >>> ylabel(r"$\operatorname{Im}(\lambda)$") Text(...) >>> tight_layout() >>> savefig("adjacency-spectrum.svg") .. figure:: adjacency-spectrum.* :align: center Adjacency matrix spectrum for the political blogs network. References ---------- .. [wikipedia-adjacency] http://en.wikipedia.org/wiki/Adjacency_matrix """ if operator: return AdjacencyOperator(g, weight=weight, vindex=vindex) if vindex is None: if g.get_vertex_filter() is not None: vindex = g.new_vertex_property("int64_t") vindex.fa = numpy.arange(g.num_vertices()) else: vindex = g.vertex_index E = g.num_edges() if g.is_directed() else 2 * g.num_edges() data = numpy.zeros(E, dtype="double") i = numpy.zeros(E, dtype="int32") j = numpy.zeros(E, dtype="int32") libgraph_tool_spectral.adjacency(g._Graph__graph, _prop("v", g, vindex), _prop("e", g, weight), data, i, j) if E > 0: V = max(g.num_vertices(), max(i.max() + 1, j.max() + 1)) else: V = g.num_vertices() m = scipy.sparse.coo_matrix((data, (i,j)), shape=(V, V)) if csr: m = m.tocsr() return m
[docs] class AdjacencyOperator(scipy.sparse.linalg.LinearOperator): def __init__(self, g, weight=None, vindex=None): r"""A :class:`scipy.sparse.linalg.LinearOperator` representing the adjacency matrix of a graph. See :func:`adjacency` for details.""" self.g = g self.weight = weight if vindex is None: if g.get_vertex_filter() is not None: self.vindex = g.new_vertex_property("int64_t") self.vindex.fa = numpy.arange(g.num_vertices()) N = g.num_vertices() else: self.vindex = g.vertex_index N = g.num_vertices() else: self.vindex = vindex if vindex is vindex.get_graph().vertex_index: N = g.num_vertices() else: N = vindex.fa.max() + 1 self.shape = (N, N) self.dtype = numpy.dtype("float") def _matvec(self, x): y = numpy.zeros(self.shape[0]) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.adjacency_matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), x, y) return y def _matmat(self, x): y = numpy.zeros((self.shape[0], x.shape[1])) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.adjacency_matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), x, y) return y def _adjoint(self): if self.g.is_directed(): return AdjacencyOperator(GraphView(self.g, reversed=True), self.weight, self.vindex) else: return self
[docs] @_limit_args({"deg": ["total", "in", "out"]}) @_parallel @_operator def laplacian(g, deg="out", norm=False, weight=None, r=1, vindex=None, operator=False, csr=True): r"""Return the Laplacian (or Bethe Hessian if :math:`r > 1`) matrix of the graph. Parameters ---------- g : :class:`~graph_tool.Graph` Graph to be used. deg : str (optional, default: "out") Degree to be used, in case of a directed graph. norm : bool (optional, default: False) Whether to compute the normalized Laplacian. weight : :class:`~graph_tool.EdgePropertyMap` (optional, default: ``None``) Edge property map with the edge weights. r : ``double`` (optional, default: ``1.``) Regularization parameter. If :math:`r > 1`, and ``norm`` is ``False``, then this corresponds to the Bethe Hessian. (This parameter has an effect only for ``norm == False``.) vindex : :class:`~graph_tool.VertexPropertyMap` (optional, default: ``None``) Vertex property map specifying the row/column indices. If not provided, the internal vertex index is used. operator : ``bool`` (optional, default: ``False``) If ``True``, a :class:`scipy.sparse.linalg.LinearOperator` subclass is returned, instead of a sparse matrix. csr : ``bool`` (optional, default: ``True``) If ``True``, and ``operator`` is ``False``, a :class:`scipy.sparse.csr_matrix` sparse matrix is returned, otherwise a :class:`scipy.sparse.coo_matrix` is returned instead. Returns ------- L : :class:`~scipy.sparse.csr_matrix` or :class:`LaplacianOperator` The (sparse) Laplacian matrix. Notes ----- The weighted Laplacian matrix is defined as .. math:: \ell_{ij} = \begin{cases} \Gamma(v_i) & \text{if } i = j \\ -w_{ij} & \text{if } i \neq j \text{ and } (j, i) \in E \\ 0 & \text{otherwise}. \end{cases} Where :math:`\Gamma(v_i)=\sum_j A_{ij}w_{ij}` is sum of the weights of the edges incident on vertex :math:`v_i`. In case of :math:`r > 1`, the matrix returned is the Bethe Hessian [bethe-hessian]_: .. math:: \ell_{ij} = \begin{cases} \Gamma(v_i) + (r^2 - 1) & \text{if } i = j \\ -r w_{ij} & \text{if } i \neq j \text{ and } (j, i) \in E \\ 0 & \text{otherwise}. \end{cases} The normalized version is .. math:: \ell_{ij} = \begin{cases} 1 & \text{ if } i = j \text{ and } \Gamma(v_i) \neq 0 \\ -\frac{w_{ij}}{\sqrt{\Gamma(v_i)\Gamma(v_j)}} & \text{ if } i \neq j \text{ and } (j, i) \in E \\ 0 & \text{otherwise}. \end{cases} In the case of unweighted edges, it is assumed :math:`w_{ij} = 1`. For directed graphs, it is assumed :math:`\Gamma(v_i)=\sum_j A_{ij}w_{ij} + \sum_j A_{ji}w_{ji}` if ``deg=="total"``, :math:`\Gamma(v_i)=\sum_j A_{ji}w_{ji}` if ``deg=="out"`` or :math:`\Gamma(v_i)=\sum_j A_{ij}w_{ij}` if ``deg=="in"``. .. note:: For directed graphs the definition above means that the entry :math:`\ell_{i,j}` corresponds to the directed edge :math:`j\to i`. Although this is a typical definition in network and graph theory literature, many also use the transpose of this matrix. @operator@ Examples -------- .. testsetup:: import scipy.linalg from pylab import * >>> g = gt.collection.data["polblogs"] >>> L = gt.laplacian(g, operator=True) >>> N = g.num_vertices() >>> ew1 = scipy.sparse.linalg.eigs(L, k=N//2, which="LR", return_eigenvectors=False) >>> ew2 = scipy.sparse.linalg.eigs(L, k=N-N//2, which="SR", return_eigenvectors=False) >>> ew = np.concatenate((ew1, ew2)) >>> figure(figsize=(8, 2)) <...> >>> scatter(real(ew), imag(ew), c=sqrt(abs(ew)), linewidths=0, alpha=0.6) <...> >>> xlabel(r"$\operatorname{Re}(\lambda)$") Text(...) >>> ylabel(r"$\operatorname{Im}(\lambda)$") Text(...) >>> tight_layout() >>> savefig("laplacian-spectrum.svg") .. figure:: laplacian-spectrum.* :align: center Laplacian matrix spectrum for the political blogs network. >>> L = gt.laplacian(g, norm=True, operator=True) >>> ew1 = scipy.sparse.linalg.eigs(L, k=N//2, which="LR", return_eigenvectors=False) >>> ew2 = scipy.sparse.linalg.eigs(L, k=N-N//2, which="SR", return_eigenvectors=False) >>> ew = np.concatenate((ew1, ew2)) >>> figure(figsize=(8, 2)) <...> >>> scatter(real(ew), imag(ew), c=sqrt(abs(ew)), linewidths=0, alpha=0.6) <...> >>> xlabel(r"$\operatorname{Re}(\lambda)$") Text(...) >>> ylabel(r"$\operatorname{Im}(\lambda)$") Text(...) >>> tight_layout() >>> savefig("norm-laplacian-spectrum.svg") .. figure:: norm-laplacian-spectrum.* :align: center Normalized Laplacian matrix spectrum for the political blogs network. References ---------- .. [wikipedia-laplacian] http://en.wikipedia.org/wiki/Laplacian_matrix .. [bethe-hessian] Saade, Alaa, Florent Krzakala, and Lenka Zdeborová. "Spectral clustering of graphs with the bethe hessian." Advances in Neural Information Processing Systems 27 (2014): 406-414, :arxiv:`1406.1880`, https://proceedings.neurips.cc/paper/2014/hash/63923f49e5241343aa7acb6a06a751e7-Abstract.html """ if operator: return LaplacianOperator(g, deg=deg, norm=norm, weight=weight, r=r, vindex=vindex) if vindex is None: if g.get_vertex_filter() is not None: vindex = g.new_vertex_property("int64_t") vindex.fa = numpy.arange(g.num_vertices()) else: vindex = g.vertex_index V = g.num_vertices() nself = int(label_self_loops(g, mark_only=True).a.sum()) E = g.num_edges() - nself if not g.is_directed(): E *= 2 N = E + g.num_vertices() data = numpy.zeros(N, dtype="double") i = numpy.zeros(N, dtype="int32") j = numpy.zeros(N, dtype="int32") if norm: libgraph_tool_spectral.norm_laplacian(g._Graph__graph, _prop("v", g, vindex), _prop("e", g, weight), deg, data, i, j) else: libgraph_tool_spectral.laplacian(g._Graph__graph, _prop("v", g, vindex), _prop("e", g, weight), deg, r, data, i, j) if E > 0: V = max(g.num_vertices(), max(i.max() + 1, j.max() + 1)) else: V = g.num_vertices() m = scipy.sparse.coo_matrix((data, (i, j)), shape=(V, V)) if csr: m = m.tocsr() return m
[docs] class LaplacianOperator(scipy.sparse.linalg.LinearOperator): @_limit_args({"deg": ["total", "in", "out"]}) def __init__(self, g, weight=None, deg="out", r=1, norm=False, vindex=None): r"""A :class:`scipy.sparse.linalg.LinearOperator` representing the laplacian matrix of a graph. See :func:`laplacian` for details.""" self.g = g self.weight = weight self.r = r if vindex is None: if g.get_vertex_filter() is not None: self.vindex = g.new_vertex_property("int64_t") self.vindex.fa = numpy.arange(g.num_vertices()) N = g.num_vertices() else: self.vindex = g.vertex_index N = g.num_vertices() else: self.vindex = vindex if vindex is vindex.get_graph().vertex_index: N = g.num_vertices() else: N = vindex.fa.max() + 1 self.shape = (N, N) self.deg = deg self.d = self.g.degree_property_map(deg, weight) if norm: idx = self.d.a > 0 d = g.new_vp("double") d.a[idx] = 1./numpy.sqrt(self.d.a[idx]) self.d = d else: self.d = self.d.copy("double") self.norm = norm self.dtype = numpy.dtype("float") def _matvec(self, x): y = numpy.zeros(self.shape[0]) x = numpy.asarray(x, dtype="float") if self.norm: matvec = libgraph_tool_spectral.norm_laplacian_matvec matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), _prop("v", self.g, self.d), x, y) else: matvec = libgraph_tool_spectral.laplacian_matvec matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), _prop("v", self.g, self.d), self.r, x, y) return y def _matmat(self, x): y = numpy.zeros((self.shape[0], x.shape[1])) x = numpy.asarray(x, dtype="float") if self.norm: matmat = libgraph_tool_spectral.norm_laplacian_matmat matmat(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), _prop("v", self.g, self.d), x, y) else: matmat = libgraph_tool_spectral.laplacian_matmat matmat(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), _prop("v", self.g, self.d), self.r, x, y) return y def _adjoint(self): if self.g.is_directed(): deg = self.deg if deg == "in": deg = "out" elif deg == "out": deg = "in" return LaplacianOperator(GraphView(self.g, reversed=True), self.weight, deg=deg, norm=self.norm, vindex=self.vindex) else: return self
[docs] @_parallel @_operator def incidence(g, vindex=None, eindex=None, operator=False, csr=True): r"""Return the incidence matrix of the graph. Parameters ---------- g : :class:`~graph_tool.Graph` Graph to be used. vindex : :class:`~graph_tool.VertexPropertyMap` (optional, default: ``None``) Vertex property map specifying the row indices. If not provided, the internal vertex index is used. eindex : :class:`~graph_tool.EdgePropertyMap` (optional, default: ``None``) Edge property map specifying the column indices. If not provided, the internal edge index is used. operator : ``bool`` (optional, default: ``False``) If ``True``, a :class:`scipy.sparse.linalg.LinearOperator` subclass is returned, instead of a sparse matrix. csr : ``bool`` (optional, default: ``True``) If ``True``, and ``operator`` is ``False``, a :class:`scipy.sparse.csr_matrix` sparse matrix is returned, otherwise a :class:`scipy.sparse.coo_matrix` is returned instead. Returns ------- a : :class:`~scipy.sparse.csr_matrix` or :class:`IncidenceOperator` The (sparse) incidence matrix. Notes ----- For undirected graphs, the incidence matrix is defined as .. math:: b_{i,j} = \begin{cases} 1 & \text{if vertex } v_i \text{and edge } e_j \text{ are incident}, \\ 0 & \text{otherwise} \end{cases} For directed graphs, the definition is .. math:: b_{i,j} = \begin{cases} 1 & \text{if edge } e_j \text{ enters vertex } v_i, \\ -1 & \text{if edge } e_j \text{ leaves vertex } v_i, \\ 0 & \text{otherwise} \end{cases} @operator@ Examples -------- .. testsetup:: import scipy.linalg from pylab import * >>> g = gt.collection.data["polblogs"] >>> B = gt.incidence(g, operator=True) >>> N = g.num_vertices() >>> s1 = scipy.sparse.linalg.svds(B, k=N//2, which="LM", return_singular_vectors=False) >>> s2 = scipy.sparse.linalg.svds(B, k=N-N//2, which="SM", return_singular_vectors=False) >>> s = np.concatenate((s1, s2)) >>> s.sort() >>> figure(figsize=(8, 2)) <...> >>> plot(s, "s") [...] >>> xlabel(r"$i$") Text(...) >>> ylabel(r"$\lambda_i$") Text(...) >>> tight_layout() >>> savefig("polblogs-indidence-svd.svg") .. figure:: polblogs-indidence-svd.* :align: center Incidence singular values for the political blogs network. References ---------- .. [wikipedia-incidence] http://en.wikipedia.org/wiki/Incidence_matrix """ if operator: return IncidenceOperator(g, vindex=vindex, eindex=eindex) if vindex is None: if g.get_edge_filter() is not None: vindex = g.new_vertex_property("int64_t") vindex.fa = numpy.arange(g.num_vertices()) else: vindex = g.vertex_index if eindex is None: if g.get_edge_filter() is not None: eindex = g.new_edge_property("int64_t") eindex.fa = numpy.arange(g.num_edges()) else: eindex = g.edge_index E = g.num_edges() if E == 0: raise ValueError("Cannot construct incidence matrix for a graph with no edges.") data = numpy.zeros(2 * E, dtype="double") i = numpy.zeros(2 * E, dtype="int32") j = numpy.zeros(2 * E, dtype="int32") libgraph_tool_spectral.incidence(g._Graph__graph, _prop("v", g, vindex), _prop("e", g, eindex), data, i, j) m = scipy.sparse.coo_matrix((data, (i,j))) if csr: m = m.tocsr() return m
[docs] class IncidenceOperator(scipy.sparse.linalg.LinearOperator): def __init__(self, g, vindex=None, eindex=None, transpose=False): r"""A :class:`scipy.sparse.linalg.LinearOperator` representing the incidence matrix of a graph. See :func:`incidence` for details. """ self.g = g self.transpose = transpose self.dtype = numpy.dtype("float") if vindex is None: if g.get_vertex_filter() is not None: self.vindex = g.new_vertex_property("int64_t") self.vindex.fa = numpy.arange(g.num_vertices()) N = g.num_vertices() else: self.vindex = g.vertex_index N = g.num_vertices() else: self.vindex = vindex if vindex is vindex.get_graph().vertex_index: N = g.num_vertices() else: N = vindex.fa.max() + 1 if eindex is None: if g.get_edge_filter() is not None: self.eindex = g.new_edge_property("int64_t") self.eindex.fa = numpy.arange(g.num_edges()) E = g.num_edges() else: self.eindex = g.edge_index E = g.edge_index_range else: self.eindex = eindex if eindex is g.edge_index: E = g.edge_index_range else: E = self.eindex.fa.max() + 1 if not transpose: self.shape = (N, E) else: self.shape = (E, N) def _matvec(self, x): y = numpy.zeros(self.shape[0]) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.incidence_matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.eindex), x, y, self.transpose) return y def _matmat(self, x): y = numpy.zeros((self.shape[0], x.shape[1])) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.incidence_matmat(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.eindex), x, y, self.transpose) return y def _adjoint(self): return IncidenceOperator(self.g, vindex=self.vindex, eindex=self.eindex, transpose=not self.transpose)
[docs] @_parallel @_operator def transition(g, weight=None, vindex=None, operator=False, csr=True): r"""Return the transition matrix of the graph. Parameters ---------- g : :class:`~graph_tool.Graph` Graph to be used. weight : :class:`~graph_tool.EdgePropertyMap` (optional, default: ``None``) Edge property map with the edge weights. vindex : :class:`~graph_tool.VertexPropertyMap` (optional, default: ``None``) Vertex property map specifying the row/column indices. If not provided, the internal vertex index is used. operator : ``bool`` (optional, default: ``False``) If ``True``, a :class:`scipy.sparse.linalg.LinearOperator` subclass is returned, instead of a sparse matrix. csr : ``bool`` (optional, default: ``True``) If ``True``, and ``operator`` is ``False``, a :class:`scipy.sparse.csr_matrix` sparse matrix is returned, otherwise a :class:`scipy.sparse.coo_matrix` is returned instead. Returns ------- T : :class:`~scipy.sparse.csr_matrix` or :class:`TransitionOperator` The (sparse) transition matrix. Notes ----- The transition matrix is defined as .. math:: T_{ij} = \frac{A_{ij}}{k_j} where :math:`k_i = \sum_j A_{ji}`, and :math:`A_{ij}` is the adjacency matrix. In the case of weighted edges, the values of the adjacency matrix are multiplied by the edge weights. .. note:: For directed graphs the definition above means that the entry :math:`T_{ij}` corresponds to the directed edge :math:`j\to i`. Although this is a typical definition in network and graph theory literature, many also use the transpose of this matrix. @operator@ Examples -------- .. testsetup:: import scipy.linalg from pylab import * >>> g = gt.collection.data["polblogs"] >>> T = gt.transition(g, operator=True) >>> N = g.num_vertices() >>> ew1 = scipy.sparse.linalg.eigs(T, k=N//2, which="LR", return_eigenvectors=False) >>> ew2 = scipy.sparse.linalg.eigs(T, k=N-N//2, which="SR", return_eigenvectors=False) >>> ew = np.concatenate((ew1, ew2)) >>> figure(figsize=(8, 2)) <...> >>> scatter(real(ew), imag(ew), c=sqrt(abs(ew)), linewidths=0, alpha=0.6) <...> >>> xlabel(r"$\operatorname{Re}(\lambda)$") Text(...) >>> ylabel(r"$\operatorname{Im}(\lambda)$") Text(...) >>> tight_layout() >>> savefig("transition-spectrum.svg") .. figure:: transition-spectrum.* :align: center Transition matrix spectrum for the political blogs network. References ---------- .. [wikipedia-transition] https://en.wikipedia.org/wiki/Stochastic_matrix """ if operator: return TransitionOperator(g, weight=weight, vindex=vindex) if vindex is None: if g.get_vertex_filter() is not None: vindex = g.new_vertex_property("int64_t") vindex.fa = numpy.arange(g.num_vertices()) else: vindex = g.vertex_index E = g.num_edges() if g.is_directed() else 2 * g.num_edges() data = numpy.zeros(E, dtype="double") i = numpy.zeros(E, dtype="int32") j = numpy.zeros(E, dtype="int32") libgraph_tool_spectral.transition(g._Graph__graph, _prop("v", g, vindex), _prop("e", g, weight), data, i, j) if E > 0: V = max(g.num_vertices(), max(i.max() + 1, j.max() + 1)) else: V = g.num_vertices() m = scipy.sparse.coo_matrix((data, (i,j)), shape=(V, V)) if csr: m = m.tocsr() return m
[docs] class TransitionOperator(scipy.sparse.linalg.LinearOperator): def __init__(self, g, weight=None, transpose=False, inv_d=None, vindex=None): r"""A :class:`scipy.sparse.linalg.LinearOperator` representing the transition matrix of a graph. See :func:`transition` for details. """ self.g = g self.weight = weight if vindex is None: if g.get_vertex_filter() is not None: self.vindex = g.new_vertex_property("int64_t") self.vindex.fa = numpy.arange(g.num_vertices()) N = g.num_vertices() else: self.vindex = g.vertex_index N = g.num_vertices() else: self.vindex = vindex if vindex is vindex.get_graph().vertex_index: N = g.num_vertices() else: N = vindex.fa.max() + 1 self.shape = (N, N) if inv_d is None: d = self.g.degree_property_map("out", weight) nd = g.new_vp("double") idx = d.a > 0 nd.a[idx] = 1/d.a[idx] self.d = nd else: self.d = inv_d.copy("double") self.dtype = numpy.dtype("float") self.transpose = transpose def _matvec(self, x): y = numpy.zeros(self.shape[0]) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.transition_matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), _prop("v", self.g, self.d), x, y, self.transpose) return y def _matmat(self, x): y = numpy.zeros((self.shape[0], x.shape[1])) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.transition_matmat(self.g._Graph__graph, _prop("v", self.g, self.vindex), _prop("e", self.g, self.weight), _prop("v", self.g, self.d), x, y, self.transpose) return y def _adjoint(self): if self.g.is_directed(): g = GraphView(self.g, reversed=True) else: g = self.g return TransitionOperator(g, self.weight, inv_d=self.d, transpose=not self.transpose, vindex=self.vindex)
[docs] @_parallel @_operator def modularity_matrix(g, weight=None, vindex=None): r"""Return the modularity matrix of the graph. Parameters ---------- g : :class:`~graph_tool.Graph` Graph to be used. weight : :class:`~graph_tool.EdgePropertyMap` (optional, default: ``None``) Edge property map with the edge weights. index : :class:`~graph_tool.VertexPropertyMap` (optional, default: ``None``) Vertex property map specifying the row/column indices. If not provided, the internal vertex index is used. Returns ------- B : :class:`~scipy.sparse.linalg.LinearOperator` The (sparse) modularity matrix, represented as a :class:`~scipy.sparse.linalg.LinearOperator`. Notes ----- The modularity matrix is defined as .. math:: B_{ij} = A_{ij} - \frac{k^+_i k^-_j}{2E} where :math:`k^+_i = \sum_j A_{ji}`, :math:`k^-_i = \sum_j A_{ij}`, :math:`2E=\sum_{ij}A_{ij}` and :math:`A_{ij}` is the adjacency matrix. In the case of weighted edges, the values of the adjacency matrix are multiplied by the edge weights. Examples -------- .. testsetup:: import scipy.linalg from pylab import * >>> g = gt.collection.data["polblogs"] >>> B = gt.modularity_matrix(g) >>> N = g.num_vertices() >>> ew1 = scipy.sparse.linalg.eigs(B, k=N//2, which="LR", return_eigenvectors=False) >>> ew2 = scipy.sparse.linalg.eigs(B, k=N-N//2, which="SR", return_eigenvectors=False) >>> ew = np.concatenate((ew1, ew2)) >>> figure(figsize=(8, 2)) <...> >>> scatter(real(ew), imag(ew), c=sqrt(abs(ew)), linewidths=0, alpha=0.6) <...> >>> xlabel(r"$\operatorname{Re}(\lambda)$") Text(...) >>> ylabel(r"$\operatorname{Im}(\lambda)$") Text(...) >>> autoscale() >>> tight_layout() >>> savefig("modularity-spectrum.svg") .. figure:: modularity-spectrum.* :align: center Modularity matrix spectrum for the political blogs network. References ---------- .. [newman-modularity] M. E. J. Newman, M. Girvan, "Finding and evaluating community structure in networks", Phys. Rev. E 69, 026113 (2004). :doi:`10.1103/PhysRevE.69.026113` """ A = adjacency(g, weight=weight, vindex=vindex, operator=True) A_T = A.adjoint() if g.is_directed(): k_in = g.degree_property_map("in", weight=weight).fa else: k_in = g.degree_property_map("out", weight=weight).fa k_out = g.degree_property_map("out", weight=weight).fa N = g.num_vertices() E2 = float(k_out.sum()) def matvec(x): M = x.shape[0] if len(x.shape) > 1: x = x.reshape(M) nx = A.matvec(x) - k_out * numpy.dot(k_in, x) / E2 return nx def rmatvec(x): M = x.shape[0] if len(x.shape) > 1: x = x.reshape(M) nx = A_T.matvec(x) - k_in * numpy.dot(k_out, x) / E2 return nx B = scipy.sparse.linalg.LinearOperator((g.num_vertices(), g.num_vertices()), matvec=matvec, rmatvec=rmatvec, dtype="float") return B
[docs] @_parallel @_operator def hashimoto(g, index=None, compact=False, operator=False, csr=True): r"""Return the Hashimoto (or non-backtracking) matrix of a graph. Parameters ---------- g : :class:`~graph_tool.Graph` Graph to be used. index : :class:`~graph_tool.VertexPropertyMap` (optional, default: ``None``) Edge property map specifying the row/column indices. If not provided, the internal edge index is used. compact : ``boolean`` (optional, default: ``False``) If ``True``, a compact :math:`2|V|\times 2|V|` version of the matrix is returned. operator : ``bool`` (optional, default: ``False``) If ``True``, a :class:`scipy.sparse.linalg.LinearOperator` subclass is returned, instead of a sparse matrix. csr : ``bool`` (optional, default: ``True``) If ``True``, and ``operator`` is ``False``, a :class:`scipy.sparse.csr_matrix` sparse matrix is returned, otherwise a :class:`scipy.sparse.coo_matrix` is returned instead. Returns ------- H : :class:`~scipy.sparse.csr_matrix` or :class:`HashimotoOperator` or :class:`CompactHashimotoOperator` The (sparse) Hashimoto matrix. Notes ----- The Hashimoto (a.k.a. non-backtracking) matrix [hashimoto]_ is defined as .. math:: h_{k\to l,i\to j} = \begin{cases} 1 & \text{if } (k,l) \in E, (i,j) \in E, l=i, k\ne j,\\ 0 & \text{otherwise}, \end{cases} where :math:`E` is the edge set. It is therefore a :math:`2|E|\times 2|E|` asymmetric square matrix (or :math:`|E|\times |E|` for directed graphs), indexed over edge directions. If the option ``compact=True`` is passed, the matrix returned has shape :math:`2|V|\times 2|V|`, where :math:`|V|` is the number of vertices, and is given by .. math:: \boldsymbol h = \left(\begin{array}{c|c} \boldsymbol A & -\boldsymbol 1 \\ \hline \boldsymbol D-\boldsymbol 1 & \boldsymbol 0 \end{array}\right) where :math:`\boldsymbol A` is the adjacency matrix, and :math:`\boldsymbol D` is the diagonal matrix with the node degrees [krzakala_spectral]_. @operator@ Examples -------- .. testsetup:: import scipy.linalg from pylab import * >>> g = gt.collection.data["football"] >>> H = gt.hashimoto(g, operator=True) >>> N = 2 * g.num_edges() >>> ew1 = scipy.sparse.linalg.eigs(H, k=N//2, which="LR", return_eigenvectors=False) >>> ew2 = scipy.sparse.linalg.eigs(H, k=N-N//2, which="SR", return_eigenvectors=False) >>> ew = np.concatenate((ew1, ew2)) >>> figure(figsize=(8, 4)) <...> >>> scatter(real(ew), imag(ew), c=sqrt(abs(ew)), linewidths=0, alpha=0.6) <...> >>> xlabel(r"$\operatorname{Re}(\lambda)$") Text(...) >>> ylabel(r"$\operatorname{Im}(\lambda)$") Text(...) >>> tight_layout() >>> savefig("hashimoto-spectrum.svg") .. figure:: hashimoto-spectrum.* :align: center Hashimoto matrix spectrum for the network of American football teams. References ---------- .. [hashimoto] Hashimoto, Ki-ichiro. "Zeta functions of finite graphs and representations of p-adic groups." Automorphic forms and geometry of arithmetic varieties. 1989. 211-280. :DOI:`10.1016/B978-0-12-330580-0.50015-X` .. [krzakala_spectral] Florent Krzakala, Cristopher Moore, Elchanan Mossel, Joe Neeman, Allan Sly, Lenka Zdeborová, and Pan Zhang, "Spectral redemption in clustering sparse networks", PNAS 110 (52) 20935-20940, 2013. :doi:`10.1073/pnas.1312486110`, :arxiv:`1306.5550` """ if compact: if operator: return CompactHashimotoOperator(g) i = Vector_int64_t() j = Vector_int64_t() x = Vector_double() libgraph_tool_spectral.compact_nonbacktracking(g._Graph__graph, i, j, x) N = g.num_vertices(ignore_filter=True) m = scipy.sparse.coo_matrix((x, (i.a,j.a)), shape=(2 * N, 2 * N)) else: if operator: return HashimotoOperator(g, eindex=index) if index is None: if g.get_edge_filter() is not None: index = g.new_edge_property("int64_t") index.fa = numpy.arange(g.num_edges()) E = index.fa.max() + 1 else: index = g.edge_index E = g.edge_index_range if not g.is_directed(): E *= 2 i = Vector_int64_t() j = Vector_int64_t() libgraph_tool_spectral.nonbacktracking(g._Graph__graph, _prop("e", g, index), i, j) data = numpy.ones(i.a.shape) m = scipy.sparse.coo_matrix((data, (i.a,j.a)), shape=(E, E)) if csr: m = m.tocsr() return m
[docs] class HashimotoOperator(scipy.sparse.linalg.LinearOperator): def __init__(self, g, eindex=None, transpose=False): r"""A :class:`scipy.sparse.linalg.LinearOperator` representing the hashimoto matrix of a graph. See :func:`hashimoto` for details. """ self.g = g if eindex is None: if g.get_edge_filter() is not None: self.eindex = g.new_edge_property("int64_t") self.eindex.fa = numpy.arange(g.num_edges()) E = g.num_edges() else: self.eindex = g.edge_index E = g.edge_index_range else: self.eindex = eindex if eindex is g.edge_index: E = g.edge_index_range else: E = self.eindex.fa.max() + 1 if g.is_directed(): self.shape = (E, E) else: self.shape = (2 * E, 2 * E) self.dtype = numpy.dtype("float") self.transpose = transpose def _matvec(self, x): y = numpy.zeros(self.shape[0]) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.nonbacktracking_matvec(self.g._Graph__graph, _prop("e", self.g, self.eindex), x, y, self.transpose) return y def _matmat(self, x): y = numpy.zeros((self.shape[0], x.shape[1])) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.nonbacktracking_matmat(self.g._Graph__graph, _prop("e", self.g, self.eindex), x, y, self.transpose) return y def _adjoint(self): if self.g.is_directed(): g = GraphView(self.g, reversed=True) else: g = self.g return HashimotoOperator(g, self.eindex, transpose=not self.transpose)
[docs] class CompactHashimotoOperator(scipy.sparse.linalg.LinearOperator): def __init__(self, g, vindex=None, transpose=False): r"""A :class:`scipy.sparse.linalg.LinearOperator` representing the compact hashimoto matrix of a graph. See :func:`hashimoto` for details. """ self.g = g if vindex is None: if g.get_vertex_filter() is not None: self.vindex = g.new_vertex_property("int64_t") self.vindex.fa = numpy.arange(g.num_vertices()) N = g.num_vertices() else: self.vindex = g.vertex_index N = g.num_vertices() else: self.vindex = vindex if vindex is vindex.get_graph().vertex_index: N = g.num_vertices() else: N = vindex.fa.max() + 1 self.shape = (2 * N, 2 * N) self.dtype = numpy.dtype("float") self.transpose = transpose def _matvec(self, x): y = numpy.zeros(self.shape[0]) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.compact_nonbacktracking_matvec(self.g._Graph__graph, _prop("v", self.g, self.vindex), x, y, self.transpose) return y def _matmat(self, x): y = numpy.zeros((self.shape[0], x.shape[1])) x = numpy.asarray(x, dtype="float") libgraph_tool_spectral.compact_nonbacktracking_matmat(self.g._Graph__graph, _prop("v", self.g, self.vindex), x, y, self.transpose) return y def _adjoint(self): if self.g.is_directed(): g = GraphView(self.g, reversed=True) else: g = self.g return CompactHashimotoOperator(g, self.vindex, transpose=not self.transpose)