{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "id": "wLflAAnJ_wf5",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:28:27.286065Z",
     "start_time": "2023-10-27T21:28:24.014185Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: tikzplotlib in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (0.10.1)\r\n",
      "Requirement already satisfied: numpy in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from tikzplotlib) (1.23.5)\r\n",
      "Requirement already satisfied: matplotlib>=1.4.0 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from tikzplotlib) (3.7.0)\r\n",
      "Requirement already satisfied: Pillow in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from tikzplotlib) (9.4.0)\r\n",
      "Requirement already satisfied: webcolors in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from tikzplotlib) (1.13)\r\n",
      "Requirement already satisfied: contourpy>=1.0.1 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (1.0.5)\r\n",
      "Requirement already satisfied: fonttools>=4.22.0 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (4.25.0)\r\n",
      "Requirement already satisfied: cycler>=0.10 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (0.11.0)\r\n",
      "Requirement already satisfied: kiwisolver>=1.0.1 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (1.4.4)\r\n",
      "Requirement already satisfied: python-dateutil>=2.7 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (2.8.2)\r\n",
      "Requirement already satisfied: packaging>=20.0 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (22.0)\r\n",
      "Requirement already satisfied: pyparsing>=2.3.1 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from matplotlib>=1.4.0->tikzplotlib) (3.0.9)\r\n",
      "Requirement already satisfied: six>=1.5 in /Users/cyrille.kone/anaconda3/lib/python3.10/site-packages (from python-dateutil>=2.7->matplotlib>=1.4.0->tikzplotlib) (1.16.0)\r\n"
     ]
    }
   ],
   "source": [
    "# @title Install \n",
    "!pip install tikzplotlib \n",
    "import abc \n",
    "import numpy as np \n",
    "import pandas as pd\n",
    "import joblib as jlb\n",
    "import seaborn as sns\n",
    "from operator import add \n",
    "import tikzplotlib as tkz\n",
    "import tqdm.notebook as tqdm \n",
    "import matplotlib.pyplot as plt \n",
    "from joblib import Parallel, delayed\n",
    "from functools import reduce, partial\n",
    "from IPython.display import clear_output\n",
    "%matplotlib inline "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "id": "WYMNV4gLAoOS",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:14.856315Z",
     "start_time": "2023-10-27T21:31:14.844340Z"
    }
   },
   "outputs": [],
   "source": [
    "#@title  Set up \n",
    "class BernoulliBandit(object):\n",
    "    r\"\"\"Implement a Bernoulli bandit\"\"\"\n",
    "\n",
    "    def __init__(self, K=2, arms_means=None, D=1) -> None:\n",
    "        r\"\"\"\n",
    "        @constructor\n",
    "        Parameters\n",
    "        ----------\n",
    "        K: int > 0\n",
    "           Number of arms of the bandit\n",
    "        arms_means: np.ndarray of shape [K, d]\n",
    "           Mean reward of each arm\n",
    "        D: int>0\n",
    "           Dimension of the reward vector\n",
    "                \"\"\"\n",
    "        self.D = D if arms_means is None else [*np.shape(arms_means), 1][1]\n",
    "        self.arms_means = np.random.uniform(size=(K, self.D)) if arms_means is None else np.asarray(arms_means)\n",
    "        self.arms_means = self.arms_means.reshape(-1, self.D).squeeze(-1) if self.D == 1 else self.arms_means\n",
    "        super(BernoulliBandit, self).__init__()\n",
    "\n",
    "    def sample(self, arms):\n",
    "        r\"\"\"\n",
    "         Sample from a Bernoulli bandit\n",
    "         Parameters\n",
    "         -----------\n",
    "         arms : set  of arms to sample\n",
    "         Returns\n",
    "         ------\n",
    "         Samples from arms\n",
    "         Test\n",
    "         ----\n",
    "         >>> bernoulli_bandit = BernoulliBandit(K=10)\n",
    "         >>> bernoulli_bandit.sample([1,2,4])\n",
    "         \"\"\"\n",
    "        arms = [arms] if isinstance(arms, int) else arms\n",
    "        return np.random.binomial(1, self.arms_means[arms]).reshape(-1, self.D)\n",
    "    \n",
    "def M(X_i,X_j):\n",
    "  return np.max(X_i - X_j, -1)\n",
    "def m(X_i,X_j):\n",
    "  return np.min(X_j - X_i, -1)\n",
    "def beta_ij(T_i, T_j):\n",
    "  return 0.5*np.sqrt(\n",
    "      2*(\n",
    "          2*(np.log(K*(K-1)*D/(2*delta))/2 + np.log(np.log(K*(K-1)*D/(2*delta))/2)) + 2*np.log(4+ np.log(T_i)) + 2*np.log(4+ np.log(T_j)) \n",
    "      )*(\n",
    "          (1/T_i) + (1/T_j)\n",
    "      )\n",
    "  )\n",
    "\n",
    "def beta(T_i, delta_=None):\n",
    "  delta_ = delta/K/D if delta_ is None else delta_\n",
    "  return 0.5*np.sqrt((2*np.log(1/delta_) + max(6*np.log(np.log(1/delta_)), 0) + 3*np.log(np.log(np.e*T_i))) /  T_i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "3tcQjYzIA6UP",
    "outputId": "03f1fc20-dab4-4154-c584-a892675af9a5",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:15.770451Z",
     "start_time": "2023-10-27T21:31:15.765067Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "9\n",
      "1994.3852446576025 534.7305332945461\n"
     ]
    },
    {
     "data": {
      "text/plain": "array([0.2607893 , 0.53337664, 0.60640098, 0.42862911, 0.270513  ,\n       0.2274008 , 0.01550271, 0.87559034, 0.59689447, 0.91963784])"
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#@title Bandit instance\n",
    "K = 10\n",
    "D = 1 \n",
    "delta = 0.01\n",
    "inf = (1<<31)*1.\n",
    "arms_means = np.random.uniform(0, 1, size=K).reshape(-1)\n",
    "best_arm = np.argmax(arms_means)\n",
    "pareto_arms = set([best_arm])\n",
    "Deltas = np.max(arms_means) - arms_means\n",
    "Deltas[Deltas==0] = np.inf \n",
    "Deltas[Deltas==np.inf]= np.min(Deltas)\n",
    "H = sum(1/Deltas**2)*0.5 #subgaussian proxy\n",
    "bandit = BernoulliBandit(arms_means=arms_means)\n",
    "print(best_arm)\n",
    "print(H*np.log(1/(2.4*delta)), H)\n",
    "arms_means"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {
    "id": "AoeliK7jCPz1",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:16.819347Z",
     "start_time": "2023-10-27T21:31:16.815327Z"
    }
   },
   "outputs": [],
   "source": [
    "# @title ape\n",
    "def ape(seed):\n",
    "  inf = (1<<31)*1.\n",
    "  np.random.seed(seed)\n",
    "  arms = np.arange(K)\n",
    "  total = np.zeros((K, D))\n",
    "  #initialize bandit \n",
    "  total[arms] += bandit.sample(arms)\n",
    "  Nc = np.ones(K, dtype=int)\n",
    "  mus = total/Nc[:, None]\n",
    "  xt = mus.reshape(-1)\n",
    "  I = np.eye(K)\n",
    "  z1_t = lambda : min([min(xt[i]-xt - beta_ij(Nc[i], Nc) + inf*I[i])  for i in a_star_t])\n",
    "  a_star_t = [np.argmax(xt)]\n",
    "  while z1_t() < 0:\n",
    "    bt = np.argmax([min(- xt  + beta_ij(Nc[i], Nc) + inf*I[i]) + xt[i]  for i in arms])\n",
    "    ct = np.argmax(xt + beta_ij(Nc[bt], Nc) - inf*I[bt])\n",
    "    idx = [bt, ct]\n",
    "    idx = [idx[np.argmin(Nc[idx])]]\n",
    "    for a in idx: \n",
    "      total[a] += bandit.sample([a]).reshape(-1)\n",
    "      Nc[a] += 1  \n",
    "    mus = total/Nc[:, None]\n",
    "    xt = mus.reshape(-1)\n",
    "    a_star_t = [np.argmax(xt)]\n",
    "  return set(pareto_arms)==(set(a_star_t)), Nc.sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {
    "cellView": "form",
    "id": "PYBq8OsnLYgK",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:17.341341Z",
     "start_time": "2023-10-27T21:31:17.337140Z"
    }
   },
   "outputs": [],
   "source": [
    "# @title lucb\n",
    "def lucb(seed):\n",
    "  np.random.seed(seed)\n",
    "  Nc = np.zeros(K, dtype=int)\n",
    "  arms = np.arange(K)\n",
    "  total = np.zeros((K, D))\n",
    "  #initialize bandit \n",
    "  total[arms] += bandit.sample(arms)\n",
    "  Nc = np.ones(K, dtype=int)\n",
    "  mus = total/Nc[:, None]\n",
    "  I = np.eye(K)\n",
    "  x_t = mus.reshape(-1)\n",
    "  ht = np.argmax(x_t)\n",
    "  lt = np.argmax(x_t + beta(Nc) - inf*I[ht])\n",
    "  stop = ((x_t[lt] + beta(Nc[lt])) - (x_t[ht] - beta(Nc[ht])))< 0\n",
    "  while not stop:\n",
    "      ht = np.argmax(x_t)\n",
    "      lt = np.argmax(x_t + beta(Nc) - inf*I[ht])\n",
    "      for i in [lt, ht]:\n",
    "        total[i] += bandit.sample([i]).reshape(-1)\n",
    "        Nc[i] += 1\n",
    "      mus = total / Nc[:,None]\n",
    "      x_t = mus.reshape(-1) \n",
    "      ht = np.argmax(x_t)\n",
    "      lt = np.argmax(x_t + beta(Nc) - inf*I[ht])\n",
    "      stop = ((x_t[lt] + beta(Nc[lt])) - (x_t[ht] - beta(Nc[ht])))< 0\n",
    "  return set([ht]) == set(pareto_arms), Nc.sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {
    "cellView": "form",
    "id": "r05rKIHHQd05",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:17.821511Z",
     "start_time": "2023-10-27T21:31:17.812631Z"
    }
   },
   "outputs": [],
   "source": [
    "# @title lucb++\n",
    "def lucbpp(seed):\n",
    "  np.random.seed(seed)\n",
    "  Nc = np.zeros(K, dtype=int)\n",
    "  arms = np.arange(K)\n",
    "  total = np.zeros((K, D))\n",
    "  #initialize bandit \n",
    "  total[arms] += bandit.sample(arms)\n",
    "  Nc = np.ones(K, dtype=int)\n",
    "  mus = total/Nc[:, None]\n",
    "  x_t = mus.reshape(-1)\n",
    "  I = np.eye(K)\n",
    "  ht = np.argmax(x_t)\n",
    "  lt = np.argmax(x_t + beta(Nc) - inf*I[ht])\n",
    "  stop = ((x_t[lt] + beta(Nc[lt], delta/2)) - (x_t[ht] - beta(Nc[ht], delta/(2*(K-1)))))< 0\n",
    "  while not stop:\n",
    "      ht = np.argmax(x_t)\n",
    "      lt = np.argmax(x_t + beta(Nc, delta/2) - inf*I[ht])\n",
    "      for i in [lt, ht]:\n",
    "        total[i] += bandit.sample([i]).reshape(-1)\n",
    "        Nc[i] += 1\n",
    "      mus = total / Nc[:,None]\n",
    "      x_t = mus.reshape(-1) \n",
    "      ht = np.argmax(x_t)\n",
    "      lt = np.argmax(x_t + beta(Nc, delta/2) - inf*I[ht])\n",
    "      stop = ((x_t[lt] + beta(Nc[lt], delta/2)) - (x_t[ht] - beta(Nc[ht], delta/(2*(K-1)))))< 0\n",
    "  return set([ht]) == set(pareto_arms), Nc.sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {
    "cellView": "form",
    "id": "OosMcE59Nc9q",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:18.320470Z",
     "start_time": "2023-10-27T21:31:18.314193Z"
    }
   },
   "outputs": [],
   "source": [
    "# @title ugape\n",
    "def ugape(seed):\n",
    "  np.random.seed(seed)\n",
    "  Nc = np.zeros(K, dtype=int)\n",
    "  arms = np.arange(K)\n",
    "  total = np.zeros((K, D))\n",
    "  #initialize bandit \n",
    "  total[arms] += bandit.sample(arms)\n",
    "  Nc = np.ones(K, dtype=int)\n",
    "  mus = total/Nc[:, None]\n",
    "  x_t = mus.reshape(-1)\n",
    "  I = np.eye(K)\n",
    "  ht = np.argmin([np.max(x_t + beta(Nc) -inf*I[i]) - (x_t[i] - beta(Nc[i])) for i in arms])\n",
    "  lt = np.argmax(x_t + beta(Nc) -inf*I[ht])\n",
    "  stop = ((x_t[lt] + beta(Nc[lt])) - (x_t[ht] - beta(Nc[ht])))< 0\n",
    "  while not stop:\n",
    "      ht = np.argmin([max(x_t + beta(Nc) -inf*I[i]) - (x_t[i] - beta(Nc[i])) for i in arms])\n",
    "      lt = np.argmax(x_t + beta(Nc) -inf*I[ht])\n",
    "      idx = [ht, lt]\n",
    "      idx = [idx[np.argmax(beta(Nc[idx]))]]\n",
    "      for i in idx:\n",
    "        total[i] += bandit.sample([i]).reshape(-1)\n",
    "        Nc[i]+= 1\n",
    "      mus = total / Nc[:,None]\n",
    "      x_t = mus.reshape(-1) \n",
    "      ht = np.argmin([np.max(x_t + beta(Nc) -inf*I[i]) - (x_t[i] - beta(Nc[i])) for i in arms])\n",
    "      lt = np.argmax(x_t + beta(Nc) -inf*I[ht])\n",
    "      stop = ((x_t[lt] + beta(Nc[lt])) - (x_t[ht] - beta(Nc[ht])))< 0\n",
    "  return set([ht]) == set(pareto_arms), Nc.sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {
    "id": "9UepD1HyESvz",
    "ExecuteTime": {
     "end_time": "2023-10-27T21:31:19.369775Z",
     "start_time": "2023-10-27T21:31:19.361655Z"
    }
   },
   "outputs": [],
   "source": [
    "numiter = 1000\n",
    "seeds = (np.random.uniform(size=numiter)*2**31).astype(int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "xu5--2UpKUJU",
    "outputId": "bc4d8694-e7b0-402c-8624-a1e94b3164ab"
   },
   "outputs": [],
   "source": [
    "res_lucbpp = Parallel(n_jobs=-1, verbose=1)(delayed(lucbpp)(i) for i in seeds)\n",
    "res_lucbpp = np.array(res_lucbpp).T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "4j0l6oPrW-eD",
    "outputId": "51cac569-c947-446c-ea45-f74369db911c"
   },
   "outputs": [],
   "source": [
    "print(np.mean(res_lucbpp[1]), np.mean(res_lucbpp[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 438,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "GhUz-aZnRop4",
    "outputId": "6ba517bd-ef1c-4029-e9b2-234dedbc6764"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[Parallel(n_jobs=-1)]: Using backend LokyBackend with 8 concurrent workers.\n",
      "[Parallel(n_jobs=-1)]: Done  34 tasks      | elapsed:    1.6s\n",
      "[Parallel(n_jobs=-1)]: Done 184 tasks      | elapsed:    7.9s\n",
      "[Parallel(n_jobs=-1)]: Done 434 tasks      | elapsed:   18.7s\n",
      "[Parallel(n_jobs=-1)]: Done 784 tasks      | elapsed:   33.8s\n",
      "[Parallel(n_jobs=-1)]: Done 1000 out of 1000 | elapsed:   43.0s finished\n"
     ]
    }
   ],
   "source": [
    "res_lucb = Parallel(n_jobs=-1, verbose=1)(delayed(lucb)(i) for i in seeds)\n",
    "res_lucb = np.array(res_lucb).T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 439,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "MCIgyjLuRqYl",
    "outputId": "d1b27bac-23e4-4d79-8de8-8054efa179e8"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "11908.046 1.0\n"
     ]
    }
   ],
   "source": [
    "print(np.mean(res_lucb[1]), np.mean(res_lucb[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 440,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "4YN9DxC2MpkE",
    "outputId": "ac41f42b-6b86-4bcd-812d-803279f61db0"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[Parallel(n_jobs=-1)]: Using backend LokyBackend with 8 concurrent workers.\n",
      "[Parallel(n_jobs=-1)]: Done  34 tasks      | elapsed:    7.4s\n",
      "[Parallel(n_jobs=-1)]: Done 184 tasks      | elapsed:   37.3s\n",
      "[Parallel(n_jobs=-1)]: Done 434 tasks      | elapsed:  1.5min\n",
      "[Parallel(n_jobs=-1)]: Done 784 tasks      | elapsed:  2.6min\n",
      "[Parallel(n_jobs=-1)]: Done 1000 out of 1000 | elapsed:  3.3min finished\n"
     ]
    }
   ],
   "source": [
    "res_ape = Parallel(n_jobs=-1, verbose=1)(delayed(ape)(i) for i in seeds)\n",
    "res_ape = np.array(res_ape).T "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 441,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "46yJaFFFtqsq",
    "outputId": "7b7a4fca-2f12-42f5-f8d0-d391f2c51c8b"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "8138.381 1.0\n"
     ]
    }
   ],
   "source": [
    "print(np.mean(res_ape[1]), np.mean(res_ape[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 442,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "UaZmQSPPNivU",
    "outputId": "d21fb155-0ead-4d41-e8f3-7ee56d08db4c"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[Parallel(n_jobs=-1)]: Using backend LokyBackend with 8 concurrent workers.\n",
      "[Parallel(n_jobs=-1)]: Done  34 tasks      | elapsed:   19.5s\n",
      "[Parallel(n_jobs=-1)]: Done 184 tasks      | elapsed:  1.7min\n",
      "[Parallel(n_jobs=-1)]: Done 434 tasks      | elapsed:  3.9min\n",
      "[Parallel(n_jobs=-1)]: Done 784 tasks      | elapsed:  7.0min\n",
      "[Parallel(n_jobs=-1)]: Done 1000 out of 1000 | elapsed:  8.9min finished\n"
     ]
    }
   ],
   "source": [
    "res_ugape = Parallel(n_jobs=-1, verbose=1)(delayed(ugape)(i) for i in seeds)\n",
    "res_ugape = np.array(res_ugape).T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 443,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "euCkevy3Wqio",
    "outputId": "d6e4bae4-6240-43a6-a0b0-dd005d8fb8f1"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "11789.861 1.0\n"
     ]
    }
   ],
   "source": [
    "print(np.mean(res_ugape[1]), np.mean(res_ugape[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 445,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 441
    },
    "id": "3ptpyEq1vmtB",
    "outputId": "078702d8-6359-44ef-bbe5-2b1a3f1629ac"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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xw7v1+jRoyVJ69at06pVq7Ry5Url5OTo8ssv19y5c+mnB1qxnj17Ki0tzRtuDMNQWloal6UDrZClgScvL09bt27Vz3/+c8XExKhXr16aNWuWsrKy6m175MgRffPNNxo0aJDfY7355pu66667NGDAAEVHR2v+/PkqLCxUTk5Oc58GgBA2depUde7cWZIUFxentLQ0iysCYIUIK598//796tixo7p16+Zd1r9/fxUWFurkyZM+E//l5uYqNjZW6enpys3NVVxcnH784x9r8uTJkqQDBw5o5syZ3u0jIyOVkJCgPXv26Kqrrgq4prMHOAIIb5GRkUpPT9fzzz+vuXPnKjIykt9zwEYC/X22NPCcPn1aMTExPsvqHpeVlfkEnqqqKiUnJys9PV0DBgxQTk6O5syZo9jYWI0bN87vsVwul8rKyppUU25u7nmeDYBQ1aZNGy1YsECStHPnTmuLAWAJSwNPmzZtVF5e7rOs7nFsbKzP8okTJ2rixInex6NGjdLEiRP1l7/8RePGjVNMTIwqKip89qmoqKh3nHNJSkqqd1UHAAAITW63O6DGCksDz4ABA3TixAkdP35ccXFxkqSDBw+qe/fuateunc+2a9as8bbm1KmqqlJ0dLT3WPv379d1110nSaqurtbhw4c1cODAJtXkdDoJPAAA2Iylg5YTEhKUmpqqp59+WqWlpTpy5IhefPFF77icM5WWluqJJ57Q7t275fF4tHnzZm3YsEFTpkyRJE2aNEmrV6/Wnj17VFlZqYyMDMXFxWno0KEtfVoAACDEWNrCI0nPP/+8Fi5cqBtuuEEOh0MTJ07UrFmzJEkpKSl6/PHHNX78eN19990qKyvT/fffr+LiYvXq1UuLFy/2BprJkyfr1KlTmj17tkpKSpSUlKSXXnpJkZGRVp4eAAAIAYbJjWok1fYB7ty5U8nJyXRpAQAQJgL9/Lb8xoMAAADNjcADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8ADAABsj8AD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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.set_style('whitegrid')\n",
    "data = np.concatenate((res_lucbpp[1], res_lucb[1], res_ape[1], res_ugape[1])) / (H*np.log(1/(2.4*delta)))\n",
    "names = reduce(add, [[txt,]*len(seeds) for txt in [r\"LUCB++\", r\"LUCB\", r\"APE\", r\"UGap\"]])\n",
    "df = pd.DataFrame({\"sc\": data, \"names\": names})\n",
    "fig = sns.boxplot(data=df, x=\"names\", y=\"sc\",  orient=\"v\", showmeans=True, showfliers=True, notch=True)\n",
    "fig.set_xlabel('',fontweight='bold')\n",
    "fig.set_ylabel(r'$\\tau_{\\delta}$')\n",
    "fig.text(6, 1, \"lbd\")\n",
    "plt.hlines(1, -2, 6)\n",
    "plt.ticklabel_format(axis=\"y\", style=\"sci\", scilimits=(0,0))\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "provenance": []
  },
  "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.10.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
