{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b433b2b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import math"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "9c4c9b95",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Data Generation\n",
    "\n",
    "n,d = 100, 30\n",
    "\n",
    "L = np.ones(n)\n",
    "mu = .1 * np.ones(n)\n",
    "\n",
    "# Generate A\n",
    "\n",
    "A = []\n",
    "for k in range(n):\n",
    "    evalues = np.random.uniform(mu[k], L[k], d)\n",
    "    if  k == 0:\n",
    "        evalues[0] = L[k]\n",
    "    elif k == (n-1):\n",
    "        evalues[0] = mu[k]\n",
    "    rndm_mx = np.random.normal(0, 1, (d,d))\n",
    "    _, Q = np.linalg.eig(rndm_mx.T @ rndm_mx)\n",
    "    \n",
    "    A.append(Q @ np.diag(evalues) @ Q.T)\n",
    "\n",
    "# Generate C\n",
    "\n",
    "C = []\n",
    "\n",
    "for k in range(n):\n",
    "    evalues = np.random.uniform(mu[k], L[k], d)\n",
    "    if k == 0:\n",
    "        evalues[0] = L[k]\n",
    "    elif k == (n - 1):\n",
    "        evalues[0] = mu[k]\n",
    "    rndm_mx = np.random.normal(0, 1, (d,d))\n",
    "    _, Q = np.linalg.eig(rndm_mx.T @ rndm_mx)\n",
    "    \n",
    "    C.append(Q @ np.diag(evalues) @ Q.T)\n",
    "    \n",
    "# Generate B\n",
    "\n",
    "B = []\n",
    "mu = np.zeros(n)\n",
    "\n",
    "for k in range(n):\n",
    "    evalues = np.random.uniform(mu[k], L[k], d)\n",
    "    if k == 0:\n",
    "        evalues[0] = L[k]\n",
    "    elif k == (n - 1):\n",
    "        evalues[0] = mu[k]\n",
    "    rndm_mx = np.random.normal(0, 1, (d,d))\n",
    "    _, Q = np.linalg.eig(rndm_mx.T @ rndm_mx)\n",
    "    \n",
    "    B.append(Q @ np.diag(evalues) @ Q.T)\n",
    "    \n",
    "x_optimal = np.random.normal(0, 1, 2*d)\n",
    "\n",
    "M = []\n",
    "z = []\n",
    "L_M = []\n",
    "\n",
    "for k in range(n):\n",
    "    M.append(np.block([[A[k], B[k]], [-B[k].T, C[k]]]))\n",
    "    z.append( -M[k] @ x_optimal)\n",
    "    eigenvalues, _ = np.linalg.eig(M[k].T @ M[k])\n",
    "    L_M.append(np.sqrt(max(eigenvalues)))\n",
    "\n",
    "M_mean = np.mean(M, axis = 0)\n",
    "eigen, _ = np.linalg.eig(M_mean.T @ M_mean)\n",
    "L_total = np.sqrt(np.max(eigen))\n",
    "mu_total = np.sqrt(np.min(eigen))\n",
    "\n",
    "# Store the Data to use for Operator F\n",
    "dat = (M,z)\n",
    "\n",
    "# Uniform Sampling parameters\n",
    "p_uniform = np.ones(n)/n\n",
    "delta_uniform = 2*np.mean(np.array(L_M)**2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "784ae988",
   "metadata": {},
   "outputs": [],
   "source": [
    "def operator_F(x, data, i):\n",
    "    return data[0][i] @ x + data[1][i]\n",
    "\n",
    "def SPEG_constant(x_0, x_optimal, L, mu, delta, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    omega_hat = min(.25/L, mu/(18 * delta))\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "        op = operator(x_0, data, i)\n",
    "        x = x_0 - omega_hat * op\n",
    "        \n",
    "        for k in range(iterations):\n",
    "            x_mid = x - omega_hat * op\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_mid, data, i)\n",
    "            x = x - omega_hat * op\n",
    "            \n",
    "            relative_error[trial, k] = np.sum((x - x_optimal)**2)/ initial_error\n",
    "\n",
    "    return np.mean(relative_error, axis = 0)\n",
    "\n",
    "def SPEG_Hsieh(x_0, x_optimal, L, mu, gamma, b, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "        op = operator(x_0, data, i)\n",
    "        x = x_0 - (gamma * op)/b\n",
    "        \n",
    "        for k in range(iterations):\n",
    "            step = gamma/(k + 1 + b)\n",
    "            x_mid = x - step * op\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_mid, data, i)\n",
    "            x = x - step * op\n",
    "            \n",
    "            relative_error[trial, k] = np.sum((x - x_optimal)**2)/ initial_error\n",
    "    \n",
    "\n",
    "    return np.mean(relative_error, axis = 0)\n",
    "        \n",
    "\n",
    "def SPEG_decreasing(x_0, x_optimal, L, mu, delta, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    omega_hat = min(1/(4 * L), mu / (18 * delta))\n",
    "    k_0 = math.ceil(iterations / 2)\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        if iterations <= 2/(mu * omega_hat):\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_0, data, i)\n",
    "            x = x_0 - omega_hat * op\n",
    "            for k in range(iterations):\n",
    "                x_mid = x - omega_hat * op\n",
    "                i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "                op = operator(x_mid, data, i)\n",
    "                x = x - omega_hat * op\n",
    "                \n",
    "                relative_error[trial, k] = np.sum((x - x_optimal)**2)/initial_error\n",
    "        else:\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_0, data, i)\n",
    "            x = x_0 - omega_hat * op\n",
    "            for k in range(iterations):\n",
    "                if k <= k_0:\n",
    "                    step = omega_hat\n",
    "                else:\n",
    "                    step = (4 * omega_hat)/(4 + mu * omega_hat * (k - k_0))\n",
    "                \n",
    "                x_mid = x - step * op\n",
    "                i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "                op = operator(x_mid, data, i)\n",
    "                x = x - step * op\n",
    "                \n",
    "                relative_error[trial, k] = np.sum((x - x_optimal)**2)/initial_error\n",
    "\n",
    "    \n",
    "    return np.mean(relative_error, axis = 0)\n",
    "\n",
    "\n",
    "def SPEG_switch(x_0, x_optimal, L, mu, delta, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    omega_hat = min(1/(4*L), mu/(18 * delta))\n",
    "    k_0 = math.ceil(4/(mu * omega_hat))\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "        op = operator(x_0, data, i)\n",
    "        x = x_0 - omega_hat * op\n",
    "        \n",
    "        for k in range(iterations):\n",
    "            if k <= k_0:\n",
    "                step = omega_hat\n",
    "            else:\n",
    "                step = (4*k + 2)/(mu * (k + 1)**2)\n",
    "                \n",
    "            x_mid = x - step * op\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_mid, data, i)\n",
    "            x = x - step * op\n",
    "            \n",
    "            relative_error[trial, k] = np.sum((x - x_optimal)**2)/ initial_error\n",
    "            \n",
    "    return np.mean(relative_error, axis = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "05be6931",
   "metadata": {},
   "outputs": [],
   "source": [
    "SPEG_constant_error = SPEG_constant(np.zeros(2*d), x_optimal, L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 10000)\n",
    "SPEG_Hsieh_error = SPEG_Hsieh(np.zeros(2*d), x_optimal, L_total, mu_total, 1.45, 5, n, (M,z), operator_F, p_uniform, iterations = 10000)\n",
    "# SPEG_decreasing_error = SPEG_decreasing(np.zeros(2*d), x_optimal, L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 10000)\n",
    "# SPEG_switch_error = SPEG_switch(np.zeros(2*d), x_optimal, L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 10000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d823fdc8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZUAAAEOCAYAAABB+oq7AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAABI4UlEQVR4nO3deZzN9f7A8dd7hrFTWQapLNeSbYZBSLauyBIVRRsVrkrbjSK3Ut1SUrqVLi2oFJX6lYtW0apCqSgkWSZEJDP2mXn//vicGWeOc86cmTlzzizv5+Pxfcz5fs53eX/OcN7z/X4+389HVBVjjDEmHGKiHYAxxpjiw5KKMcaYsLGkYowxJmwsqRhjjAkbSyrGGGPCxpKKMcaYsLGkYowxJmwsqRhjjAmbUtEOIFxEpD4wAaiiqgNz2r5atWpat27dPJ/vwIEDVKhQIc/7FzUlrb5gdS4prM65s2rVqj9UtXrADVS10C7ATGAXsManvBewHtgIjPN5b34ox05KStL8WLp0ab72L2pKWn1Vrc4lhdU5d4CVGuS7tbDf/pqNSyBZRCQWmAacDzQFhohI08iHZowxxpdoIR/7S0TqAgtVtblnvQMwUVV7etbHA6jqJM/6fA1w+0tERgIjAeLj45PmzZuX57hSU1OpWLFinvcvakpafcHqXFJYnXOnW7duq1S1TaD3i2KbyqnANq/1ZOAsEakKPAC0EpHxmUnGm6o+AzwD0KZNG+3atWueg1i2bBn52b+oKWn1BatzSWF1Dq+imFTET5mq6h5gVKSDMflz7NgxkpOTOXz4cLRDOUGVKlX46aefoh1GRFmdS4ZQ6ly2bFnq1KlD6dKlc3XsophUkoHTvNbrANujFIvJp+TkZCpVqkTdunUR8ff3QvSkpKRQqVKlaIcRUVbnkiGnOqsqe/bsITk5mXr16uXq2IW9od6fFUBDEaknInHAYGBBlGMyeXT48GGqVq1a6BKKMSWZiFC1atU83UEo1ElFROYCy4HGIpIsIteqahowGngP+Al4TVXXRjNOkz+WUIwpfPL6/7JQ3/5S1SEByhcDiyMcDgAHDsCECdC9e6H+6IwxJioK9ZVKYfTjj/Df/8I99zTj6NFoR2PCYefOnQwePJgGDRrQtGlTevfuzYYNG8J2/Lfeeosff/wxz/tv3ryZV155JVf7PPjgg3k+XyALFizgoYceyvdxduzYQd++fXnvvfdITEwkMTGRihUr0rhxYxITE7nqqquYPXs2o0ePDkPUBS89PZ1WrVrRt29fv++vW7eODh06UKZMGaZMmZJVfvToUTp37kxaWlqkQo0ISyq51LbM9+yPq0qt1cu57joo5I/5FCutWsH118OOHeE7pqpy4YUX0rVrV3755Rd+/PFHHnzwQX7//fewnaO4JJULLriAcePG5fs4jz32GCNGjKBnz56sXr2a1atX06ZNG15++WVWr17Niy++GIZo/UtPTw/7Mf/zn/9w5plnBnz/lFNO4YknnmDMmDHZyuPi4jj33HN59dVXwx5TNFlSya0KFSiTupf+HX9i5kx45JFoB1RyrF4Nzz8P9euHL7ksXbqU0qVLM2rU8d7oiYmJnHPOOagqY8eOpXnz5rRo0SLrP39mH/+BAwfSpEkTLr/88swhghg3bhxNmzalZcuWjBkzhi+++IIFCxYwduxYEhMT+eWXX3j22Wdp27YtCQkJXHzxxRw8eBCAYcOGcdNNN9GxY0fq16/P/Pnzs4756aefkpiYyNSpU7PFv2PHDjp37kxiYiLNmzfn008/Zdy4cRw6dIjExEQuv/xyAObMmUO7du1ITEzkH//4R9aXa8WKFbntttto3bo15557Ln/88QcATzzxRFY9Bg8eDJDt6iHzCiMxMZFy5crx8ccfc+DAAa655hratm1Lq1atePvtt/1+5m+88Qa9evXy+5637du306tXLxo2bMjtt9+eVf7+++/ToUMHWrduzaBBg0hNTQVgyZIltGrVihYtWnDNNddw5MgRAOrWrct9991Hp06deP311wPuX7duXe688046dOhAmzZt+Oabb+jZsycNGjRg+vTpfmNMTk5m0aJFDB8+PGA9atSoQdu2bf12zR0wYAAvv/xyjp9FkRJsDJfivOR57K+9e1VBN1x3vV5yiaqI6htv5O1QRUlBjY/0448/Zr2++WbVLl0CL+660C0iqjExqrVqqbZvH3ifm28Ofv7//Oc/esstt/h976WXXtK///3vmpaWpjt37tTTTjtNt2/frkuXLtXKlSvrtm3bND09Xdu3b6+ffvqp7tmzRxs1aqQZGRmqqvrnn3+qqurQoUP19ddfzzruH3/8kfV6woQJ+sQTT2RtN3DgQE1PT9e1a9dqgwYNVNV99n369PEb45QpU/Tf//63qqqmpaXp/v37VVW1QoUK2T7jvn376tGjR1VV9brrrtMXXnhBVVUBnTNnjqqq3nvvvTpixAhVVa1Vq5YePnw4Wz1mzZqlN9xwQ7bzL1iwQDt16qRHjx7V8ePH60svvZS1T8OGDTU1NTXb9ps2bdLWrVufUI8uXbroihUrstZnzZql9erV03379umhQ4f09NNP161bt+ru3bv1nHPOyTruQw89pPfee68eOnRI69Spo+vXr1dV1SuvvFKnTp2qqqpnnHGGPvzww6qqfve/8847s7Z7+umnVVX1lltu0RYtWuj+/ft1165dWr16db+f/8UXX6wrV64M+jvKdM899+gjjzySrSwtLU2rVasWdL+CkPnvJCfe/z8zkcPYX9banFtVqkBMDHEp+5k9G7ZsgSuugE8+gTYBBy4w4ZaZXnbsgIMHITEx/OdYvnw5Q4YMITY2lvj4eLp06cKKFSuoXLky7dq1o06dOoD7q33z5s20b9+esmXLMnz4cPr06RPwHvuaNWv417/+xb59+0hNTaVnz55Z7w0YMICYmBiaNm0a0i24tm3bcs0113Ds2DEGDBhAop8PYsmSJaxatYq2bdsCcOjQIWrUqAFATEwMl156KQBXXHEFAwYMAKBly5ZcfvnlDBgwIKvM188//8zYsWP56KOPKF26NO+//z4LFizIajc4fPgwW7duzXZraMeOHVSvHniAW2/nnnsuVapUAaBp06Zs2bKFffv28eOPP3L22WcDrl2iQ4cOrF+/nnr16tGoUSMAhg4dyrRp07jlllsAsur45ZdfnrB/G6//uBdccAEALVq0IDU1lUqVKlGpUiXKli3Lvn37OOmkk7K2XbhwITVq1CApKYlly5aFVCdfsbGxxMXFFatnZSyp5FZMDJxyCqX276dcOXj7bTjrLOjXD776Ck4/PdoBFl2PPx78fe8ejnFxEBsLV18Nd90FNWvm7ZzNmjXLus3kS4M0mJUpUybrdWxsLGlpaZQqVYqvv/6aJUuWMG/ePJ566ik++uijE/YdNmwYb731FgkJCcyePTvbF5L3cYOdP1Pnzp355JNPWLRoEVdeeSVjx47lqquuOqEeQ4cOZdKkE0YuOkFmN9JFixbxySefsGDBAu6//37Wrs3ea//AgQNccsklPPvss9SuXTvrPG+88QaNGzcOePxy5cqF/OyDv89YVenRowdz587Ntu3q1auDHitzmHd/+6ekpJxwzpiYmGznj4mJOaFB/fPPP2fBggUsXryYw4cPs3//fq644grmzJkTUv0yHTlyhLJly+Zqn8LM2lTyonp14v78E4D4eFi0yP213Lcv7N8f5diKubg4KFcOhg+HTZtg2rS8JxSA7t27c+TIEZ599tmsshUrVvDxxx9z9tln8+qrr5Kens7u3bv55JNPaNeuXcBjpaam8tdff9G7d28ef/zxrC+6SpUqZfviSklJoVatWhw7diyk++m++3vbsmULNWrUYMSIEVx77bV88803AJQuXZpjx44B7i/++fPns2vXLgD27t3Lli1bAMjIyMhKqq+88grt27cnIyODbdu20a1bNyZPnpx1ReXt6quv5uqrr+acc87JKuvZsydPPvlkVjL89ttvT4i3UaNGbN68Occ6B9K+fXs+//xzNm7cCMDBgwfZsGEDTZo0YfPmzVnlL730El26dAlp/59//jlPsUyaNInk5GQ2b97MvHnz6N69e64Typ49e6hevXquh0IpzCyp5EXDhpTfdnxMy2bNYP5819340kuhmPUQLDQSE8OXTDKJCP/3f//HBx98QIMGDWjWrBkTJ06kdu3a9OvXj5YtW5KQkED37t2ZPHkyNYOcNCUlhb59+9KyZUu6dOmS1ag+ePBgHnnkEVq1asUvv/zC/fffz1lnnUWPHj1o0qRJjjG2bNmSUqVKkZCQcEJD/bJly0hMTKRVq1a88cYb3HzzzQCMHDky6xZW06ZN+fe//815551Hy5Yt6dGjBzs8vRwqVKjA2rVrSUpK4qOPPmLcuHGkp6dzxRVX0KJFC1q1asWtt96a7bbPli1bmD9/PjNnzsxqrF+5ciV33XUXx44do2XLljRv3py77rrrhLpUqFCBBg0aZH2p51b16tWZPXs2Q4YMoWXLlrRv355169ZRtmxZZs2axaBBg2jRogUxMTHZOl8E2z+vSSWY6dOnZzXu79y5kzp16vDYY4/x73//mzp16rDf89fn0qVL6d27d9jPH1XBGlyK85KvSbpuv13TS5dWTUvLVjxjhrvTf/31qp622mIjEg31hU2ojZlFmXeDvmpk6vzmm2/qhAkTCvw8oYrm7/nCCy/UdevWRfy81lBf2DRrRsyxY/DTT9C8eVbxyJHw888wZQo0agSePxqNMV4uvPBC9uzZE+0wou7o0aMMGDAgaBtUUWS3v/Ii816tn0bYhx6CAQPg1lvhf/+LbFjG5JZvW0mkBHuuo6SIi4s7oVNFcWBJJS/OOINDtWv7zRqxsTBnDrRuDUOGgJ+2SmOMKbYsqeTRzvPOgw8/dK3GPipUcPnmlFNcj7DffotCgMYYEwWWVPJoZ+/ern/r/ff7fb9WLVi40HUx7tcPonSXwRhjIsqSSh4dqV4dbroJXngBPv3U7zYtW8Krr8J338Fll0EBjGVnjDGFiiWV/Lj7bmjQAAYPhl9+8btJ797wxBPudpjPIKWmkIiNjSUxMZFmzZqRkJDAY489RkZGRrTDyrJy5UpuuummAjv+6tWrWbw4/NMTDR8+PF+jM2d6/PHHs0YuHjZs2AkjIFSsWDHo/r1792bfvn1Bt8npGMHk5fPbvHkzzb16jvpz8OBB+vTpQ5MmTWjWrFm2EaKPHDnCpZdeyt/+9jfOOuusbA+U9urVi5NOOumEYYI++ugjWrduTfPmzfnHP/6RNULAwoULueeee3IVfzCWVPKjUiV44w04fNj1CAswB8cNN7juxY8/Dk8/HdkQi5WaNd1YLb5LPp+CLFeuHKtXr2bt2rV88MEHLF68mHvvvTff4YZrnow2bdrwxBNPhOVY/hRUUnnuuedo2rRpvo6RlpbGzJkzueyyy/J8jMWLF2d7eDPcCurzAxgzZgzr1q3j22+/5fPPP+edd94B4Pnnn+fkk09m48aN3Hrrrdxxxx1Z+4wdO5aXXnop23EyMjIYOnQo8+bNY82aNZx22mm88MILAPTp04cFCxZkjZadX8UmqYjImSIyXUTmi8h1ETtxy5awdCkcOQJt27r7XX48+qhrtL/xRvD8uzC5FWiAxTDOfVKjRg2eeeYZnnrqKVSV9PR0xo4dS9u2bWnZsiUzZszI2nby5Mm0aNGChISErL8iu3btyp133kmXLl34z3/+w6pVq+jSpQtJSUn07Nkz60n2QMPfv/766zRv3pyEhAQ6d+4MuKfmM//qnDhxItdccw1du3alfv362ZLN/fffT5MmTejRowdDhgzJNiFUJt/jHz16lLvvvptXX32VxMRE3njjjYBD2M+ePZv+/fvTq1cvGjdunJV4Dxw4QJ8+fUhISKB58+ZZUwR07dqVlStXsmDBgqwn7xs3bky9evUAAn423jL/ui5VKudH6vxNAwBuSPvMIf0DTQEAMGHCBBISEmjfvr3fwTz9fS6+n5/v3CibN2/mnHPOoXXr1rRu3Zovvvgix3pkKl++PN26dQNc9+PWrVuTnJwMwNtvv83QoUMBGDhwIEuWLMkaHufcc889YXDKPXv2UKZMmawBN7t3784bb7wBuFElunbtysKFC0OOLahgT0ZGagFmAruANT7lvYD1wEZgXIjHigGez2m7fD1Rr36eMP/1VzcGO6j276/6888n7JOSopqYqFqxoup33+Xr9BEXkSfqczP2ve+S17Hv9cSnylVVTzrpJN24caPOmDFD77//flVVPXz4sCYlJemmTZt08eLF2qFDBz1w4ICqqu7Zs0dV3RDu1113naqqHj16VDt06KC7du1SVdV58+bp1VdfraqBh79v3ry5Jicnq+rxIee9h1W/5557tEOHDnr48GHdvXu3nnLKKXr06FFdsWKFJiQk6MGDB3X//v36t7/97YRh1gMd33tI+/379wccwn7WrFlas2ZN/eOPP/TgwYParFkzXbFihc6fP1+HDx+edY59+/ZlfRbew9mrqg4aNEifeuqpoJ+Nt7vvvjvrs1F10wPUrVtXExISspbM31+gaQDOOOMM3b17d8ApAPbv36+ALliwQFVVx44dm/U79xbsc/GdEiDTgQMH9NChQ6qqumHDBs383vn111+1WbNmfvfx588//9R69erpL7/8oqqqzZo1023btmW9X79+fd29e3fWuu9Q/BkZGXr66adn/T5GjRqlzZs3z3p/zpw5Onr06BPOW5SfqJ8NPAVkTfkmIrHANKAHkAysEJEFQCzgO9zqNaq6S0QuAMZ5jhVZdeu68e8ffRT+/W9o2tSNiT9mjHsNVKzo2lbOOstdtXz1leslZgof9fzV9/777/P9999n3cf/66+/+Pnnn/nwww+5+uqrKV++POBm98uUOcz6+vXrWbNmDT169ADcrIO1PL/wQMPfn3322QwbNoxLLrmEiy66yG9sffr0oUyZMpQpU4YaNWrw+++/89lnn9G/f3/KlSsHQL9+/fzuG8rxAw1hD9CjRw+qVq0KwEUXXcRnn31G7969GTNmDHfccQd9+/bNNsikt8mTJ1OuXDluuOEG1qxZE/Cz8bZjx44TZlV85JFHGDhwYNZ6ZntITtMABJsCIC4uLutqMCkpiQ8++CBXn0sgx44dY/To0axevZrY2Ng8TVOdlpbGkCFDuOmmm6hfvz7gfwRr8R7G28978+bN49Zbb+XIkSN06dIl29VfjRo12L59e65j86dQJBVV/URE6voUtwM2quomABGZB/RX1UmA34kqVHUBsEBEFgG5m381HEqXhnHjYOhQePBBN03hrFnQubMrGziQOnUqs3AhdOoEF1wAy5a551oMuRv73lce57PwZ9OmTcTGxlK9enVUlSeffDLbnCcA7777bsD/xN7DrDdr1ozly5efsE2g4e+nT5/OV199xaJFi0hMTPQ7pHugIeFDEcrxVf0PYf/VV1+dUGcRoVGjRqxatYrFixczfvx4zjvvPO6+++5s2y1ZsoTXX3+dTz75JOscgT4bb7kZKj+naQBU/U8BkJKSQunSpbPqlvmZ+gr2uQQydepU4uPj+e6778jIyMjTEPcjR46kYcOGWXPDANSpU4dt27ZRp04d0tLS+Ouvv7L9YeNPhw4dsm4JvvXWW1kjVYNLkJl/kORXoUgqAZwKbPNaTwbOCrSxiHQFLgLKAH5bzURkJDASID4+Ps8T64Ab3iLo/hdfTOm//51aCxdS8733KH/ttWSMGsW+xEQqdOzIpOv7cMujf+f88/9g4sS1xBTy1q0c65tHVapUCTisu69gUxiFeoyc9v/jjz8YPnw4I0aMICMjgy5duvDkk09mTQf7888/U7t2bTp16sTDDz9Mv379KF++PHv37uWUU04hPT2dAwcOkJKSQu3atfn999/58MMPOeusszh27BgbN27kzDPPZP/+/VSqVIm9e/fy4osvUqtWLVJSUti0aRNNmzaladOmvP3226xbt46DBw+SlpZGSkoKR44coXTp0lnxZmRkkJqaSqtWrbjlllsYPXo0aWlp/O9//2PYsGEnfC7+jl+qVCn27t1LSkoK6enpdOvWjUcffZQpU6YgInz33XckJCRw+PBh3n//fbZs2UK5cuV48803mTZtGhs2bODkk0+mf//+xMbG8vLLL2cd68CBA6xdu5ZRo0bx5ptvZtUj2GfjrV69eqxduzarHseOHePQoUMn1CslJYWtW7dSu3ZtBg8ezJ49e/jyyy+58MILUVVSU1Np3749gwcPZsSIEVSvXp29e/eSmprKqaeemu3fwKFDhzh27NgJ5wj0uXh/fr52797NqaeeyoEDB5gzZw7p6emkpKSQmppKRkZGjv9u77vvPvbs2cPUqVOzbXve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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig = plt.figure()\n",
    "markers_on = [1000, 2000, 3000, 4000, 5000, 6000, 7000, 8000, 9000]\n",
    "plt.plot(np.arange(len(SPEG_constant_error)), SPEG_constant_error, '-b>', markevery = markers_on, label= 'Constant stepsize (Theorem 4.1)')\n",
    "plt.plot(np.arange(len(SPEG_Hsieh_error)), SPEG_Hsieh_error, '-rs', markevery = markers_on, label= 'Decreasing stepsize (Hsieh et al. 2019)')\n",
    "# plt.plot(np.arange(len(SPEG_decreasing_error)), SPEG_decreasing_error, label= 'SPEG decreasing')\n",
    "# plt.plot(np.arange(len(SPEG_switch_error)), SPEG_switch_error, label= 'SPEG switch')\n",
    "\n",
    "plt.yscale('log')\n",
    "plt.grid(True)\n",
    "plt.ylabel(\"Relative Error\", fontsize = 15)\n",
    "plt.xlabel(\"Number of Iterations\", fontsize = 15)\n",
    "plt.legend(fontsize = 10)\n",
    "#plt.title(\"Comparison on Interpolated Data\")\n",
    "plt.savefig('Constant vs Hsieh on Interpolated Data.pdf', format = 'pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "id": "27d71405",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(1.4692377997527208, 0.3159037827355898)"
      ]
     },
     "execution_count": 60,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "1/mu_total, 1/(4 * L_total)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "1afefc2e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(1.4692377997527208, 1.5795189136779488)"
      ]
     },
     "execution_count": 61,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "b = 5\n",
    "1/mu_total, b/(4 * L_total)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "d9577c6e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "539"
      ]
     },
     "execution_count": 62,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "omega_hat = min(1/(4*L_total), mu_total/(18 * delta_uniform))\n",
    "math.ceil(4/(mu_total * omega_hat))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "id": "055e7828",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "3.4657194364734996"
      ]
     },
     "execution_count": 63,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "delta_uniform"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
