{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from scipy.optimize import brentq\n",
    "from copy import deepcopy"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# B-UCB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BUCB(object):\n",
    "    def __init__(self, ub, lb, avg):\n",
    "        \n",
    "        self.means = avg\n",
    "        self.num_arms = avg.size\n",
    "        self.var =  0.25\n",
    "        self.ub = ub\n",
    "        self.lb = lb\n",
    "        \n",
    "        self.best_arm = np.argmax(self.means)\n",
    "\n",
    "        self.restart()\n",
    "        return None\n",
    "    \n",
    "    def restart(self):\n",
    "        self.time = 0.0\n",
    "        self.emp_means = np.zeros(self.num_arms)\n",
    "        self.num_plays = np.zeros(self.num_arms)\n",
    "        self.ucb_arr = 1e5*np.ones(self.num_arms)\n",
    "        self.cum_reg = [0]\n",
    "        return None\n",
    "    \n",
    "    def get_best_arm(self):\n",
    "        if np.argmax(self.ucb_arr).size == 1:\n",
    "            return np.argmax(self.ucb_arr)\n",
    "        else:\n",
    "            return np.argmax(self.ucb_arr)[0]\n",
    "    \n",
    "    def update_stats(self, arm, rew):\n",
    "        self.time += 1.0\n",
    "        self.num_plays[arm] += 1.0\n",
    "        self.emp_means[arm] = (self.emp_means[arm]*(self.num_plays[arm]-1.0) + rew)/self.num_plays[arm]\n",
    "        return None\n",
    "    \n",
    "    def update_ucb(self):\n",
    "        func = 2*self.var*np.log(1 + self.time*(np.log(self.time)**2))\n",
    "        for i in range(self.num_arms):\n",
    "            if self.num_plays[i] == 0:\n",
    "                continue\n",
    "            else:\n",
    "                temp = self.emp_means[i] + np.sqrt(func/self.num_plays[i])\n",
    "                self.ucb_arr[i] = max(self.lb[i], min(temp, self.ub[i]))\n",
    "        return None\n",
    "    \n",
    "    def update_reg(self, arm, rew_vec):\n",
    "        self.cum_reg += [self.cum_reg[-1] + rew_vec[self.best_arm] - rew_vec[arm]]\n",
    "        return None\n",
    "    \n",
    "    def iterate(self, rew_vec):\n",
    "        play = self.get_best_arm()\n",
    "        self.update_stats(play, rew_vec[play])\n",
    "        self.update_ucb()\n",
    "        self.update_reg(play, rew_vec)\n",
    "        return None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Improved UCB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "class UCBImproved(object):\n",
    "    def __init__(self, ub, lb, avg):\n",
    "        \n",
    "        self.means = avg\n",
    "        self.num_arms = avg.size\n",
    "        self.ub = ub\n",
    "        self.lb = lb\n",
    "        \n",
    "        self.sg = 0.25*np.ones(self.num_arms)\n",
    "        for i in range(self.num_arms):\n",
    "            if self.lb[i]>0.5:\n",
    "                self.sg[i] = self.get_sg(self.lb[i])\n",
    "            elif self.ub[i]<0.5:\n",
    "                self.sg[i] = self.get_sg(self.ub[i])\n",
    "        \n",
    "        self.best_arm = np.argmax(self.means)\n",
    "        \n",
    "        self.restart()\n",
    "        return None\n",
    "    \n",
    "    \n",
    "    def func(self, x, m):\n",
    "        fx = m*np.exp(x*(1-m)) + (1-m)*np.exp(-x*m)\n",
    "        gx = m*(1-m)*(np.exp((1-m)*x) - np.exp(-m*x))/fx - (2/x)*np.log(fx)\n",
    "        return gx\n",
    "    \n",
    "    def get_sg(self, m):\n",
    "        if m > 0.5: \n",
    "            m = 1-m\n",
    "        x = brentq(self.func, 0.0005, 100, args = m)\n",
    "        sg = (2/x**2)*np.log(m*np.exp((1-m)*x) + (1-m)*np.exp(-m*x))\n",
    "        return sg\n",
    "    \n",
    "    def restart(self):\n",
    "        self.time = 0.0\n",
    "        self.emp_means = np.zeros(self.num_arms)\n",
    "        self.num_plays = np.zeros(self.num_arms)\n",
    "        self.ucb_arr = 1e5*np.ones(self.num_arms)\n",
    "        self.cum_reg = [0]\n",
    "        return None\n",
    "    \n",
    "    def get_best_arm(self):\n",
    "        if np.argmax(self.ucb_arr).size == 1:\n",
    "            return np.argmax(self.ucb_arr)\n",
    "        else:\n",
    "            return np.argmax(self.ucb_arr)[0]\n",
    "    \n",
    "    def update_stats(self, arm, rew):\n",
    "        self.time += 1.0\n",
    "        self.num_plays[arm] += 1.0\n",
    "        self.emp_means[arm] = (self.emp_means[arm]*(self.num_plays[arm]-1.0) + rew)/self.num_plays[arm]\n",
    "        return None\n",
    "    \n",
    "    def update_ucb(self):\n",
    "        for i in range(self.num_arms):\n",
    "            if self.num_plays[i] == 0:\n",
    "                self.ucb_arr[i] = self.ub[i]\n",
    "            else:\n",
    "                func = 2*self.sg[i]*np.log(1 + self.time*(np.log(self.time)**2))\n",
    "                temp = self.emp_means[i] + np.sqrt(func/self.num_plays[i])\n",
    "                self.ucb_arr[i] = max(self.lb[i],min(temp, self.ub[i]))\n",
    "        return None\n",
    "    \n",
    "    def update_reg(self, arm, rew_vec):\n",
    "        self.cum_reg += [self.cum_reg[-1] + rew_vec[self.best_arm] - rew_vec[arm]]\n",
    "        return None\n",
    "    \n",
    "    def iterate(self, rew_vec):\n",
    "        play = self.get_best_arm()\n",
    "        self.update_stats(play, rew_vec[play])\n",
    "        self.update_ucb()\n",
    "        self.update_reg(play, rew_vec)\n",
    "        return None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## B-kl-UCB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BKL(object):\n",
    "    \n",
    "    def __init__(self, ub, lb, avg):\n",
    "        \n",
    "        self.means = avg\n",
    "        self.num_arms = avg.size\n",
    "        self.ub = ub\n",
    "        self.lb = lb\n",
    "        \n",
    "        self.best_arm = np.argmax(self.means)\n",
    "        self.max_rew = np.max(self.means)\n",
    "        \n",
    "        self.restart()\n",
    "        return None\n",
    "    \n",
    "    def restart(self):\n",
    "        self.time = 0.0\n",
    "        self.emp_means = np.zeros(self.num_arms)\n",
    "        self.plays = np.zeros(self.num_arms)\n",
    "        self.ucb_arr = 1e5*np.ones(self.num_arms)\n",
    "        self.cum_reg = [0]\n",
    "        return None\n",
    "        \n",
    "    def get_best_arm(self):\n",
    "        if np.argmax(self.ucb_arr).size == 1:\n",
    "            return np.argmax(self.ucb_arr)\n",
    "        else:\n",
    "            return np.argmax(self.ucb_arr)[0]\n",
    "        \n",
    "    def update_stats(self, arm, rew):\n",
    "        self.time += 1.0\n",
    "        self.plays[arm] += 1.0\n",
    "        self.emp_means[arm] = (self.emp_means[arm]*(self.plays[arm] -1.0) + rew)/(1.0*self.plays[arm])\n",
    "        function = np.log(1 + self.time*(np.log(self.time)**2))\n",
    "        return function\n",
    "    \n",
    "    def klfunc(self, p, q):\n",
    "        if p == 0:\n",
    "            return np.log(1.0/(1.0-q))\n",
    "        if p == 1:\n",
    "            return np.log(1.0/q)\n",
    "        else:\n",
    "            return p*np.log(p/q) + (1.0-p)*np.log((1.0-p)/(1.0-q))\n",
    "        \n",
    "    \n",
    "    def klucb(self, x, d, upper=1, lower=0, precision=1e-6, max_iterations=50):\n",
    "        value = max(x, lower)\n",
    "        u = upper\n",
    "        _count_iteration = 0\n",
    "        while _count_iteration < max_iterations and u - value > precision:\n",
    "            _count_iteration += 1\n",
    "            m = (value + u) / 2.0\n",
    "            if self.klfunc(x, m) > d:\n",
    "                u = m\n",
    "            else:\n",
    "                value = m\n",
    "        return (value + u) / 2.0\n",
    "        \n",
    "        \n",
    "    def update_ucb(self, exp_num):\n",
    "        for i in range(self.num_arms):\n",
    "            if self.plays[i] == 0:\n",
    "                self.ucb_arr[i] = 1e5\n",
    "            else:\n",
    "                temp = self.klucb(self.emp_means[i],exp_num/self.plays[i])\n",
    "                self.ucb_arr[i] = min(temp,self.ub[i])\n",
    "        return None\n",
    "    \n",
    "    def update_reg(self,arm,rew_vec):\n",
    "        self.cum_reg += [self.cum_reg[-1] + rew_vec[self.best_arm] - rew_vec[arm]]\n",
    "        return None\n",
    "    \n",
    "    def iterate(self, rew_vec):\n",
    "        play = self.get_best_arm()\n",
    "        exp_num = self.update_stats(play,rew_vec[play])\n",
    "        self.update_ucb(exp_num)\n",
    "        self.update_reg(play, rew_vec)\n",
    "        return None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# GLUE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "class GLUE(object):\n",
    "    def __init__(self, ub, lb, avg):\n",
    "        \n",
    "        self.means = avg\n",
    "        self.num_arms = avg.size\n",
    "        self.ub = ub\n",
    "        self.lb = lb\n",
    "        \n",
    "        self.sg = 0.25*np.ones(self.num_arms)\n",
    "        self.lmax = np.max(self.lb)\n",
    "        if self.lmax.size>1: self.lmax = self.lmax[0]\n",
    "        if self.lmax >0.5:\n",
    "            self.sg = self.get_sg(self.lmax)*np.ones(self.num_arms)\n",
    "        else:\n",
    "            for i in range(self.num_arms):\n",
    "                if self.ub[i]<0.5:\n",
    "                    self.sg[i] = self.get_sg(self.ub[i])\n",
    "                \n",
    "        \n",
    "        self.best_arm = np.argmax(self.means)\n",
    "        \n",
    "        self.restart()\n",
    "        return None\n",
    "    \n",
    "    \n",
    "    def func(self, x, m):\n",
    "        fx = m*np.exp(x*(1-m)) + (1-m)*np.exp(-x*m)\n",
    "        gx = m*(1-m)*(np.exp((1-m)*x) - np.exp(-m*x))/fx - (2/x)*np.log(fx)\n",
    "        return gx\n",
    "    \n",
    "    def get_sg(self, m):\n",
    "        if m > 0.5: \n",
    "            m = 1-m\n",
    "        x = brentq(self.func, 0.0005, 100, args = m)\n",
    "        sg = (2/x**2)*np.log(m*np.exp((1-m)*x) + (1-m)*np.exp(-m*x))\n",
    "        return sg\n",
    "    \n",
    "    def restart(self):\n",
    "        self.time = 0.0\n",
    "        self.emp_means = np.zeros(self.num_arms)\n",
    "        self.num_plays = np.zeros(self.num_arms)\n",
    "        self.ucb_arr = 1e5*np.ones(self.num_arms)\n",
    "        self.cum_reg = [0]\n",
    "        return None\n",
    "    \n",
    "    def get_best_arm(self):\n",
    "        if np.argmax(self.ucb_arr).size == 1:\n",
    "            return np.argmax(self.ucb_arr)\n",
    "        else:\n",
    "            return np.argmax(self.ucb_arr)[0]\n",
    "    \n",
    "    def update_stats(self, arm, rew):\n",
    "        self.time += 1.0\n",
    "        self.num_plays[arm] += 1.0\n",
    "        self.emp_means[arm] = (self.emp_means[arm]*(self.num_plays[arm]-1.0) + rew)/self.num_plays[arm]\n",
    "        return None\n",
    "    \n",
    "    def update_ucb(self):\n",
    "        for i in range(self.num_arms):\n",
    "            if self.num_plays[i] == 0:\n",
    "                self.ucb_arr[i] = self.ub[i]\n",
    "            else:\n",
    "                func = 2*self.sg[i]*np.log(1 + self.time*(np.log(self.time)**2))\n",
    "                temp = self.emp_means[i] + np.sqrt(func/self.num_plays[i])\n",
    "                self.ucb_arr[i] = max(self.lb[i],min(temp, self.ub[i]))\n",
    "        return None\n",
    "    \n",
    "    def update_reg(self, arm, rew_vec):\n",
    "        self.cum_reg += [self.cum_reg[-1] + rew_vec[self.best_arm] - rew_vec[arm]]\n",
    "        return None\n",
    "    \n",
    "    def iterate(self, rew_vec):\n",
    "        play = self.get_best_arm()\n",
    "        self.update_stats(play, rew_vec[play])\n",
    "        self.update_ucb()\n",
    "        self.update_reg(play, rew_vec)\n",
    "        return None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Reward function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_rew_vec_bern(avg):\n",
    "    rew = np.zeros(avg.size)\n",
    "    for i in range(avg.size):\n",
    "        temp = np.random.rand()\n",
    "        if temp<=avg[i]: rew[i] = 1\n",
    "    return rew\n",
    "\n",
    "def get_rew_vec_clippedgaussian(avg, var = 0.1):\n",
    "    rew = np.zeros(avg.size)\n",
    "    for i in range(avg.size):\n",
    "        temp = avg[i] + np.sqrt(var)*np.random.randn()\n",
    "        rew[i] = max(0, min(temp, 1))\n",
    "    return rew\n",
    "\n",
    "def get_rew_vec_clippedunif(avg):\n",
    "    rew = np.zeros(avg.size)\n",
    "    bound = 0.1*np.ones(avg.size)\n",
    "    for i in range(avg.size):\n",
    "        while avg[i]+bound[i]>1 or avg[i]-bound[i]<0:\n",
    "            bound[i] /= 2.0\n",
    "        rew[i] = np.random.uniform(avg[i]-bound[i], avg[i]+bound[i])\n",
    "    return rew"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_algos(ub, lb, avg, num_iter, num_inst):\n",
    "    \n",
    "    reg1, reg2 = np.zeros((num_inst, num_iter)), np.zeros((num_inst, num_iter))\n",
    "    reg3, reg4 = np.zeros((num_inst, num_iter)), np.zeros((num_inst, num_iter))\n",
    "    \n",
    "    algo1 = BUCB(ub, lb, avg)\n",
    "    algo2 = UCBImproved(ub, lb, avg)\n",
    "    algo3 = BKL(ub, lb, avg)\n",
    "    algo4 = GLUE(ub, lb, avg)\n",
    "    \n",
    "    for k in range(num_inst):\n",
    "        algo1.restart()\n",
    "        algo2.restart()\n",
    "        algo3.restart()\n",
    "        algo4.restart()\n",
    "        \n",
    "        if (k+1)%10 == 0:\n",
    "            print('Instance number = ', k+1)\n",
    "        \n",
    "        for t in range(num_iter-1):\n",
    "            rew_vec = get_rew_vec_clippedunif(avg)\n",
    "            \n",
    "            algo1.iterate(rew_vec)\n",
    "            algo2.iterate(rew_vec)\n",
    "            algo3.iterate(rew_vec)\n",
    "            algo4.iterate(rew_vec)\n",
    "      \n",
    "        \n",
    "        reg1[k,:], reg2[k,:] = np.asarray(algo1.cum_reg), np.asarray(algo2.cum_reg)\n",
    "        reg3[k,:], reg4[k,:] = np.asarray(algo3.cum_reg), np.asarray(algo4.cum_reg)\n",
    "        \n",
    "    return reg1, reg2, reg3, reg4"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Params"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "avg = np.asarray([0.98,0.2, 0.6])\n",
    "ub = np.asarray([1,1,1])\n",
    "lb = np.asarray([0.98,0,0.2])\n",
    "num_iter,num_inst = int(5e4),100\n",
    "save = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "ename": "ValueError",
     "evalue": "could not broadcast input array from shape (99999) into shape (50000)",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mValueError\u001b[0m                                Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-9-ab044426c79a>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mreg1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg2\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg3\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg4\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mrun_algos\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mub\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mlb\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mavg\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnum_iter\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnum_inst\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m<ipython-input-7-919dbdd1eddc>\u001b[0m in \u001b[0;36mrun_algos\u001b[0;34m(ub, lb, avg, num_iter, num_inst)\u001b[0m\n\u001b[1;32m     26\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     27\u001b[0m         \u001b[0mreg1\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg2\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0masarray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0malgo1\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcum_reg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0masarray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0malgo2\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcum_reg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 28\u001b[0;31m         \u001b[0mreg3\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg4\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0masarray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0malgo3\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcum_reg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0masarray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0malgo4\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcum_reg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     29\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     30\u001b[0m     \u001b[0;32mreturn\u001b[0m \u001b[0mreg1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg2\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg3\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mreg4\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mValueError\u001b[0m: could not broadcast input array from shape (99999) into shape (50000)"
     ]
    }
   ],
   "source": [
    "reg1, reg2, reg3, reg4 = run_algos(ub, lb, avg, num_iter, num_inst)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "r1 = np.mean(reg1,0)\n",
    "r2 = np.mean(reg2,0)\n",
    "r3 = np.mean(reg3,0)\n",
    "r4 = np.mean(reg4,0)\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "plt.figure(figsize = (12,4))\n",
    "plt.style.use('tableau-colorblind10')\n",
    "plt.subplot(1,2,1)\n",
    "x = len(avg)+1\n",
    "plt.plot(range(1,x,1),avg,'ro',label='Mean')\n",
    "plt.plot(range(1,x,1),ub,'gs',label='Upper bound')\n",
    "plt.plot(range(1,x,1),lb,'bs',label='Lower bound')\n",
    "plt.vlines(range(1,x,1), ymin=ub, ymax=lb, color = 'k')\n",
    "plt.xticks(range(1,x,1))\n",
    "plt.xlabel('Arms', fontsize=14)\n",
    "plt.ylabel('Reward', fontsize=14)\n",
    "plt.legend()\n",
    "\n",
    "plt.subplot(1,2,2)\n",
    "plt.plot(r1,label = 'BUCB', linewidth = 3.0)\n",
    "plt.plot(r2,label = 'Improved UCB', linewidth = 3.0)\n",
    "plt.plot(r3,label = 'B-KL-UCB', linewidth = 3.0)\n",
    "plt.plot(r4,label = 'GLUE', linewidth = 3.0)\n",
    "\n",
    "plt.grid()\n",
    "plt.legend()\n",
    "plt.show()\n",
    "plt.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.7.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
