{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "toc_visible": true
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "cells": [
    {
      "cell_type": "markdown",
      "source": [
        "# Unconstrained Non-Monotone Submodular Maximization"
      ],
      "metadata": {
        "id": "fiPHq611S_db"
      }
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "FOj4I-MZRLyb",
        "outputId": "1ec97d12-6658-4428-9a5c-f23d7ab5bbc2"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "Example: Unweighted graph cut (n = 6)\n",
            "Algorithm value: 9  |  Optimum: 9  |  Set found: [1, 2, 3]\n",
            "\n",
            "Example: Weighted graph cut (n = 8)\n",
            "Mean value over 300 runs: 42.88\n",
            "Best run value: 48.11  |  Optimum: 48.11\n"
          ]
        }
      ],
      "source": [
        "import random, itertools, statistics, math\n",
        "import numpy as np\n",
        "import tqdm\n",
        "\n",
        "def randomized_double_greedy(f, ground_set, seed=None):\n",
        "    \"\"\"\n",
        "    Randomized double‑greedy algorithm (Buchbinder et al., FOCS 2012) for\n",
        "    unconstrained non‑monotone submodular maximization.\n",
        "    https://theory.epfl.ch/moranfe/Publications/FOCS2012.pdf\n",
        "\n",
        "    Parameters\n",
        "    ----------\n",
        "    f : callable\n",
        "        Oracle that maps a subset (any iterable of ground‑set elements) to a real value.\n",
        "    ground_set : iterable\n",
        "        The ground set V.\n",
        "    seed : int, optional\n",
        "        RNG seed for reproducibility.\n",
        "\n",
        "    Returns\n",
        "    -------\n",
        "    chosen_set : set\n",
        "        The set returned by the algorithm.\n",
        "    value : float\n",
        "        f(chosen_set).\n",
        "    \"\"\"\n",
        "    # print(f(ground_set))\n",
        "    rng = random.Random(seed)\n",
        "    A, B = set(), set(ground_set)\n",
        "\n",
        "    for e in list(ground_set):\n",
        "        delta_add    = f(A | {e}) - f(A)\n",
        "        delta_remove = f(B - {e}) - f(B)\n",
        "\n",
        "        # Use non‑negative gains as in the paper's analysis\n",
        "        pos_add, pos_rem = max(delta_add, 0.0), max(delta_remove, 0.0)\n",
        "        p = 0.5 if pos_add + pos_rem == 0 else pos_add / (pos_add + pos_rem)\n",
        "\n",
        "        if rng.random() < p:\n",
        "            A.add(e)\n",
        "        else:\n",
        "            B.remove(e)      # equivalent to discarding e from B\n",
        "\n",
        "    return A, f(A)\n",
        "\n",
        "\n",
        "def brute_force_optimum(f, ground_set):\n",
        "    \"\"\"Exhaustive search — feasible only for tiny ground sets.\"\"\"\n",
        "    best_S, best_val = None, float(\"-inf\")\n",
        "    for r in range(len(ground_set) + 1):\n",
        "        for S in itertools.combinations(ground_set, r):\n",
        "            val = f(S)\n",
        "            if val > best_val:\n",
        "                best_S, best_val = set(S), val\n",
        "    return best_S, best_val\n",
        "\n",
        "\n",
        "# ---------- Example: Symmetric cut function on an unweighted complete graph ----------\n",
        "n = 6\n",
        "V = list(range(n))\n",
        "def cut_unweighted(S, n=n):\n",
        "    k = len(S)\n",
        "    return k * (n - k)          # |S| · |V\\S|\n",
        "print(\"\\nExample: Unweighted graph cut (n = 6)\")\n",
        "sol, val = randomized_double_greedy(lambda S: cut_unweighted(S), V, seed=42)\n",
        "opt_S, opt_val = brute_force_optimum(lambda S: cut_unweighted(S), V)\n",
        "print(f\"Algorithm value: {val}  |  Optimum: {opt_val}  |  Set found: {sorted(sol)}\")\n",
        "\n",
        "\n",
        "# ---------- Example: Weighted graph cut ----------\n",
        "n3 = 8\n",
        "V3 = list(range(n3))\n",
        "random.seed(3)\n",
        "weights_w = [[0.0]*n3 for _ in V3]\n",
        "for i in range(n3):\n",
        "    for j in range(i+1, n3):\n",
        "        w = random.uniform(0, 5)\n",
        "        weights_w[i][j] = weights_w[j][i] = w\n",
        "\n",
        "def cut_weighted(S, W=weights_w):\n",
        "    S = set(S)\n",
        "    total = 0.0\n",
        "    for i in range(len(W)):\n",
        "        for j in range(i+1, len(W)):\n",
        "            if (i in S) ^ (j in S):\n",
        "                total += W[i][j]\n",
        "    return total\n",
        "\n",
        "print(\"\\nExample: Weighted graph cut (n = 8)\")\n",
        "vals3 = []\n",
        "for run in range(300):\n",
        "    _, v = randomized_double_greedy(cut_weighted, V3, seed=run)\n",
        "    vals3.append(v)\n",
        "print(f\"Mean value over 300 runs: {statistics.mean(vals3):.2f}\")\n",
        "opt_S3, opt_val3 = brute_force_optimum(cut_weighted, V3)\n",
        "print(f\"Best run value: {max(vals3):.2f}  |  Optimum: {opt_val3:.2f}\")\n"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "class NoisyValueOracle:\n",
        "    \"\"\"\n",
        "    Wrapper that turns a deterministic value‑oracle `f` into a *noisy* oracle\n",
        "    f_noisy(S) = x_S * f(S).\n",
        "\n",
        "    • x_S has mean 1 and user‑specified variance.\n",
        "    • {x_S}_S are *independent* for different sets.\n",
        "    • Repeated queries on the *same* set return the *same* value.\n",
        "\n",
        "    Parameters\n",
        "    ----------\n",
        "    f : callable\n",
        "        Original deterministic oracle.\n",
        "    dist : {\"normal\", \"uniform\"}, default \"normal\"\n",
        "        Distribution family for the multiplicative noise x_S.\n",
        "    variance : float, default 0.1\n",
        "        Desired variance of x_S (must be < 1 for the uniform case to keep\n",
        "        the support positive).\n",
        "    seed : int, optional\n",
        "        Seed for reproducibility.\n",
        "    \"\"\"\n",
        "    def __init__(self, f, *, dist=\"normal\", variance=0.1, seed=None):\n",
        "        if variance < 0:\n",
        "            raise ValueError(\"variance must be non‑negative\")\n",
        "        self.f = f\n",
        "        self.dist = dist\n",
        "        self.variance = variance\n",
        "        self.rng = random.Random(seed)\n",
        "        self.noise_cache = {}       # maps frozenset(S) -> x_S\n",
        "        self.seed = seed\n",
        "\n",
        "        if dist == \"normal\":\n",
        "            self.sigma = math.sqrt(variance)          # N(1, σ²)\n",
        "        elif dist == \"uniform\":\n",
        "            width = math.sqrt(12 * variance)          # Var(U(a,b)) = (b‑a)² / 12\n",
        "            if width >= 2:\n",
        "                raise ValueError(\"variance too large for uniform mean‑1 support\")\n",
        "            self.a = 1 - width / 2\n",
        "            self.b = 1 + width / 2\n",
        "        else:\n",
        "            raise ValueError(\"dist must be 'normal' or 'uniform'\")\n",
        "\n",
        "    # ---------- internal helpers ----------\n",
        "    def _draw_noise(self, seed, key):\n",
        "        rng = random.Random(f'{seed}_{key}')\n",
        "        if self.dist == \"normal\":\n",
        "            return rng.gauss(1, self.sigma)\n",
        "        else:  # uniform\n",
        "            return rng.uniform(self.a, self.b)\n",
        "\n",
        "    # ---------- the oracle interface ----------\n",
        "    def __call__(self, S):\n",
        "        key = frozenset(S)          # ensures that order / duplicates don’t matter\n",
        "        noise = self._draw_noise(self.seed, key)\n",
        "        # if key not in self.noise_cache:\n",
        "        #     self.noise_cache[key] = self._draw_noise()\n",
        "        # return self.noise_cache[key] * self.f(S)\n",
        "        return noise * self.f(S)\n",
        "\n",
        "\n",
        "# -------------------- Demo --------------------\n",
        "def demo():\n",
        "    # A simple base oracle: f(S) = |S|·(5‑|S|) on V = {0,…,4}\n",
        "    V = list(range(5))\n",
        "    def cut(S):\n",
        "        k = len(S)\n",
        "        return k * (len(V) - k)\n",
        "\n",
        "    oracle = NoisyValueOracle(cut, dist=\"uniform\", variance=0.04, seed=0)\n",
        "\n",
        "    S = {1, 2}\n",
        "    print(\"First three queries on the *same* set S = {1,2}:\")\n",
        "    for _ in range(3):\n",
        "        print(f\"    f_noisy(S) = {oracle(S):.4f}\")\n",
        "\n",
        "    print(\"\\nQueries on all 32 subsets (showing mean & variance of the noise):\")\n",
        "    noises = []\n",
        "    for r in range(len(V)+1):\n",
        "        for subset in itertools.combinations(V, r):\n",
        "            val = oracle(subset)\n",
        "            noises.append(val / cut(subset) if cut(subset) != 0 else 1.0)\n",
        "    print(f\"    empirical mean  ≈ {statistics.mean(noises):.4f}\")\n",
        "    print(f\"    empirical var   ≈ {statistics.variance(noises):.4f}\")\n",
        "\n",
        "demo()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "zK5gD9xuc9SF",
        "outputId": "d782a0fa-3654-45e1-a24c-a2111a53d3b3"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "First three queries on the *same* set S = {1,2}:\n",
            "    f_noisy(S) = 6.4419\n",
            "    f_noisy(S) = 6.4419\n",
            "    f_noisy(S) = 6.4419\n",
            "\n",
            "Queries on all 32 subsets (showing mean & variance of the noise):\n",
            "    empirical mean  ≈ 1.0329\n",
            "    empirical var   ≈ 0.0358\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "class SurrogateFunction:\n",
        "    \"\"\"g(S) = (1/m) ∑_{i=1}^m f_noisy(S ∪ H_i).\n",
        "\n",
        "    • A set H⊂V of size h is given as input.\n",
        "    • m subsets H_i (size t, sampled *with* replacement from H) are\n",
        "      also fixed at construction.\n",
        "    • Re‑querying the same S returns the same value because f_noisy\n",
        "      caches its noise and the H_i are fixed.\n",
        "    \"\"\"\n",
        "    def __init__(self, f_noisy, ground_set, H, t, m, seed=None):\n",
        "        self.f_noisy = f_noisy\n",
        "        rng          = random.Random(seed)\n",
        "        self.H       = list(H)\n",
        "        self.H_subs  = [frozenset(rng.sample(self.H, t)) for _ in range(m)]\n",
        "\n",
        "    def __call__(self, S):\n",
        "        S = set(S)\n",
        "        return sum(self.f_noisy(S | Hi) for Hi in self.H_subs) / len(self.H_subs)\n",
        "\n",
        "def meta_algorithm_double_greedy(f_noisy, ground_set, h, t, m, seed=None):\n",
        "    rng  = random.Random(seed)\n",
        "    H    = rng.sample(list(ground_set), h)            # Step 2\n",
        "    g    = SurrogateFunction(f_noisy, ground_set, H, t, m, seed=seed)\n",
        "    rem  = [e for e in ground_set if e not in H]      # contracted ground set\n",
        "    S_H, _ = randomized_double_greedy(g, rem, seed=seed)  # Step 3\n",
        "    H_prime = set(rng.sample(H, t))                   # Step 4\n",
        "    return S_H | H_prime                              # Step 5\n",
        "\n",
        "# ---------- 5. Tiny demonstration ----------\n",
        "if __name__ == \"__main__\":\n",
        "    n, V  = 10, list(range(10))\n",
        "    f     = lambda S: len(S)*(n-len(S))               # symmetric cut\n",
        "    f_noisy = NoisyValueOracle(f, variance=0.05, seed=7)\n",
        "    sol   = meta_algorithm_double_greedy(f_noisy, V, h=4, t=2, m=30, seed=42)\n",
        "    print(\"Meta‑algorithm solution:\", sorted(sol))\n",
        "    print(\"True objective value   :\", f(sol))"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "dDtFXz4zlgDP",
        "outputId": "414280dd-bef2-44c5-910b-8771a47da717"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Meta‑algorithm solution: [0, 1, 3, 5, 6]\n",
            "True objective value   : 25\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "## Example 1: Additive Weights Minus Quadratic Cost"
      ],
      "metadata": {
        "id": "11hQrnzsq6sJ"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "# ---------- Example 1: Additive weights minus quadratic cost ----------\n",
        "\n",
        "def get_modular_weights_quadratic_cost_function(n, weights, c):\n",
        "  def f(S):\n",
        "    S = list(S)\n",
        "    return sum(weights[i] for i in S) - c * (len(S)**2)\n",
        "  return f\n",
        "\n",
        "def greedy_opt_modular_minus_quadratic(weights, c):\n",
        "    \"\"\"\n",
        "    Optimal algorithm for f(S) = sum_{i∈S} w_i - c|S|^2.\n",
        "\n",
        "    Parameters\n",
        "    ----------\n",
        "    weights : list[float]\n",
        "        The item weights w_i.\n",
        "    c       : float, default 0.1\n",
        "        Quadratic penalty coefficient.\n",
        "\n",
        "    Returns\n",
        "    -------\n",
        "    chosen  : list[int]\n",
        "        Indices (into `weights`) of the selected items, in the order added.\n",
        "    value   : float\n",
        "        Optimal objective value f(chosen).\n",
        "    \"\"\"\n",
        "    # 1. sort indices by non‑increasing weight\n",
        "    idx_sorted = sorted(range(len(weights)),\n",
        "                        key=lambda i: weights[i],\n",
        "                        reverse=True)\n",
        "    best_val = 0.0           # k = 0  (empty set)\n",
        "    best_k   = 0\n",
        "    cum_sum  = 0.0           # Σ_{i≤k} w_(i)\n",
        "    for k, i in enumerate(idx_sorted, start=1):\n",
        "        cum_sum += weights[i]\n",
        "        val = cum_sum - c * k * k\n",
        "        if val >= best_val:          # non‑decreasing so far\n",
        "            best_val = val\n",
        "            best_k   = k\n",
        "        else:                        # value has dropped ⇒ stop\n",
        "            break\n",
        "    chosen = idx_sorted[:best_k]\n",
        "    return chosen, best_val\n",
        "\n",
        "\n",
        "def check_minus_quadratic_nonnegative(weights, c):\n",
        "    \"\"\"\n",
        "    Checks if f(S) = sum_{i∈S} w_i - c|S|^2 is ever negative.\n",
        "\n",
        "    Parameters\n",
        "    ----------\n",
        "    weights : list[float]\n",
        "        The item weights w_i.\n",
        "    c       : float, default 0.1\n",
        "        Quadratic penalty coefficient.\n",
        "\n",
        "    Returns\n",
        "    -------\n",
        "    True / False depending on if it is always nonnegative\n",
        "    \"\"\"\n",
        "    # 1. sort indices by non‑increasing weight\n",
        "    idx_sorted = sorted(range(len(weights)),\n",
        "                        key=lambda i: weights[i],\n",
        "                        reverse=False)\n",
        "    cum_sum = 0.\n",
        "    for k, i in enumerate(idx_sorted, start=1):\n",
        "        cum_sum += weights[i]\n",
        "        val = cum_sum - c * k * k\n",
        "        if val < 0:\n",
        "          return False\n",
        "    return True\n",
        "\n",
        "\n",
        "def one_experiment(n, runs, htm_list, verbose=False):\n",
        "  mean_weight = 10\n",
        "  c = mean_weight * n / n**2\n",
        "  groundset = list(range(n))\n",
        "  print(f\"Experimen with: Modular weights – c|S|² (n={n}, c={c})\\n\")\n",
        "\n",
        "  exact_DG_values = np.zeros(runs)\n",
        "  random_set = np.zeros(runs)\n",
        "  noisy_DG_values = np.zeros(runs)\n",
        "  denoised_DG_values = [np.zeros(runs) for j in range(len(htm_list))]\n",
        "\n",
        "  for i in tqdm.tqdm(range(runs)):\n",
        "    # Generate a non-negative submodular function\n",
        "    while True:\n",
        "      weights=[random.uniform(0, 2*mean_weight) for i in range(n)]\n",
        "      if check_minus_quadratic_nonnegative(weights, c):\n",
        "        break\n",
        "    f = get_modular_weights_quadratic_cost_function(n, weights, c)\n",
        "    optimal = greedy_opt_modular_minus_quadratic(weights, c)[1]\n",
        "    if verbose:\n",
        "      print(f\" * Optimal value: {optimal}\")\n",
        "\n",
        "    # Double greedy algorithm with exact value oracle\n",
        "    exact_DG_values[i] = randomized_double_greedy(f, groundset, seed=i)[1] / optimal\n",
        "\n",
        "    # Random subset\n",
        "    s = int(n/2)\n",
        "    random_set[i] = f(random.sample(range(n), s)) / optimal\n",
        "\n",
        "    # Double greedy algorithm with noisy value oracle\n",
        "    f_noisy = NoisyValueOracle(f, dist=\"normal\", variance=0.1, seed=i)\n",
        "    noisy_DG_values[i] = f(randomized_double_greedy(f_noisy, groundset, seed=i)[0]) / optimal\n",
        "\n",
        "    # Our algorithm: double greedy algorithm on surrogate function\n",
        "    for j in range(len(htm_list)):\n",
        "      (h, t, m) = htm_list[j]\n",
        "      denoised_DG_values[j][i] = f(meta_algorithm_double_greedy(f_noisy, groundset, h, t, m, seed=i)) / optimal\n",
        "\n",
        "  exact_DG_mean = statistics.mean(exact_DG_values)\n",
        "  exact_DG_std = statistics.stdev(exact_DG_values)\n",
        "  random_mean = statistics.mean(random_set)\n",
        "  random_std = statistics.stdev(random_set)\n",
        "  noisy_DG_mean = statistics.mean(noisy_DG_values)\n",
        "  noisy_DG_std = statistics.stdev(noisy_DG_values)\n",
        "\n",
        "  print(\"\\n\")\n",
        "  print(f\"* Exact DG:\\n\"\n",
        "      f\"    Mean over {runs} runs : {exact_DG_mean:.3f} (std: {exact_DG_std:.3f})\\n\"\n",
        "  )\n",
        "  print(f\"* Noisy DG:\\n\"\n",
        "      f\"    Mean over {runs} runs : {noisy_DG_mean:.3f} (std: {noisy_DG_std:.3f})\\n\"\n",
        "  )\n",
        "  print(f\"* Random set of size {s}:\\n\"\n",
        "      f\"    Mean over {runs} runs : {random_mean:.3f} (std: {random_std:.3f})\\n\"\n",
        "  )\n",
        "\n",
        "  for j in range(len(htm_list)):\n",
        "    (h, t, m) = htm_list[j]\n",
        "    denoised_DG_mean = statistics.mean(denoised_DG_values[j])\n",
        "    denoised_DG_std = statistics.stdev(denoised_DG_values[j])\n",
        "    print(f\"* Our algorithm with m={m}, h={h}, t={t}:\\n\"\n",
        "        f\"    Mean over {runs} runs : {denoised_DG_mean:.3f} (std: {denoised_DG_std:.3f})\\n\"\n",
        "  )\n",
        "\n",
        "\n",
        "htm_list = [(20, 4, 50), (20, 4, 100), (20, 4, 150), (20, 4, 200)]\n",
        "one_experiment(n=20, runs=10, htm_list=htm_list, verbose=False)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "QJyCfJMdwGuF",
        "outputId": "f91f0d42-4056-4f0f-d6a5-24d38fac0545"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Experimen with: Modular weights – c|S|² (n=20, c=0.5)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "100%|██████████| 10/10 [00:00<00:00, 96.96it/s]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "\n",
            "* Exact DG:\n",
            "    Mean over 10 runs : 0.929 (std: 0.061)\n",
            "\n",
            "* Noisy DG:\n",
            "    Mean over 10 runs : 0.642 (std: 0.103)\n",
            "\n",
            "* Random set of size 10:\n",
            "    Mean over 10 runs : 0.619 (std: 0.074)\n",
            "\n",
            "* Our algorithm with m=50, h=20, t=4:\n",
            "    Mean over 10 runs : 0.291 (std: 0.066)\n",
            "\n",
            "* Our algorithm with m=100, h=20, t=4:\n",
            "    Mean over 10 runs : 0.291 (std: 0.066)\n",
            "\n",
            "* Our algorithm with m=150, h=20, t=4:\n",
            "    Mean over 10 runs : 0.291 (std: 0.066)\n",
            "\n",
            "* Our algorithm with m=200, h=20, t=4:\n",
            "    Mean over 10 runs : 0.291 (std: 0.066)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "one_experiment(n=100, runs=500, htm_list=htm_list, verbose=False)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "YRI1BUIODZ38",
        "outputId": "3b1c223f-0139-43e9-eaf6-8c3378eadf95"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Experimen with: Modular weights – c|S|² (n=100, c=0.1)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "100%|██████████| 500/500 [32:36<00:00,  3.91s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "\n",
            "* Exact DG:\n",
            "    Mean over 500 runs : 0.945 (std: 0.019)\n",
            "\n",
            "* Noisy DG:\n",
            "    Mean over 500 runs : 0.573 (std: 0.055)\n",
            "\n",
            "* Random set of size 50:\n",
            "    Mean over 500 runs : 0.533 (std: 0.055)\n",
            "\n",
            "* Our algorithm with m=50, h=20, t=4:\n",
            "    Mean over 500 runs : 0.653 (std: 0.048)\n",
            "\n",
            "* Our algorithm with m=100, h=20, t=4:\n",
            "    Mean over 500 runs : 0.690 (std: 0.046)\n",
            "\n",
            "* Our algorithm with m=150, h=20, t=4:\n",
            "    Mean over 500 runs : 0.714 (std: 0.046)\n",
            "\n",
            "* Our algorithm with m=200, h=20, t=4:\n",
            "    Mean over 500 runs : 0.732 (std: 0.044)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "one_experiment(n=50, runs=1000, htm_list=[(20, 4, 50), (20, 4, 200)], verbose=False)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "QskC3WWdorBc",
        "outputId": "213d424d-08a1-42ef-b240-20a195ab24b6"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Experimen with: Modular weights – c|S|² (n=50, c=0.2)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "100%|██████████| 1000/1000 [09:43<00:00,  1.71it/s]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "\n",
            "* Exact DG:\n",
            "    Mean over 1000 runs : 0.944 (std: 0.028)\n",
            "\n",
            "* Noisy DG:\n",
            "    Mean over 1000 runs : 0.601 (std: 0.074)\n",
            "\n",
            "* Random set of size 25:\n",
            "    Mean over 1000 runs : 0.550 (std: 0.079)\n",
            "\n",
            "* Our algorithm with m=50, h=20, t=4:\n",
            "    Mean over 1000 runs : 0.674 (std: 0.067)\n",
            "\n",
            "* Our algorithm with m=200, h=20, t=4:\n",
            "    Mean over 1000 runs : 0.735 (std: 0.059)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "one_experiment(n=100, runs=1000, htm_list=[(20, 4, 50), (20, 4, 200)], verbose=False)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "0i23gfQxgOHQ",
        "outputId": "0ece982e-5aaa-4601-8f83-3f633bacb210"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Experimen with: Modular weights – c|S|² (n=100, c=0.1)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "100%|██████████| 1000/1000 [32:47<00:00,  1.97s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "\n",
            "* Exact DG:\n",
            "    Mean over 1000 runs : 0.944 (std: 0.019)\n",
            "\n",
            "* Noisy DG:\n",
            "    Mean over 1000 runs : 0.565 (std: 0.055)\n",
            "\n",
            "* Random set of size 50:\n",
            "    Mean over 1000 runs : 0.536 (std: 0.057)\n",
            "\n",
            "* Our algorithm with m=50, h=20, t=4:\n",
            "    Mean over 1000 runs : 0.657 (std: 0.047)\n",
            "\n",
            "* Our algorithm with m=200, h=20, t=4:\n",
            "    Mean over 1000 runs : 0.731 (std: 0.041)\n",
            "\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\n"
          ]
        }
      ]
    }
  ]
}