{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "fQPQTJXePjde"
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "BIpyjtMRCYiB"
   },
   "outputs": [],
   "source": [
    "def nentr(p, base=None):\n",
    "    \"\"\"\n",
    "    Calculates entropy of p to the base b. If base is None, the natural logarithm is used.\n",
    "    :param p: batches of class label probability distributions (softmax output)\n",
    "    :param base: base b\n",
    "    :return:\n",
    "    \"\"\"\n",
    "    eps = torch.tensor([1e-16], device=p.device)\n",
    "    if base:\n",
    "        base = torch.tensor([base], device=p.device, dtype=torch.float32)\n",
    "        return (p.mul(p.add(eps).log().div(base.log()))).sum(dim=1).abs()\n",
    "    else:\n",
    "        return (p.mul(p.add(eps).log())).sum(dim=1).abs()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "_GULFGbqCdWt"
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "\n",
    "\n",
    "def classwise_uce(num_classes, softmaxes, labels, n_bins=15):\n",
    "     uce_c = []\n",
    "     err_in_bin_c = []\n",
    "     avg_entropy_in_bin_c = []\n",
    "     for c in range(num_classes):\n",
    "         softmaxes_c = softmaxes[torch.where(labels == c)]\n",
    "         labels_c = labels[torch.where(labels == c)]\n",
    "         uce, err_in_bin, avg_entropy_in_bin = uceloss(softmaxes_c, labels_c, n_bins)\n",
    "         uce_c.append(uce)\n",
    "         err_in_bin_c.append(err_in_bin)\n",
    "         avg_entropy_in_bin_c.append(avg_entropy_in_bin)\n",
    "     return uce_c, err_in_bin_c, avg_entropy_in_bin_c\n",
    "\n",
    "\n",
    "def classwise_ace(softmaxes, labels, n_bins=15):\n",
    "    num_classes = softmaxes.size(1)\n",
    "    ace_c = []\n",
    "    acc_in_bin_c = []\n",
    "    avg_confidence_in_bin_c = []\n",
    "    for c in range(num_classes):\n",
    "        softmaxes_c = softmaxes[torch.where(labels == c)]\n",
    "        labels_c = labels[torch.where(labels == c)]\n",
    "        ace, acc_in_bin, avg_confidence_in_bin = aceloss(softmaxes_c, labels_c, n_bins)\n",
    "        ace_c.append(ace.item())\n",
    "        acc_in_bin_c.append(acc_in_bin)\n",
    "        avg_confidence_in_bin_c.append(avg_confidence_in_bin)\n",
    "    return ace_c, acc_in_bin_c, avg_confidence_in_bin_c\n",
    "\n",
    "\n",
    "def uceloss(softmaxes, labels, n_bins=15):\n",
    "    d = softmaxes.device\n",
    "    bin_boundaries = torch.linspace(0, 1, n_bins + 1, device=d)\n",
    "    bin_lowers = bin_boundaries[:-1]\n",
    "    bin_uppers = bin_boundaries[1:]\n",
    "\n",
    "    _, predictions = torch.max(softmaxes, 1)\n",
    "    errors = predictions.ne(labels)\n",
    "    uncertainties = nentr(softmaxes, base=softmaxes.size(1))\n",
    "    errors_in_bin_list = []\n",
    "    avg_entropy_in_bin_list = []\n",
    "\n",
    "    uce = torch.zeros(1, device=d)\n",
    "    for bin_lower, bin_upper in zip(bin_lowers, bin_uppers):\n",
    "        # Calculate |uncert - err| in each bin\n",
    "        in_bin = uncertainties.gt(bin_lower.item()) * uncertainties.le(bin_upper.item())\n",
    "        prop_in_bin = in_bin.float().mean()  # |Bm| / n\n",
    "        if prop_in_bin.item() > 0.0:\n",
    "            errors_in_bin = errors[in_bin].float().mean()  # err()\n",
    "            avg_entropy_in_bin = uncertainties[in_bin].mean()  # uncert()\n",
    "            uce += torch.abs(avg_entropy_in_bin - errors_in_bin) * prop_in_bin\n",
    "\n",
    "            errors_in_bin_list.append(errors_in_bin)\n",
    "            avg_entropy_in_bin_list.append(avg_entropy_in_bin)\n",
    "\n",
    "    err_in_bin = torch.tensor(errors_in_bin_list, device=d)\n",
    "    avg_entropy_in_bin = torch.tensor(avg_entropy_in_bin_list, device=d)\n",
    "\n",
    "    return uce, err_in_bin, avg_entropy_in_bin\n",
    "\n",
    "\n",
    "def aceloss(softmaxes, labels, n_bins=15):\n",
    "    \"\"\"\n",
    "    Modified from https://github.com/gpleiss/temperature_scaling/blob/master/temperature_scaling.py\n",
    "    \"\"\"\n",
    "    d = softmaxes.device\n",
    "\n",
    "    confidences, predictions = torch.max(softmaxes, 1)\n",
    "\n",
    "    conf_sorted = torch.sort(confidences)[0]\n",
    "    n_per_bin = conf_sorted.size(0) / n_bins\n",
    "    bin_boundaries = torch.zeros(n_bins+1, device=d)\n",
    "\n",
    "    for i in range(1, n_bins):\n",
    "        bin_boundaries[i] = conf_sorted[int(np.round(i*n_per_bin))-1]\n",
    "\n",
    "    bin_boundaries[0] = 0\n",
    "    bin_boundaries[-1] = 1\n",
    "\n",
    "    bin_lowers = bin_boundaries[:-1]\n",
    "    bin_uppers = bin_boundaries[1:]\n",
    "    \n",
    "    accuracies = predictions.eq(labels)\n",
    "    accuracy_in_bin_list = []\n",
    "    avg_confidence_in_bin_list = []\n",
    "\n",
    "    ece = torch.zeros(1, device=d)\n",
    "    for bin_lower, bin_upper in zip(bin_lowers, bin_uppers):\n",
    "        in_bin = confidences.gt(bin_lower.item()) * confidences.le(bin_upper.item())\n",
    "        prop_in_bin = in_bin.float().mean()\n",
    "        if prop_in_bin.item() > 0.0:\n",
    "            accuracy_in_bin = accuracies[in_bin].float().mean()\n",
    "            avg_confidence_in_bin = confidences[in_bin].mean()\n",
    "            ece += torch.abs(avg_confidence_in_bin - accuracy_in_bin) * prop_in_bin\n",
    "\n",
    "            accuracy_in_bin_list.append(accuracy_in_bin)\n",
    "            avg_confidence_in_bin_list.append(avg_confidence_in_bin)\n",
    "\n",
    "    acc_in_bin = torch.tensor(accuracy_in_bin_list, device=d)\n",
    "    avg_conf_in_bin = torch.tensor(avg_confidence_in_bin_list, device=d)\n",
    "\n",
    "    return ece, acc_in_bin, avg_conf_in_bin"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "n96Av8UqDqhN"
   },
   "outputs": [],
   "source": [
    "labels = torch.tensor(np.concatenate([np.zeros(50), np.ones(50)]))\n",
    "perfect_model = torch.cat([torch.tensor([[1.0, 0.0]]).repeat(50,1),\n",
    "                           torch.tensor([[0.0, 1.0]]).repeat(50,1)])\n",
    "constant_model = torch.tensor([[0.5, 0.5]]).repeat(100,1)\n",
    "random_model_1 = (1*torch.rand(100,2)).softmax(dim=1)\n",
    "random_model_2 = (1.5*torch.rand(100,2)).softmax(dim=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 34
    },
    "colab_type": "code",
    "id": "od8plCe6C7Pm",
    "outputId": "4d875340-8f90-4b38-8481-79ac0a9a8270"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "50.0\n"
     ]
    }
   ],
   "source": [
    "ace, _, _ = classwise_ace(constant_model, labels, 10)\n",
    "print(np.mean(ace)*100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 34
    },
    "colab_type": "code",
    "id": "lg8c8VhzDAlc",
    "outputId": "73f773ce-7779-4858-da4e-c042b68e7134"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([50.])\n"
     ]
    }
   ],
   "source": [
    "uce, _, _ = uceloss(constant_model, labels)\n",
    "print(uce*100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "WOqYPhgrWgnY"
   },
   "outputs": [],
   "source": [
    "uce_p = []\n",
    "uce_c = []\n",
    "uce_1 = []\n",
    "uce_2 = []\n",
    "for n_bins in range(5, 100, 5):\n",
    "    uce_p.append(uceloss(perfect_model, labels, n_bins)[0])\n",
    "    uce_c.append(uceloss(constant_model, labels, n_bins)[0])\n",
    "    uce_1.append(uceloss(random_model_1, labels, n_bins)[0])\n",
    "    uce_2.append(uceloss(random_model_2, labels, n_bins)[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 297
    },
    "colab_type": "code",
    "id": "hQ7oG3dkhUyk",
    "outputId": "cb47eb0f-258c-4922-a480-14dca91e959f"
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'UCE')"
      ]
     },
     "execution_count": 30,
     "metadata": {
      "tags": []
     },
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYIAAAEHCAYAAACjh0HiAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAVbElEQVR4nO3dbZAd5Xmn8eueN0mjAUlIA5JGwhKSEiTjgJ0JsIVtsNepgootUoWzhqwd8DpFtjZs7Nj7QrJb7Jr9YscuJ06FSoXYJK7NxphlvVllVw5xHAPGSSiJQMVGMrbEq0YSGoEkGAnNm+79cI7E0ehIA2J6jkbP9auamu6nn9N9T1dP/093n9MdmYkkqVxtrS5AktRaBoEkFc4gkKTCGQSSVDiDQJIK19HqAt6sRYsW5YoVK1pdhiTNKI899tjezOxtNm3GBcGKFSvYvHlzq8uQpBklIp472TRPDUlS4QwCSSqcQSBJhTMIJKlwBoEkFa7SIIiIayPiqYjYFhG3N5l+S0QMRsQT9Z9frbIeSdKJKvv4aES0A3cBPw/sADZFxIbM3DKh6zcy87aq6pAknVqV3yO4HNiWmU8DRMS9wPXAxCCYHt+6HXb/oCWLlqQpsfgdcN3npny2VZ4a6gNeaBjfUW+b6IaI+KeIuD8iljebUUTcGhGbI2Lz4OBgFbVKUrFa/c3ivwS+npnDEfFrwNeA90/slJl3A3cD9Pf3n96TdCpIUUk6G1R5RDAANL7DX1ZvOyYzX8rM4froV4CfrbAeSVITVQbBJmBNRKyMiC7gRmBDY4eIWNIwuh7YWmE9kqQmKjs1lJljEXEb8ADQDtyTmU9GxJ3A5szcAPxGRKwHxoCXgVuqqkeS1FzMtIfX9/f3p3cflaQ3JyIey8z+ZtP8ZrEkFc4gkKTCGQSSVDiDQJIKZxBIUuEMAkkqnEEgSYUzCCSpcAaBJBXOIJCkwhkEklQ4g0CSCmcQSFLhDAJJKpxBIEmFMwgkqXCtfni9JKlBZvLKyCsMDA2wa2hX7ffB2u+P/PRHuKrvqilfpkEgSdMoM9k3vO/YTn7n0E52HtzJzqGdx3b6B0cPHvea7o5ulvYsZWh0qJKaigmCRwYe4dvPfbvVZUgq0JE8wt7X9rJzaCe7Du7itbHXjpve09lDX08fy85ZxhVLrmDp3KUs7an99PX0cW7XuUREZfUVEwQDrw7wyMAjrS5DUqEWzl7IynkruarvqmM7+r6ePpb0LOHcrnNbWpsPr5ekAvjweknSSRkEklQ4g0CSCmcQSFLhDAJJKpxBIEmFMwgkqXAGgSQVziCQpMIZBJJUuEqDICKujYinImJbRNx+in43RERGRNOvP0uSqlNZEEREO3AXcB2wDrgpItY16XcO8Eng0apqkSSdXJVHBJcD2zLz6cwcAe4Frm/S778BnwcOV1iLJOkkqgyCPuCFhvEd9bZjIuJdwPLM/H+nmlFE3BoRmyNi8+Dg4NRXKkkFa9nF4ohoA74EfGayvpl5d2b2Z2Z/b29v9cVJUkGqDIIBYHnD+LJ621HnAJcAD0bEs8CVwAYvGEvS9KoyCDYBayJiZUR0ATcCG45OzMwDmbkoM1dk5grgH4D1melTZyRpGlUWBJk5BtwGPABsBe7LzCcj4s6IWF/VciVJb06lzyzOzI3Axgltd5yk7zVV1iJJas5vFktS4QwCSSqcQSBJhav0GoEkzWQ5Ps7hrT+CTDqX9dE+fz4R0eqyppxBIEkNxl99lYPf/z5D332Qoe99j/GXXz42ra27m86+vtd/li2js28pnX19dC1bRtu5587IoDAIJBUtMxl55lmGHnyQoYce4tBjj8HYGG3z5tHznvfQc/V7aevuZnRggJEdOxgd2MnowACHNm/myNDQcfNq6+k5LiC6jg330bFwYe2IorOzRX/pyRkEks44mcnYiy+Shw/TsXgxbbNnT+n8j4yMcGjTJoYeeoihBx9i9PnnAZi1Zg0LP/5xeq65mjmXXkp0nHwXmZkceeWVEwJidMcORp9/noN///fkoUMnvK6tp4f2+fNpX7Cg9nv+fNoXzD823HFce214qv/+iQwCSS2V4+OMPPcch7ds5fDWLQxv3crhrT9ifN++Y33a58+nY8kSOi+4gI4li+lcvITOxRfQsXgJnUsW18Kiq+uUyxkbHGTo4YcZevBBDn7/7zhy6BAxaxbdV17BebfczDlXX01nX98p59EoImifN4/2efOYve6EO+yTmYzv38/ojgFGBwYYe/klxvfvZ3zf/trv/fsZ37ePkWeeYXz//hOOLo5b1uzZtC9YwPm/+SnmrZ/67+MaBFLBMpMjr77K2J49jA0OMrZnD6N79pCvHaZ90UI6Fi6io3cRHYtqP23d3W9peUdGRhj+8U9e3+Fv2crhH//49XfOnZ3MWrOanve/j9lr19HWM5ex3S8yunsXY7t2M7prF689/jjjBw6cMO/2886jc/HiCYGxmJFnn2PooYc4/MMfAtCxeDHnfuhD9FxzNXOvvJK2OXPe0t90MhFBx4IFdCxYwJx3XDJp/xwZYfzAAcb372ds376GsHg9ODrOv6CSWg0C6Sx0sh18bXjwWNvYnj3k8PAbnm9bdzftvYtqAVEPh47eRbQvWnR8aCxcWNvpb93K4aM7/K1bGd6+HcbGavOaO5dZay9m/g03MPvii5m9bi2zVq0iJnlnD3Dk0CFGd7/I2Iu7Gd21uxYU9cAYff55Dm3axJFXXql1jmDOZZfR+6lP0fO+a5j1Uz91Rl7Qja4uOnp76ejtZdY0L9sgkFpkfGiIke3bGd62neHt2xnevo3RgQHItzbfHB5mbHCw6Q6+be5cOs4/n47zz2fOpZfWh2s7n856e0dvL9HVxfi+fYy99BJjg3sZ27uXsb2DjO/dy9jelxjbu5fh7ds5+OijHGny7nyi9kWLmL12LT1XX83sdWuZvXYtncuXE22n91Wmtu5uZl20klkXrTxpn/Ghg4y9uJv2886jY8GC01pOKQwCqWLjBw7UdvTbth234x/bvftYn+jspOuii+hasYLoeGufKonOztq78ok7+d5e2ubOfcPzOfrulItP3e/IyAjjL9XCoRYag4zt3Uu0tTN77cXMWruWzvPPf0t/0+lo75lLe8+qaV/uTFRMEBwZHiYP+zRMVefI8DAjzz57wrv88cG9x/rE7NnMuugiui//OWatWs2s1auYtWoVncuWnfITKmeytq4u2pYsoXPJklaXotM0M7e807Dvz/6MPV/4YqvLUCHaurvpWr2anne/h1mrV9G1ahWzVq+mc+nS0z4dIlWlmCDovuJKLvjt32p1GTqbtbfTdeHbmLV6FR2LF5+RFySlZooJgjmXvJ05l7y91WVI0hnHY1RJKpxBIEmFMwgkqXAGgSQVziCQpMIZBJJUOINAkgpnEEhS4QwCSSqcQSBJhTMIJKlwBoEkFc4gkKTCGQSSVLhKgyAiro2IpyJiW0Tc3mT6v46IH0TEExHxSESsq7IeSdKJKguCiGgH7gKuA9YBNzXZ0f95Zr4jMy8Dfgf4UlX1SJKaq/KI4HJgW2Y+nZkjwL3A9Y0dMvOVhtG5QFZYjySpiSqfUNYHvNAwvgO4YmKniPh14NNAF/D+ZjOKiFuBWwEuvPDCKS9UkkrW8ovFmXlXZq4C/iPwn0/S5+7M7M/M/t7e3uktUJLOclUGwQCwvGF8Wb3tZO4FfrHCeiRJTVQZBJuANRGxMiK6gBuBDY0dImJNw+gvAD+psB5JUhOVXSPIzLGIuA14AGgH7snMJyPiTmBzZm4AbouIDwCjwD7g5qrqkSQ1V+XFYjJzI7BxQtsdDcOfrHL5kqTJtfxisSSptQwCSSqcQSBJhTMIJKlwBoEkFc4gkKTCGQSSVLhTBkFE3Ncw/PkJ0/66qqIkSdNnsiOCxltA/PyEad79TZLOApMFwameD+CzAyTpLDDZLSa6I+Kd1AJjTn046j9zqi5OklS9yYJgN68/PrJx+Oi4JGmGO2UQZOY101SHJKlFJvvU0Ecj4mNN2j8WEb9cXVmSpOky2cXifwv87ybt3wQ+M/XlSJKm22RB0JmZQxMbM/Mg0FlNSZKk6TRZEMyJiLkTGyPiHKCrmpIkSdNpsiD4KnB/RLztaENErKD2oPmvVleWJGm6TPapoS9GxKvAwxHRU28eAj6XmX9YeXWSpMqdMggi4tP1wd8Huql9kexZ4HvVliVJmi6TnRo6p/7TU+8bwM8C34qIGyuuTZI0DSY7NfTZZu0RcR7wN9SuFUiSZrDTeh5BZr5M7ehAkjTDnVYQRMT7gH1TXIskqQUmu1j8A0683fR5wE7gV6oqSpI0fSa7++gHJ4wn8FL9m8WSpLPAZBeLn5uuQiRJreHD6yWpcAaBJBXOIJCkwhkEklS4SoMgIq6NiKciYltE3N5k+qcjYktE/FNEfKfxLqeSpOlRWRBERDtwF3AdsA64KSLWTej2ONCfmT8D3A/8TlX1SJKaq/KI4HJgW2Y+nZkj1O5LdH1jh8z8bmYeqo/+A7CswnokSU1UGQR9wAsN4zvqbSfzCeBbzSZExK0RsTkiNg8ODk5hiZKkM+JicUR8FOgHvtBsembenZn9mdnf29s7vcVJ0llusltMvBUDwPKG8WX1tuNExAeA/wRcnZnDFdYjSWqiyiOCTcCaiFgZEV3AjcCGxg4R8U7gj4D1mbmnwlokSSdRWRBk5hhwG/AAsBW4LzOfjIg7I2J9vdsXqD397H9GxBMRseEks5MkVaTKU0Nk5kZg44S2OxqGP1Dl8iVJkzsjLhZLklrHIJCkwhkEklQ4g0CSCmcQSFLhDAJJKpxBIEmFMwgkqXAGgSQVziCQpMIZBJJUOINAkgpnEEhS4QwCSSqcQSBJhTMIJKlwBoEkFc4gkKTCGQSSVDiDQJIKZxBIUuEMAkkqnEEgSYUzCCSpcAaBJBXOIJCkwhkEklQ4g0CSCmcQSFLhDAJJKlylQRAR10bEUxGxLSJubzL9vRHxjxExFhEfrrIWSVJzlQVBRLQDdwHXAeuAmyJi3YRuzwO3AH9eVR2SpFPrqHDelwPbMvNpgIi4F7ge2HK0Q2Y+W592pMI6JEmnUOWpoT7ghYbxHfW2Ny0ibo2IzRGxeXBwcEqKkyTVzIiLxZl5d2b2Z2Z/b29vq8uRpLNKlUEwACxvGF9Wb5MknUGqDIJNwJqIWBkRXcCNwIYKlydJOg2VBUFmjgG3AQ8AW4H7MvPJiLgzItYDRMTPRcQO4JeAP4qIJ6uqR5LUXJWfGiIzNwIbJ7Td0TC8idopI0lSi8yIi8WSpOoYBJJUOINAkgpnEEhS4QwCSSqcQSBJhTMIJKlwBoEkFc4gkKTCGQSSVDiDQJIKZxBIUuEMAkkqnEEgSYUzCCSpcAaBJBXOIJCkwhkEklQ4g0CSCmcQSFLhDAJJKpxBIEmFMwgkqXAGgSQVziCQpMIZBJJUOINAkgpnEEhS4QwCSSqcQSBJhTMIJKlwlQZBRFwbEU9FxLaIuL3J9FkR8Y369EcjYkWV9UiSTlRZEEREO3AXcB2wDrgpItZN6PYJYF9mrgZ+F/h8VfVIkprrqHDelwPbMvNpgIi4F7ge2NLQ53rgv9aH7wf+ICIiM3Oqi/nsXz7Jlp2vTPVsJWnarFt6Lv/lQ2+f8vlWeWqoD3ihYXxHva1pn8wcAw4ACyfOKCJujYjNEbF5cHCwonIlqUxVHhFMmcy8G7gboL+//7SOFqpIUUk6G1R5RDAALG8YX1Zva9onIjqAecBLFdYkSZqgyiDYBKyJiJUR0QXcCGyY0GcDcHN9+MPA31ZxfUCSdHKVnRrKzLGIuA14AGgH7snMJyPiTmBzZm4Avgr894jYBrxMLSwkSdOo0msEmbkR2Dih7Y6G4cPAL1VZgyTp1PxmsSQVziCQpMIZBJJUOINAkgoXM+3TmhExCDzX6jrOAIuAva0u4gzi+nid6+J4ro+at2Vmb7MJMy4IVBMRmzOzv9V1nClcH69zXRzP9TE5Tw1JUuEMAkkqnEEwc93d6gLOMK6P17kujuf6mITXCCSpcB4RSFLhDAJJKpxBcIaLiOUR8d2I2BIRT0bEJ+vt50XEtyPiJ/XfC1pd63SKiPaIeDwi/m99fGVEPBoR2yLiG/VbnxchIuZHxP0R8aOI2BoR/6zU7SMifrP+f/LDiPh6RMwuedt4owyCM98Y8JnMXAdcCfx6RKwDbge+k5lrgO/Ux0vySWBrw/jngd/NzNXAPuATLamqNb4M/FVmXgxcSm29FLd9REQf8BtAf2ZeQu329zdS9rbxhhgEZ7jM3JWZ/1gffpXaP3kfcD3wtXq3rwG/2JoKp19ELAN+AfhKfTyA9wP317sUsz4iYh7wXmrP9iAzRzJzP+VuHx3AnPoTD7uBXRS6bbwZBsEMEhErgHcCjwIXZOau+qTdwAUtKqsVfg/4D8CR+vhCYH9mjtXHd1ALyxKsBAaBP6mfKvtKRMylwO0jMweALwLPUwuAA8BjlLttvGEGwQwRET3A/wI+lZmvNE6rP96ziM8BR8QHgT2Z+VirazlDdADvAv4wM98JHGTCaaBSto/6dZDrqYXjUmAucG1Li5ohDIIZICI6qYXA/8jMb9abX4yIJfXpS4A9rapvml0FrI+IZ4F7qR32fxmYXz8dALAMGGhNedNuB7AjMx+tj99PLRhK3D4+ADyTmYOZOQp8k9r2Uuq28YYZBGe4+vnvrwJbM/NLDZM2ADfXh28G/s9019YKmflbmbksM1dQuxD4t5n5L4HvAh+udytpfewGXoiIn643/XNgC2VuH88DV0ZEd/3/5ui6KHLbeDP8ZvEZLiLeDXwP+AGvnxP/bWrXCe4DLqR2W+5/kZkvt6TIFomIa4B/l5kfjIiLqB0hnAc8Dnw0M4dbWd90iYjLqF047wKeBj5O7U1ecdtHRHwW+Ai1T9s9DvwqtWsCRW4bb5RBIEmF89SQJBXOIJCkwhkEklQ4g0CSCmcQSFLhDAJJKpxBIJ2miHgwIvqbtK+PiLP+bp86e3RM3kXSm5GZG6h9s1eaETwikBpExIr6w13+uP6Ak7+OiDmneMnHIuKJ+oNQLq/P45aI+IP68J9GxO9HxN9FxNMR8eF6+5KIeLjhte+Zhj9PasogkE60BrgrM98O7AduOEXf7sy8DPg3wD0n6bMEeDfwQeBz9bZfBh6ov/ZS4ImpKFw6HZ4akk70TGYe3TE/Bqw4Rd+vA2TmwxFxbkTMb9LnLzLzCLAlIo4+F2ATcE/9zrJ/0bA8adp5RCCdqPGGZOOc+g3TxJt1Nbt5V+P8AmrBQe3JYgPAn0bEr5xGndKUMAikt+YjcOwusQcy88AbeVFEvA14MTP/mNqdQ99VXYnSqXlqSHprDkfE40An8K/exOuuAf59RIwCQ4BHBGoZb0MtSYXz1JAkFc5TQ9IkIuIuas++bfTlzPyTVtQjTTVPDUlS4Tw1JEmFMwgkqXAGgSQVziCQpML9f2scPpLEeZ9PAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light",
      "tags": []
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import pyplot as plt\n",
    "\n",
    "plt.plot(range(5, 100, 5), uce_p)\n",
    "plt.plot(range(5, 100, 5), uce_c)\n",
    "plt.plot(range(5, 100, 5), uce_1)\n",
    "plt.plot(range(5, 100, 5), uce_2)\n",
    "plt.xlabel(\"n_bins\")\n",
    "plt.ylabel(\"UCE\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "gO2ZXuYyhs4Z"
   },
   "outputs": [],
   "source": [
    "uce_p = []\n",
    "uce_c = []\n",
    "uce_1 = []\n",
    "uce_2 = []\n",
    "for n_bins in range(5, 100, 5):\n",
    "    uce_p.append(aceloss(perfect_model, labels, n_bins)[0])\n",
    "    uce_c.append(aceloss(constant_model, labels, n_bins)[0])\n",
    "    uce_1.append(aceloss(random_model_1, labels, n_bins)[0])\n",
    "    uce_2.append(aceloss(random_model_2, labels, n_bins)[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 297
    },
    "colab_type": "code",
    "id": "VLTtF-cmsXWL",
    "outputId": "f02c3190-fb44-479d-f080-6436a703f5ec"
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'UCE')"
      ]
     },
     "execution_count": 32,
     "metadata": {
      "tags": []
     },
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light",
      "tags": []
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import pyplot as plt\n",
    "\n",
    "plt.plot(range(5, 100, 5), uce_p)\n",
    "plt.plot(range(5, 100, 5), uce_c)\n",
    "plt.plot(range(5, 100, 5), uce_1)\n",
    "plt.plot(range(5, 100, 5), uce_2)\n",
    "plt.xlabel(\"n_bins\")\n",
    "plt.ylabel(\"UCE\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "H889SBgasX7Z"
   },
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "colab": {
   "collapsed_sections": [],
   "name": "UCE_vs_ACE.ipynb",
   "provenance": []
  },
  "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.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
