{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "y79IIGtv08z4"
      },
      "outputs": [],
      "source": []
    },
    {
      "cell_type": "code",
      "source": [
        "from torch_geometric.datasets import ZINC\n",
        "import torch\n",
        "import numpy as np\n",
        "\n",
        "def update_rows_with_distance_weights(eigenvectors, C=0.1):\n",
        "   N = eigenvectors.size(0)\n",
        "   x_expanded = eigenvectors.unsqueeze(1)\n",
        "   x_other = eigenvectors.unsqueeze(0)\n",
        "   distances = torch.sum((x_expanded - x_other) ** 2, dim=-1)\n",
        "   weights = torch.exp(-distances)\n",
        "   weights.fill_diagonal_(0)\n",
        "   diffs = x_expanded - x_other\n",
        "   weighted_diffs = diffs * weights[..., None]\n",
        "   updates = weighted_diffs.sum(dim=1)\n",
        "   return eigenvectors + C * updates\n",
        "\n",
        "def has_distinct_eigenvalues(eigenvalues, tol=1e-4):\n",
        "   sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "   differences = sorted_eigs[1:] - sorted_eigs[:-1]\n",
        "   return torch.all(differences > tol).item()\n",
        "\n",
        "def count_multiplicity_two_eigenvalues(eigenvalues, tol=1e-4):\n",
        "    sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "    count = 0\n",
        "    i = 0\n",
        "    while i < len(sorted_eigs) - 1:\n",
        "        if abs(sorted_eigs[i] - sorted_eigs[i + 1]) <= tol:\n",
        "            if i + 2 >= len(sorted_eigs) or abs(sorted_eigs[i] - sorted_eigs[i + 2]) > tol:\n",
        "                count += 1\n",
        "                i += 2\n",
        "            else:\n",
        "                i += 1\n",
        "        else:\n",
        "            i += 1\n",
        "    return count\n",
        "\n",
        "def count_multiplicity_three_eigenvalues(eigenvalues, tol=1e-4):\n",
        "    sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "    count = 0\n",
        "    i = 0\n",
        "    while i < len(sorted_eigs) - 2:\n",
        "        if (abs(sorted_eigs[i] - sorted_eigs[i + 1]) <= tol and\n",
        "            abs(sorted_eigs[i] - sorted_eigs[i + 2]) <= tol):\n",
        "            if i + 3 >= len(sorted_eigs) or abs(sorted_eigs[i] - sorted_eigs[i + 3]) > tol:\n",
        "                count += 1\n",
        "                i += 3\n",
        "            else:\n",
        "                i += 1\n",
        "        else:\n",
        "            i += 1\n",
        "    return count\n",
        "\n",
        "dataset = ZINC(root='data/ZINC', subset=True, split='train')\n",
        "print(f\"\\nAnalyzing ZINC dataset:\")\n",
        "print(f\"Number of graphs: {len(dataset)}\")\n",
        "\n",
        "total_zeros = 0\n",
        "total_ratios = 0\n",
        "counter = 0\n",
        "graphs_with_full_rows = 0\n",
        "graphs_with_at_most_one_zero = 0\n",
        "graphs_with_few_zeros = 0\n",
        "graphs_meeting_any_condition = 0\n",
        "distinct_eigenvalues_count = 0\n",
        "multiplicity_two_count = 0\n",
        "multiplicity_three_count = 0\n",
        "\n",
        "for idx, graph in enumerate(dataset):\n",
        "    num_nodes = graph.num_nodes\n",
        "    edge_index = graph.edge_index\n",
        "\n",
        "    adj_matrix = torch.zeros((num_nodes, num_nodes))\n",
        "    adj_matrix[edge_index[0], edge_index[1]] = 1\n",
        "    adj_matrix = adj_matrix + adj_matrix.T\n",
        "    adj_matrix[adj_matrix > 1] = 1\n",
        "\n",
        "    degree = adj_matrix.sum(dim=1)\n",
        "    degree_matrix = torch.diag(degree)\n",
        "    laplacian = degree_matrix - adj_matrix\n",
        "\n",
        "    eigenvalues, eigenvectors = torch.linalg.eigh(laplacian)\n",
        "\n",
        "    if has_distinct_eigenvalues(eigenvalues):\n",
        "        distinct_eigenvalues_count += 1\n",
        "\n",
        "    if count_multiplicity_two_eigenvalues(eigenvalues) >= 1:\n",
        "        multiplicity_two_count += 1\n",
        "\n",
        "    if count_multiplicity_three_eigenvalues(eigenvalues) >= 1:\n",
        "        multiplicity_three_count += 1\n",
        "\n",
        "    updated_eigenvectors = eigenvectors.clone()#update_rows_with_distance_weights(eigenvectors, C=0.1)\n",
        "\n",
        "    zero_eigenvectors = torch.sum(torch.abs(updated_eigenvectors) < 1e-4).item()\n",
        "    ratio = zero_eigenvectors / num_nodes\n",
        "    total_zeros += zero_eigenvectors\n",
        "\n",
        "    has_full_row = torch.any(~torch.any(torch.abs(updated_eigenvectors) < 1e-4, dim=1)).item()\n",
        "    if has_full_row:\n",
        "        graphs_with_full_rows += 1\n",
        "\n",
        "    zeros_per_column = torch.sum(torch.abs(updated_eigenvectors) < 1e-4, dim=0)\n",
        "    at_most_one_zero = torch.all(zeros_per_column <= 1).item()\n",
        "    if at_most_one_zero:\n",
        "        graphs_with_at_most_one_zero += 1\n",
        "\n",
        "    if zero_eigenvectors < num_nodes:\n",
        "        graphs_with_few_zeros += 1\n",
        "\n",
        "    if has_full_row or at_most_one_zero or zero_eigenvectors < num_nodes:\n",
        "        graphs_meeting_any_condition += 1\n",
        "\n",
        "    total_ratios += ratio\n",
        "    counter += 1\n",
        "\n",
        "if counter > 0:\n",
        "    print(f\"Graphs with distinct eigenvalues: {distinct_eigenvalues_count}/{len(dataset)} ({(distinct_eigenvalues_count/len(dataset))*100:.1f}%)\")\n",
        "    print(f\"Average ratio of zeros: {total_ratios/counter:.3f}\")\n",
        "    print(f\"Average zeros: {total_zeros/counter:.3f}\")\n",
        "    print(f\"Graphs with full rows: {graphs_with_full_rows}/{counter} ({(graphs_with_full_rows/counter)*100:.1f}%)\")\n",
        "    print(f\"Graphs with ≤1 zero per eigenvector: {graphs_with_at_most_one_zero}/{counter} ({(graphs_with_at_most_one_zero/counter)*100:.1f}%)\")\n",
        "    print(f\"Graphs with total zeros < vertices: {graphs_with_few_zeros}/{counter} ({(graphs_with_few_zeros/counter)*100:.1f}%)\")\n",
        "    print(f\"Graphs meeting any condition: {graphs_meeting_any_condition}/{counter} ({(graphs_meeting_any_condition/counter)*100:.1f}%)\")\n",
        "    print(f\"Graphs with multiplicity 2 eigenvalues: {multiplicity_two_count}/{counter} ({(multiplicity_two_count/counter)*100:.1f}%)\")\n",
        "    print(f\"Graphs with multiplicity 3 eigenvalues: {multiplicity_three_count}/{counter} ({(multiplicity_three_count/counter)*100:.1f}%)\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "S38JlO0Yoev-",
        "outputId": "e298c01d-5d4b-4000-8a12-f998dab91ea0"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "Analyzing ZINC dataset:\n",
            "Number of graphs: 10000\n",
            "Graphs with distinct eigenvalues: 4072/10000 (40.7%)\n",
            "Average ratio of zeros: 2.520\n",
            "Average zeros: 61.035\n",
            "Graphs with full rows: 6447/10000 (64.5%)\n",
            "Graphs with ≤1 zero per eigenvector: 430/10000 (4.3%)\n",
            "Graphs with total zeros < vertices: 1295/10000 (13.0%)\n",
            "Graphs meeting any condition: 6447/10000 (64.5%)\n",
            "Graphs with multiplicity 2 eigenvalues: 5928/10000 (59.3%)\n",
            "Graphs with multiplicity 3 eigenvalues: 2617/10000 (26.2%)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "from torch_geometric.datasets import TUDataset\n",
        "import torch\n",
        "import numpy as np\n",
        "\n",
        "def update_rows_with_distance_weights(eigenvectors, C=0.1):\n",
        "   N = eigenvectors.size(0)\n",
        "   x_expanded = eigenvectors.unsqueeze(1)\n",
        "   x_other = eigenvectors.unsqueeze(0)\n",
        "   distances = torch.sum((x_expanded - x_other) ** 2, dim=-1)\n",
        "   weights = torch.exp(-distances)\n",
        "   weights.fill_diagonal_(0)\n",
        "   diffs = x_expanded - x_other\n",
        "   weighted_diffs = diffs * weights[..., None]\n",
        "   updates = weighted_diffs.sum(dim=1)\n",
        "   return eigenvectors + C * updates\n",
        "\n",
        "def has_distinct_eigenvalues(eigenvalues, tol=1e-4):\n",
        "   sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "   differences = sorted_eigs[1:] - sorted_eigs[:-1]\n",
        "   return torch.all(differences > tol).item()\n",
        "\n",
        "def count_multiplicity_two_eigenvalues(eigenvalues, tol=1e-4):\n",
        "    sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "    count = 0\n",
        "    i = 0\n",
        "    while i < len(sorted_eigs) - 1:\n",
        "        if abs(sorted_eigs[i] - sorted_eigs[i + 1]) <= tol:\n",
        "            if i + 2 >= len(sorted_eigs) or abs(sorted_eigs[i] - sorted_eigs[i + 2]) > tol:\n",
        "                count += 1\n",
        "                i += 2\n",
        "            else:\n",
        "                i += 1\n",
        "        else:\n",
        "            i += 1\n",
        "    return count\n",
        "\n",
        "def count_multiplicity_three_eigenvalues(eigenvalues, tol=1e-4):\n",
        "    sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "    count = 0\n",
        "    i = 0\n",
        "    while i < len(sorted_eigs) - 2:\n",
        "        if (abs(sorted_eigs[i] - sorted_eigs[i + 1]) <= tol and\n",
        "            abs(sorted_eigs[i] - sorted_eigs[i + 2]) <= tol):\n",
        "            if i + 3 >= len(sorted_eigs) or abs(sorted_eigs[i] - sorted_eigs[i + 3]) > tol:\n",
        "                count += 1\n",
        "                i += 3\n",
        "            else:\n",
        "                i += 1\n",
        "        else:\n",
        "            i += 1\n",
        "    return count\n",
        "\n",
        "dataset_names = ['MUTAG', 'ENZYMES', 'PROTEINS']\n",
        "\n",
        "for name in dataset_names:\n",
        "   dataset = TUDataset(root='data/TU', name=name)\n",
        "   print(f\"\\nAnalyzing {name} dataset:\")\n",
        "   print(f\"Number of graphs: {len(dataset)}\")\n",
        "\n",
        "   total_zeros = 0\n",
        "   total_ratios = 0\n",
        "   counter = 0\n",
        "   graphs_with_full_rows = 0\n",
        "   graphs_with_at_most_one_zero = 0\n",
        "   graphs_with_few_zeros = 0\n",
        "   graphs_meeting_any_condition = 0\n",
        "   distinct_eigenvalues_count = 0\n",
        "   multiplicity_two_count = 0\n",
        "   multiplicity_three_count = 0\n",
        "   total_mult_two = 0\n",
        "   total_mult_three = 0\n",
        "\n",
        "   for idx, graph in enumerate(dataset):\n",
        "       num_nodes = graph.num_nodes\n",
        "       edge_index = graph.edge_index\n",
        "\n",
        "       adj_matrix = torch.zeros((num_nodes, num_nodes))\n",
        "       adj_matrix[edge_index[0], edge_index[1]] = 1\n",
        "       adj_matrix = adj_matrix + adj_matrix.T\n",
        "       adj_matrix[adj_matrix > 1] = 1\n",
        "\n",
        "       degree = adj_matrix.sum(dim=1)\n",
        "       degree_matrix = torch.diag(degree)\n",
        "       laplacian = degree_matrix - adj_matrix\n",
        "\n",
        "       eigenvalues, eigenvectors = torch.linalg.eigh(laplacian)\n",
        "\n",
        "       if has_distinct_eigenvalues(eigenvalues):\n",
        "           distinct_eigenvalues_count += 1\n",
        "\n",
        "       mult_two = count_multiplicity_two_eigenvalues(eigenvalues)\n",
        "       mult_three = count_multiplicity_three_eigenvalues(eigenvalues)\n",
        "\n",
        "       if mult_two >= 1:\n",
        "           multiplicity_two_count += 1\n",
        "       if mult_three >= 1:\n",
        "           multiplicity_three_count += 1\n",
        "\n",
        "       total_mult_two += mult_two\n",
        "       total_mult_three += mult_three\n",
        "\n",
        "       updated_eigenvectors = eigenvectors.clone()#update_rows_with_distance_weights(eigenvectors, C=0.1)\n",
        "\n",
        "       zero_eigenvectors = torch.sum(torch.abs(updated_eigenvectors) < 1e-4).item()\n",
        "       ratio = zero_eigenvectors / num_nodes\n",
        "       total_zeros += zero_eigenvectors\n",
        "\n",
        "       has_full_row = torch.any(~torch.any(torch.abs(updated_eigenvectors) < 1e-4, dim=1)).item()\n",
        "       if has_full_row:\n",
        "           graphs_with_full_rows += 1\n",
        "\n",
        "       zeros_per_column = torch.sum(torch.abs(updated_eigenvectors) < 1e-4, dim=0)\n",
        "       at_most_one_zero = torch.all(zeros_per_column <= 1).item()\n",
        "       if at_most_one_zero:\n",
        "           graphs_with_at_most_one_zero += 1\n",
        "\n",
        "       if zero_eigenvectors < num_nodes:\n",
        "           graphs_with_few_zeros += 1\n",
        "\n",
        "       if has_full_row or at_most_one_zero or zero_eigenvectors < num_nodes:\n",
        "           graphs_meeting_any_condition += 1\n",
        "\n",
        "       total_ratios += ratio\n",
        "       counter += 1\n",
        "\n",
        "   if counter > 0:\n",
        "       print(\"\\nEigenvalue Statistics:\")\n",
        "       print(f\"Graphs with distinct eigenvalues: {distinct_eigenvalues_count}/{len(dataset)} ({(distinct_eigenvalues_count/len(dataset))*100:.1f}%)\")\n",
        "       print(f\"Graphs with multiplicity 2 eigenvalues: {multiplicity_two_count}/{counter} ({(multiplicity_two_count/counter)*100:.1f}%)\")\n",
        "       print(f\"Graphs with multiplicity 3 eigenvalues: {multiplicity_three_count}/{counter} ({(multiplicity_three_count/counter)*100:.1f}%)\")\n",
        "       print(f\"Average number of multiplicity 2 eigenvalues per graph: {total_mult_two/counter:.2f}\")\n",
        "       print(f\"Average number of multiplicity 3 eigenvalues per graph: {total_mult_three/counter:.2f}\")\n",
        "\n",
        "       print(\"\\nEigenvector Statistics:\")\n",
        "       print(f\"Average ratio of zeros: {total_ratios/counter:.3f}\")\n",
        "       print(f\"Average zeros: {total_zeros/counter:.3f}\")\n",
        "       print(f\"Graphs with full rows: {graphs_with_full_rows}/{counter} ({(graphs_with_full_rows/counter)*100:.1f}%)\")\n",
        "       print(f\"Graphs with ≤1 zero per eigenvector: {graphs_with_at_most_one_zero}/{counter} ({(graphs_with_at_most_one_zero/counter)*100:.1f}%)\")\n",
        "       print(f\"Graphs with total zeros < vertices: {graphs_with_few_zeros}/{counter} ({(graphs_with_few_zeros/counter)*100:.1f}%)\")\n",
        "       print(f\"Graphs meeting any condition: {graphs_meeting_any_condition}/{counter} ({(graphs_meeting_any_condition/counter)*100:.1f}%)\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Ei9qfXKJCVb_",
        "outputId": "7875502b-166a-4bfd-d0d0-0e767c8dd5d9"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "Analyzing MUTAG dataset:\n",
            "Number of graphs: 188\n",
            "\n",
            "Eigenvalue Statistics:\n",
            "Graphs with distinct eigenvalues: 78/188 (41.5%)\n",
            "Graphs with multiplicity 2 eigenvalues: 110/188 (58.5%)\n",
            "Graphs with multiplicity 3 eigenvalues: 36/188 (19.1%)\n",
            "Average number of multiplicity 2 eigenvalues per graph: 0.74\n",
            "Average number of multiplicity 3 eigenvalues per graph: 0.26\n",
            "\n",
            "Eigenvector Statistics:\n",
            "Average ratio of zeros: 1.665\n",
            "Average zeros: 31.128\n",
            "Graphs with full rows: 141/188 (75.0%)\n",
            "Graphs with ≤1 zero per eigenvector: 0/188 (0.0%)\n",
            "Graphs with total zeros < vertices: 56/188 (29.8%)\n",
            "Graphs meeting any condition: 141/188 (75.0%)\n",
            "\n",
            "Analyzing ENZYMES dataset:\n",
            "Number of graphs: 600\n",
            "\n",
            "Eigenvalue Statistics:\n",
            "Graphs with distinct eigenvalues: 209/600 (34.8%)\n",
            "Graphs with multiplicity 2 eigenvalues: 391/600 (65.2%)\n",
            "Graphs with multiplicity 3 eigenvalues: 277/600 (46.2%)\n",
            "Average number of multiplicity 2 eigenvalues per graph: 1.01\n",
            "Average number of multiplicity 3 eigenvalues per graph: 0.58\n",
            "\n",
            "Eigenvector Statistics:\n",
            "Average ratio of zeros: 4.275\n",
            "Average zeros: 172.927\n",
            "Graphs with full rows: 215/600 (35.8%)\n",
            "Graphs with ≤1 zero per eigenvector: 38/600 (6.3%)\n",
            "Graphs with total zeros < vertices: 98/600 (16.3%)\n",
            "Graphs meeting any condition: 215/600 (35.8%)\n",
            "\n",
            "Analyzing PROTEINS dataset:\n",
            "Number of graphs: 1113\n",
            "\n",
            "Eigenvalue Statistics:\n",
            "Graphs with distinct eigenvalues: 246/1113 (22.1%)\n",
            "Graphs with multiplicity 2 eigenvalues: 867/1113 (77.9%)\n",
            "Graphs with multiplicity 3 eigenvalues: 644/1113 (57.9%)\n",
            "Average number of multiplicity 2 eigenvalues per graph: 1.24\n",
            "Average number of multiplicity 3 eigenvalues per graph: 0.71\n",
            "\n",
            "Eigenvector Statistics:\n",
            "Average ratio of zeros: 6.388\n",
            "Average zeros: 817.202\n",
            "Graphs with full rows: 413/1113 (37.1%)\n",
            "Graphs with ≤1 zero per eigenvector: 56/1113 (5.0%)\n",
            "Graphs with total zeros < vertices: 159/1113 (14.3%)\n",
            "Graphs meeting any condition: 413/1113 (37.1%)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "from torch_geometric.datasets import MNISTSuperpixels\n",
        "import torch\n",
        "import numpy as np\n",
        "\n",
        "def update_rows_with_distance_weights(eigenvectors, C=0.1):\n",
        "   N = eigenvectors.size(0)\n",
        "   x_expanded = eigenvectors.unsqueeze(1)\n",
        "   x_other = eigenvectors.unsqueeze(0)\n",
        "   distances = torch.sum((x_expanded - x_other) ** 2, dim=-1)\n",
        "   weights = torch.exp(-distances)\n",
        "   weights.fill_diagonal_(0)\n",
        "   diffs = x_expanded - x_other\n",
        "   weighted_diffs = diffs * weights[..., None]\n",
        "   updates = weighted_diffs.sum(dim=1)\n",
        "   return eigenvectors + C * updates\n",
        "\n",
        "def has_distinct_eigenvalues(eigenvalues, tol=1e-4):\n",
        "   sorted_eigs = torch.sort(eigenvalues)[0]\n",
        "   differences = sorted_eigs[1:] - sorted_eigs[:-1]\n",
        "   return torch.all(differences > tol).item()\n",
        "\n",
        "dataset = MNISTSuperpixels(root='data/MNIST', train=True)\n",
        "print(f\"Number of graphs: {len(dataset)}\")\n",
        "\n",
        "total_zeros = 0\n",
        "total_ratios = 0\n",
        "counter = 0\n",
        "graphs_with_full_rows = 0\n",
        "graphs_with_at_most_one_zero = 0\n",
        "graphs_with_few_zeros = 0\n",
        "graphs_meeting_any_condition = 0\n",
        "distinct_eigenvalues_count = 0\n",
        "\n",
        "for idx, graph in enumerate(dataset):\n",
        "   num_nodes = graph.num_nodes\n",
        "   edge_index = graph.edge_index\n",
        "\n",
        "   adj_matrix = torch.zeros((num_nodes, num_nodes))\n",
        "   adj_matrix[edge_index[0], edge_index[1]] = 1\n",
        "   adj_matrix = adj_matrix + adj_matrix.T\n",
        "   adj_matrix[adj_matrix > 1] = 1\n",
        "\n",
        "   degree = adj_matrix.sum(dim=1)\n",
        "   degree_matrix = torch.diag(degree)\n",
        "   laplacian = degree_matrix - adj_matrix\n",
        "\n",
        "   eigenvalues, eigenvectors = torch.linalg.eigh(laplacian)\n",
        "\n",
        "   if not has_distinct_eigenvalues(eigenvalues):\n",
        "       continue\n",
        "\n",
        "   distinct_eigenvalues_count += 1\n",
        "   updated_eigenvectors = eigenvectors.clone()#update_rows_with_distance_weights(eigenvectors, C=0.1)\n",
        "\n",
        "   zero_eigenvectors = torch.sum(torch.abs(updated_eigenvectors) < 1e-4).item()\n",
        "   ratio = zero_eigenvectors / num_nodes\n",
        "   total_zeros += zero_eigenvectors\n",
        "\n",
        "   has_full_row = torch.any(~torch.any(torch.abs(updated_eigenvectors) < 1e-4, dim=1)).item()\n",
        "   if has_full_row:\n",
        "       graphs_with_full_rows += 1\n",
        "\n",
        "   zeros_per_column = torch.sum(torch.abs(updated_eigenvectors) < 1e-4, dim=0)\n",
        "   at_most_one_zero = torch.all(zeros_per_column <= 1).item()\n",
        "   if at_most_one_zero:\n",
        "       graphs_with_at_most_one_zero += 1\n",
        "\n",
        "   if zero_eigenvectors < num_nodes:\n",
        "       graphs_with_few_zeros += 1\n",
        "\n",
        "   if has_full_row or at_most_one_zero or zero_eigenvectors < num_nodes:\n",
        "       graphs_meeting_any_condition += 1\n",
        "\n",
        "   total_ratios += ratio\n",
        "   counter += 1\n",
        "\n",
        "if counter > 0:\n",
        "   print(f\"Graphs with distinct eigenvalues: {distinct_eigenvalues_count}/{len(dataset)} ({(distinct_eigenvalues_count/len(dataset))*100:.1f}%)\")\n",
        "   print(f\"Average ratio of zeros: {total_ratios/counter:.3f}\")\n",
        "   print(f\"Average zeros: {total_zeros/counter:.3f}\")\n",
        "   print(f\"Graphs with full rows: {graphs_with_full_rows}/{counter} ({(graphs_with_full_rows/counter)*100:.1f}%)\")\n",
        "   print(f\"Graphs with ≤1 zero per eigenvector: {graphs_with_at_most_one_zero}/{counter} ({(graphs_with_at_most_one_zero/counter)*100:.1f}%)\")\n",
        "   print(f\"Graphs with total zeros < vertices: {graphs_with_few_zeros}/{counter} ({(graphs_with_few_zeros/counter)*100:.1f}%)\")\n",
        "   print(f\"Graphs meeting any condition: {graphs_meeting_any_condition}/{counter} ({(graphs_meeting_any_condition/counter)*100:.1f}%)\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "-lS3VpAFJ3Ks",
        "outputId": "8d499399-2834-42be-9cf3-a1dde638ce59"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "Downloading https://data.pyg.org/datasets/MNISTSuperpixels.zip\n",
            "Extracting data/MNIST/raw/MNISTSuperpixels.zip\n",
            "Processing...\n",
            "Done!\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Number of graphs: 60000\n",
            "Graphs with distinct eigenvalues: 59950/60000 (99.9%)\n",
            "Average ratio of zeros: 0.309\n",
            "Average zeros: 23.158\n",
            "Graphs with full rows: 58077/59950 (96.9%)\n",
            "Graphs with ≤1 zero per eigenvector: 12085/59950 (20.2%)\n",
            "Graphs with total zeros < vertices: 53873/59950 (89.9%)\n",
            "Graphs meeting any condition: 58077/59950 (96.9%)\n"
          ]
        }
      ]
    }
  ]
}