{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "72a27a16",
   "metadata": {},
   "source": [
    "# Import libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f031f752",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from dirrac.data.synthetic_data import DataSynthesizer\n",
    "from dirrac.classifier.logistic import logistic_classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "209ea109",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Matplotlib config\n",
    "SMALL_SIZE = 8\n",
    "MEDIUM_SIZE = 10\n",
    "BIGGER_SIZE = 13\n",
    "\n",
    "plt.rc('font', size=BIGGER_SIZE)          # controls default text sizes\n",
    "plt.rc('axes', titlesize=BIGGER_SIZE)     # fontsize of the axes title\n",
    "plt.rc('axes', labelsize=BIGGER_SIZE)    # fontsize of the x and y labels\n",
    "plt.rc('xtick', labelsize=BIGGER_SIZE)    # fontsize of the tick labels\n",
    "plt.rc('ytick', labelsize=BIGGER_SIZE)    # fontsize of the tick labels\n",
    "plt.rc('legend', fontsize=MEDIUM_SIZE)    # legend fontsize"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cb6a6b67",
   "metadata": {},
   "source": [
    "# Synthesize data and train classifiers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b8c0ebac",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Original data parameters\n",
    "mean_0 = np.ones(2) * (-3)\n",
    "mean_1 = np.ones(2) * 3\n",
    "cov_0 = cov_1 = np.identity(2)\n",
    "n = 1000 # 1000 samples, 500 each class\n",
    "\n",
    "\"\"\" Here we synthesize original data, \n",
    "and 100 data distribution shifts \n",
    "with 33 mean shifts, 33 covariance shifts and 34 mean and covariance shift\n",
    "Shifted parameters \\alpha=[1.5, 0] and \\beta=3 \"\"\"\n",
    "sd = DataSynthesizer(mean_0, cov_0, mean_1, cov_1, n)\n",
    "features, labels = sd.synthesize_modes_data(100, [0.33, 0.33, 0.34], [0.1, 0.1], same=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "c5b9e79a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train 101 classifiers, the first is on the original dataset and the others are on shifted data\n",
    "all_coef = np.zeros((101, 2))\n",
    "for i in range(101):\n",
    "    coef = logistic_classifier(features[i], labels[i])[1].T\n",
    "    all_coef[i] = np.squeeze(coef)\n",
    "\n",
    "# Get theta with 3 types of distribution shifts\n",
    "mean_shift = all_coef[1:34]\n",
    "cov_shift = all_coef[34:67]\n",
    "both_shift = all_coef[67:101]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "4a983835",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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pzV9PW3aNPr06cO0adOYNWsW1dXVblURsFRrY4xJW3FxMWvXrmXv3r20bduWv/71rxx11FFeV+sgb7zxRtPXjz76KNu3b6ewsJChQ4fSt29fSkuTX14oVdbyMcbklzXPwPS+UNnR+XfNMxm57ciRI6mqqgLg6aef5vLLL286tnv3br773e9y0kkn0b9/f/70pz8BsGnTJgYPHsyAAQMYMGBAU3B49dVXGTp0KOPGjeOb3/wm48ePJ9K0mIceeojevXtTVlbGZZdd1lS+bt06hg4dSs+ePXnooYeayg877DAAxowZQ11dHQMHDmTu3LksX76c8ePHU1FR0bRlQ9apqusP4F7gXWAnUA3MBDrFOH8ooEBdyOONsHO+DrwM7AY+BX4Urx4DBw5UY0weWT1X9a6uqre3/+pxV1enPA3FxcW6evVqveiii3Tv3r1aXl6uS5Ys0VGjRqmq6s0336y///3vVVX1iy++0F69emldXZ3u3r1b9+7dq6qq77//vgY/k5YsWaLt27fXTz75RBsbG/WUU07RZcuWHfS6JSUlWl9f33RfVdXbb79dTz31VK2vr9etW7dqp06ddN++fU31DK1z0BlnnKFvvfVWwu8XWK5pxgGvWj6NwATgCKAc6A7MineNqh4W8mjqvBSRQmAh8B7QGRgD/FhELs1G5Y0xOWrxz2B/2F/2+/c65WkqKytj06ZNPP3004wcObLZsZdeeol77rmHiooKhg4dSn19PR9//DH79+/n6quvpl+/flx88cWsW7eu6ZqTTjqJ7t27U1BQQEVFRbO9gEJfc/z48Tz55JO0avXVKMqoUaM45JBDOPLII+nSpQv//rf7KzjE48mYj6reEvJ0q4jMANJp+w4BjgZuVtU9wEoReQS4Fpibxn2NMS1JbZSdS6OVJ2nMmDHceOONvPrqq3z++edN5arKc889xze+8Y1m51dWVtK1a1dWr17NgQMHKCoqajoWbSuFUFVVVSxdupSFCxfy3//9300rYydyrdf8MuYzHFgd55xCEflERDaLSJWIlIccKwfeV9W6kLKVgfJmRGSSiCwXkeVbt25Nv+bGQNbGEUyGdeieXHmSvvvd73L77bfTr1+/ZuXnnHMOv/zlL5vGbd5++20AamtrKSkpoaCggN///vc0NjYm/FoHDhzgk08+YdiwYdx7773U1tZSV1cX/8II2rVrx65du1K6NlWeBx8RuQinhTI5xmnrgQrgWOCbwBrgFREJpma0A2rDrtkBtA+/kao+qqqDVHVQ586d06u8MeAEmoU3QO0ngDr/LrzBApAfDf8ptG7bvKx1W6c8A7p3784NN9xwUPltt93G/v37KSsro0+fPtx2222As63C7NmzKS8vZ/369UltPNfY2MiECRPo168f/fv354YbbqBjx44p1XvixIlce+21riYceLqwqIhcDDwCXKSqS5K8dgNwj6o+JiJTgImqWhFyfCzwmKoevH9tgC0sajJiet9A4AnT4Wswda379TGxrXnGGeOp/dRp8Qz/KZRd4nWtckpOb6kgIlcB/wOMVtW/pXCLA4AEvl4NHC8ixaq6O1DWn/hdecakL8vjCCbDyi6xYOMDnnS7icgNwP3AOYkEHhE5U0S+LiIFInKYiFQCXYEXA6csBT4Cfi4ibUWkArgGp1VlzFeyMTaT5XEEY1oir8Z8ZuCMxywRkbrgI3hQRMaHPsdJHFgM7AI2AqcAZ6vqJwCq2giMBvoCnwN/Ae5T1T+48m5MbsjW2EyWxxGMaYlsMzkb88kf2RybsXEEk0dyeszHGNdlc2zGxhGMSYrnqdbGuMbGZozxDQs+Jn/Y2IzJks2bN3PZZZdx3HHHMXDgQEaOHMn777+fldcKLg4a7je/+Q1PPPEEAOvXr6eiooL+/fvzwQcf8NRTT2WlLumw4GPyR9klMPohZ4wHcf4d/ZB1l5m0qCpjx45l6NChfPDBB6xYsYK7777b9fXUrr32Wq644goA/vjHPzJu3DjefvttPvnkE18GH09WtfbLw1a1Nib/LPpgkZ797Nnab1Y/PfvZs3XRB4vSut/ixYt18ODBEY8dOHBAb7zxRu3Tp4/27dtX//CHP6iq6qWXXqqLFn31uldeeaU+++yzza6trq7WwYMHa3l5ufbp00eXLl2qqs5q1LfccouWlZXpySefrJs3b1ZVZzXr++67T6uqqrRr165aWlqqQ4cO1ZNPPlnbt2+v5eXl+sADD6T1XoPI4VWtjTHGdVUbq6h8o5Ka3TUoSs3uGirfqKRqY1XK91y7di0DBw6MeGz+/PmsWrWK1atX8/LLLzNt2jRqamq49NJLeeYZJ8V/3759LF68mFGjRjW79qmnnuKcc85pur6iogJw9gY65ZRTWL16NUOGDGHmzJnNrhs5ciTXXnstU6dOZcmSJdxzzz0MHjyYVatWMXXq1JTfZ6ZZ8DHG5I0ZK2dQ31jfrKy+sZ4ZK2dk5fVef/11Lr/8cgoLC+natStnnHEGb731Ft/61rdYsmQJX375Jc8//zxDhgyhbdvm45Ennngijz/+OJWVlbzzzju0a9cOgDZt2jRtzz1w4MCIWy3kAgs+xpi8sXn35qTKE9GnTx9WrFiR1DVFRUUMHTqUF198kblz53LppQdvPTZkyBCWLl3KUUcdxcSJE5uSCVq3bo2Is7KYX7dLSIQFH2P8zLZqyKhuxd2SKk/EmWeeyZdffsmjjz7aVLZmzRqWLVvG4MGDmTt3Lo2NjWzdupWlS5dy0kknAXDppZfy+OOPs2zZMs4999yD7vvRRx/RtWtXrr76ar7//e+zcuXKlOrnxXYJibDgY4xf2VYNGTd5wGSKCoualRUVFjF5QKwdXWITERYsWMDLL7/McccdR58+fbj55pvp1q0bY8eOpaysjPLycs4880x+8Ytf0K2bE+hGjBjBa6+9xllnnUWbNm0Ouu+rr75KeXk5/fv3Z+7cuUyenFody8rKKCwspLy8nOnTp6f8PjPNltex5XWMX9lWDVlRtbGKGStnsHn3ZroVd2PygMmM6jkq/oWmiS2vY0xL1LROXITAA7ZVQ5pG9RxlwcYHLPgY4yfBrrb9MXaTDC4HZIuZmhxmwccYt8UKGot/FjvwBJcDCg9SwfEgsABkcoIlHBjjpnhJBLG61KQQyr/tBJdIQWr/XqfcmBxgwccYN0ULGs//2Pk61grb2girn3IClW3dbXKcBR9j3BQtOOzdDvceC71GHLzydqj9e2HBtdD28MjHbXsIkyMs+BjjpljBYe92p2VT/u3AyttRaCPsq4OC1s3LbXsIzxQWFlJRUUF5eTkDBgzgjTfeiHn+pk2bmq00PWvWLH74wx+m9NoTJ05k3rx5B5VXV1czbty4pueXX345ZWVlTJ8+nQcffJA9e/ak9HqZYsHHGDfFCw7798KGl5x5PLECUOM+OKSdbQ/hE23btm1aAPTuu+/m5ptvjnl+ePDJhtLS0qagtHnzZt566y3WrFnD1KlTLfgY0+KELodz77HOI3RpnLJLoG2n2PcIds1F2vwu1N4vnCBVucP51wJPQmoXLmTDmcN574TebDhzOLULF2b0/jt37uTww51uUVVl2rRp9O3bl379+jF37lwAbrrpJpYtW0ZFRUXTqgPV1dWce+659OrVi//6r/+KeO+bbrqJ3r17U1ZWxo033thUvnTpUk477TR69uzZFHA2bdpE3759AWc1hc8++4yKigruuOMOqqurGTZsGMOGDcvoe0+GpVobkynh6c97t391LDQV+lv3xp7LE+yaCwaTBdc6XW3RzjMJq124kJrbforWOytbN1RXU3Ob0xrtMHp0yvfdu3cvFRUV1NfXU1NTwyuvvAI031Jh27ZtnHjiiQwZMoR77rmH+++/n0WLFgFOt9uqVat4++23OeSQQ/jGN77B9ddfz9e+9lXr9/PPP2fBggWsX78eEWHHjh1Nx2pqanj99ddZv349Y8aMadbdBvDnP/+Z8847j1WrVgHw+OOPs2TJEo488siU33O6PGn5iMi9IvKuiOwUkWoRmSkiUf8cFJGRIvKKiGwTkS9EZJmIDA47R0Vkj4jUhTw6ZP/dGBMQb45OMBU6uKNqpBZQ+LhN2SUw9jfRt/+2hUeTsmX6g02BJ0jr69ky/cG07hvsdlu/fj0vvPACV1xxBaoadUuFSIYPH06HDh0oKiqid+/efPTRR82OB49973vfY/78+Rx66KFNxy644AIKCgro3bu36zuopsqrbrdGYAJwBFAOdAdmxTj/cOCXwNeBzsBTwPMiEt4pPkJVDwt51Ga85saECwaAaMvhhAp2qZVdAj/+EC6cGX/cJtr23+D9wqM5FvwaamqSKk/FqaeeyrZt29i6dWtS1x1yyCFNX0faKqFVq1b885//ZNy4cSxatKjZStih1+bKep2edLup6i0hT7eKyAwg6k+tqs4JK/q1iNwOnAgk8BtvTJYkshxOqPCusrJLEhuriXTe9L7RJ5q6Mf6Tg6sstCopoaG6OmJ5pqxfv57GxkaOOOIIBg8ezCOPPMKVV17J9u3bWbp0Kffddx+fffZZ0tsc1NXVsWfPHkaOHMnpp59Oz549U65jcJsFL7vd/DLmMxxYnejJItIPOBJ4J+zQsyLSGvgAuFdV50e4dhIwCaBHjx4pV9gYIH5XW6iC1plNhU52ommm14KLtcqCT4NPl6lTmo35AEhREV2mTknrvsExH3BaHrNnz6awsJCxY8fy97//nfLyckSkaUuFI444ommbg4kTJzYlKMSya9cuzj//fOrr61FVHnjggZTrO2nSJM4991xKS0tZsmRJyvdJh+dbKojIRThdbmeoatzdkkSkC/A6MF9VbwopHw78LfD0/MA9x6rqC9HuZVsqmLRVdgQS/B0qbAPn/2/mPpijdfVJIeiB5gEmUgutddv00rOjvndxMvB8qnbhQrZMf5CGmhpalZTQZeqUtJIN8lEmtlTwNPiIyMXAI8BFqho3/IpIKfBXYAlwvcaovIjMBIpU9TvRzrHgY9IWMwBEylDL4F48iXT5BQNMtC0a0qmP7TeUtzIRfDyb5yMiV+EEntEJBp5jgGXA86r6w1iBJ+AAIGlX1JhYIs3Fad02cuCBzK69Fp6IIIUHnxPsBsvGWnDR3rutsmAS4FWq9Q3A/cA5qvq3BM7/Jk5X29OqemOE431F5CQRaSMirUXkAuA7xEhiMCYjomWiRVudINNzc8ou+WqiqR6IfE5wjCfT9Yn23n063mP8xauEgxlAA7BE5KvGiaoeBiAi44FHgs+BHwNHAVNEZErIfa4JZMJ1Bn4FHAPsw0k4+K6q/jm7b8MYomesRRpjyWaroEP3KN1ggbGfWPVJNRkh0Ww9Y8J4lWodszssEFDmhDy/CrgqxvlLgD4Zq6Ax6QrdHM6tnUZ7jYDlv6NZEkAwwMSqTw6mTJvc53m2m5cs4cC0GBGTDwQGfRfOi5OSm63EAdvmu8XKRMKBX+b5GGPiSXr7bXVWyI4nG8kI1poycdiq1sbkgtljYP7VyW+/nUgAyUYygm3zbeKw4GOM3y36T/jwtYPLQz/M0wkg2UiZtm2+TRwWfIzxuxWzoh+LtfdPogEkGynT2WhNmRbFxnyM8btoE1bh4L1/Uh3gTyRlOpkEgnip3SbvWfAxxu+iLdUDB+/9EyuApJN9FimBYP4k+PjNyNl0XqSam5xiwccYvxs4EZY/dnD5sWekFzwSyT5rCliRdi5RZ15Rj1Mi38MmoJoYLPgY43fBlsWKWU4LSAqdgBRv/k6oRLY/CG8Z9RoBq5+Ks2WE+noLBeNfNsnUJpmafBBv+4Nok1QT3S4Csa61PJLTq1obY1wUL/ss2iTVhHm0jbfJWRZ8jMkH8VKxMzX/xiaSmgTZmI8xLU28ZXgilUdbETu8662wDbQ5DPZ+QdSWkU0kNQmw4GNMSxIvqy3ZeTnl33bWh4sUsKIuSGoTSU18FnyMaUkSyWqLJJV5OTaR1KTBgo8xLUk6a6olOy/HJpKaNFjwMaYlibWbaTbYRFKTIst2M6YlycYK1X605hlnzKmyo/OvpXfnHGv5GNOS5ENXmG1U1yJY8DGmpWnpXWGpJlUYX7FuN2NMbrGN6loET4KPiNwrIu+KyE4RqRaRmSLSKc415wau2Ssia0VkRNjxr4vIyyKyW0Q+FZEfZfddGGM8YRvVtQhetXwagQnAEUA50B2YFe1kEekJzAfuBjoE/l0gIscEjhcCC4H3gM7AGODHInJp1t6BMcYb+ZJU0cJ5EnxU9RZVfVtV96vqVmAGMDTGJVcCK1T1SVXdp6pzgJWBcoAhwNHAzaq6R1VXAo8A12bvXRiTh/yQZZaNbb+N6/yScDAcWB3jeDmwIqxsZaA8ePx9Va0LO/6D8BuJyCRgEkCPHj1Sra8x+cdPWWYtPakiD3iecCAiF+G0UCbHOK0dUBtWtgNon+DxJqr6qKoOUtVBnTt3TqXKpgWpXbiQDWcO570TerPhzOHULlzodZX8K1aWmTFJ8rTlIyIX43SPjQl0lUWzC2esJ1RHYGeCx405SO3ChdTc9lO0vh6Ahupqam5zxg06jB7tZdX8ybLMTAZ51vIRkatwAs9oVV0S5/TVwICwsv581VW3GjheRIqjHDfmIFumP9gUeIK0vp4t0x/0pkJ+52GWWdXGKkbMG0HZ7DJGzBtB1caqrL+myS6vUq1vAO4HzlHVvyVwyRPAIBG5XERai8jlwEBgduD4UuAj4Oci0lZEKoBrcIKbMRE11NQkVZ73PMoyq9pYReUbldTsrkFRanbXUPlGpQWgHOdVy2cGznjMEhGpCz6CB0VkfOhzVf0AuBC4Facr7VZgrKpuChxvBEYDfYHPgb8A96nqH1x6PyYHtSopSao873mUZTZj5QzqG5u3UOsb65mxckZWX9dklydjPqoqcY7PAeaElb0AvBDjmv/DyZozJiFdpk5pNuYDIEVFdJk6xbtK+Z0HWWabd29OqtwkpmpjFTNWzmDz7s10K+7G5AGTGdVzlGuv73m2m8kNLTErrMPo0ZTc+TNalZaCCK1KSym582eWbOAz3Yq7JVVu4vNDV6aoRtmHPQ8MGjRIly9f7nU1fC88KwycFoJ9UBs3BD8oQ7veigqLqDyt0tW/1FuSEfNGULP74LHNkuISXhr3UtzrRWSFqg5Kpw7W8jFxWVaY8dKonqOoPK2SkuISBKGkuMQCT5r80JXplxUOjI9ZVpjx2qieoyzYZFC34m4RWz5udmVayyeHeDXuYllhxrQskwdMpqiwqFlZUWERkwfEWmgmszIWfETkT5m6lzlYcNyloboaVJtm47sRgLpMnYIUNf9Btayw/GSTPePLhe+RH7oyk044EJFbIhUDV6rq8RmplUtyKeFgw5nDncATplVpKb1eWZy1161duJAt0x90XruwEBobaVVaSpepUyzZIM/YwH98+fI98irhYArwKfBZyONT4Mt0KmJi82LcpVlrC6CxsanFY4En/9hkz/jse5S4VBIOVgMvq2qzP8NFZEhmqmQiaVVSErnlk8Vxl1hZbhZ88o8fMqT8zr5HiYvb8hGRniIyUUT+n4icD0wIDzwAqvq9rNTQAN6Mu1iWmwllkz0dscZ07HuUuJjBR0R+ALwP3AH8P5zFQD8VkTdF5Nsu1M8EeDEb37LcTCg/ZEh5Ld7KAPY9SlzMhAMR2QxcparPh5R9AVwC3AC0wVngc0+2K5oNuZRw4AWvVzZoSnaoqaFVSYmNNfmA1+uBeS2RlQHy4XuUiYSDeMFnG3CUqn4ZUrZdVTsFvr4DOFJVD9quOhdY8InPqwDgdeAzJpKy2WUoB39mCsKaK9d4UCNvuBF8ZgIK/EBV9wfKQoPPocBHqpqT+1Fb8PEvr1LLjYkl3TXRWgo3Uq1/BJwAfCQiPxeRM8KOn4Azx8eYjLJkB+NHNqaTOTFTrVV1JzBYRCbhjPHcBKiIrAL2Av2AyizX0eQhL1LLjYknOHbT0sd03JDUCgcicixwGlCCE3yWquo7Wapb1lm3m3/ZmI8x/pWJbrekJpmq6ofAh+m8oDGJCAYYy3YzpmWyLRWMb3UYPdqCjTEtlG2pYIwxEeTC6tS5zFo+xhgTJnx16uBKBoAlF2SIJy0fEblMRJaJyE4RaYhz7ngRqQt7NIrIn0POeVVEvgw757zsvxNjTEtkq1Nnn1fdbl8AD+NszxCTqs5R1cOCD+AooB54MuzUO0PPU9VFGa+1ySivdmY1Jh5bnTr7POl2U9UXAURkaAqXfwfYBSzIYJWMy8JTqYM7swKWZGA81624W8SVDGx16szJxYSDa4DfBZf7CTFFRLaLyLsicrOItPaiciYx0fYKqr7pZmsJGc/ZSgbZl1MJByJyOtAbGBN26GZgPbATOBGYA7QPlIffYxIwCaBHjx7ZrK6JIeoyOY2NznFrCRkP2UoG2ZfUCgcZf3Gn2+1lVU0oCIrIE0BnVf1WnPPGA/eo6tdinWcrHLivaZXsCEvnRGILiRrjP66vcOAlEekEXAxclsDpB7AFT30n0pI58dhCosa0TF6lWheKSBHOZnSISFHgEStgXAlsA5plsYlIRxE5T0QOE0d/nMVO52ap+iZFkcZ5mhQWRiy2hUSNaZm8Sjj4Ds7CpC8ChYGv9wJHi8jgwDyd8AGZScBvVbUxrLw1cCvwGc6Yz1zgKSKM95jUZColOmorRoTSe+5GipoP8EpREV2mTknptYwx/uZVqvUsYFaUw5uAwyJcc0KUe20FTslQ1XzJy+2kM5kSHWubBFtIND/lw5bTJjJPEw68lgsJB15vLZDJHUW9fi/GX8KXsAEnnbnytEoLQD7nxk6mxmPR5sNsmf6gK6+fyR1FO4weTcmdP6NVaSmI0Kq01AJPHrMlbPJbzmS75Suvt5PO9I6itk2CCUpmCRvrnmt5rOXjc9E+5LOZBRaaYNC4Zw/SuvliEZYIYDIh2lI14eXB7rma3TUo2rTCtG1xkNss+Phcl6lTXM0CC47LNFRXgyq6YweqSmHHjlG7ymyBUJOKRJewse65lsm63XzO7SywiHNxGhqQQw/lhDf/ftD5tkCoSVWiS9jYCtMtkwWfHODmOEmyY0yxEiIs+Lgv18ZGRvUcFbd+tsJ0y2Tdbj4Rq+vKzW4t6dAhqXKvEyLMV1rq2IitMN0yWcvHB2J1XQGudmsVAOFLSATLI8l0NpxJXayxET+3fuKxFaZbJgs+PhBvLk+i3VrRVkJIZoWExtrapMq7TJ0SceKoZcO5ryWPjSTSPWdyiwUfH0il66qhuprahQubgki01tOelSupXfDHhFtOybZkbFkc/7CxEZNLbMzHB2LN5YnVfVVz20+bxn+itZ52PPNsUiskuJ3abTLHxkZMLrHg4wOxPvAjHQsKDSLxdgYNF+38ZJfACZ8XFGxZ2Vwf943qOYrK0yopKS5BEEqKS2ydNONb1u3mA4l0XVVP+6+I1waDSLTuMgoLIwagWC2qZFK7LdXaX2xsxOQKa/n4RIfRo+n1ymJOeG8dvV5Z3OyDu8Po0U5LJIJgEInWeup4ycVZ7UazVGtjTCos+OSIiN1vIhx2xhDgq+4y6dix6XBBURGHDhiQ1ZWkvVh7LlV+XAaoamMVI+aNoGx2GSPmjcj4nJxs39+YVFnwyREdRo+mw9gLmheqUrvgj80/REO6wBp37GjKbIvWqkpXriQo+HFsKtuTQlvqpFPTMljwySF1ry09qCw06SBTe/8k00LIlT16MvG9yXTLKdsLZtqCnMbPLOEgh8QbX8nE+EsqC4Xmwh49Ub831dW8d0LvuPOTsrGAarYnhfpl0unO+v1c+PAbzL/uNNoXtY5/gckL1vLJIfHGVzIx/uL1zqnZEvN7kEA3XDa+L4nuZ+PX+yfqlfe28H9b6liyfourr2v8zYJPDok3vpLK+Et4V1LEdG2cv/T9MkifiljzpYJiBZNsZPVle1KoXyadzlvxabN/jQHrdssp8eYDJbvUTaSupFhyea+e8O8NqhHPixZMsrGAarILZia7XYJXC3K+sLaGNzdub3r+z03O1//4cDuVf363qfyUnp04t6//siKNO0Sj/BJm9UVFLgN+AJQDh6pq1CAoIscAHwJ7gGBld6hq95BzugC/Ac4G6oHfATer6oFY9Rg0aJAuX748jXeS26K2dESifjgDtCotpdcrixN6jWQWNXVTtPce7b2FB2pwWpVuJVcEM9dCEwiKCot8uYLBwtXVTJ27ioYDMX6GCoQHL63gvPLI89eMv4nIClUdlM49vOp2+wJ4GJiSxDXfUNXDAo/uYcfmBP7tDpwMjAWmpV3LFi5ql5Fq1EmtMa8L48f05qBkuyi9zupLN3PNzfk+o8tLeX7yYHp0OpSi1s0/YopaF9Cj06E8P3mwBZ4850nwUdUXVfVpYGO69xKRY4GzgGmqWquqG4F7gWvTvbcXMpXOm8h9oiYoBP76j7eqQjx+Tl5IJZjEWoUi2+Jlrt315l2UP1FOv9n9KH+inLvevKvpHC/m+/Tq2o6F1/8H+xuat372NyqLbvgPenVtl7XXTpZNxPVGLiUc/ENEtorIqyIyNKS8HKhV1Q9CylYCx4hI+/CbiMgkEVkuIsu3bt2a3RonKVMthUTvk40EhlB+X3rHy2CSrFiZa3e9eRdz/zWXA4Fe5gN6gLn/mtsUgLya7/PWh9spalNAqwKhQKCwQChqXcBbH26Pf7FLbCKud3Ih+GwDTgWOBY4BngOeF5GywPF2QPhOZzsC/x4UfFT1UVUdpKqDOnfuHPfF3VySJVMthUTvE++v/3S7mnJp6R2/i5W59uz7z0a8Jlju1Xyf+Ss/Zc+XjfQubc/8606nT2l79nzZyPyVn2X1dZNhE3G94/tsN1WtA94MPN0H/FJExgAXA2uAXUCHsMs6Bv7dlc5rZ2NiYSyZaikkc594E0TTmUBqu5xmTqzMtZuW3RTxmmBLyKtN5j78fDeTh/fi+uG9KCwQFlx3Or9cvIGX1v07q6+bDL9MxM1Hvg8+URwAJPD1aqCDiPQMjPcA9Ac2qWrkvZ8T5PZ2AZlK581GWnAqbJfTzIq2XUKBFDQFmvBycFpNkTLlsj3f5/nJQ5o9LywQppx9PFPOPj6rr5sM2/3VO550u4lIoYgUAW0Cz4sCD4lw7iki0ldEWgXOmQScASwAUNUPgZeBX4hI+0ACwo+BR9Ktp9tjFplapNNPi33m0rhKrjqp60kxy22Tuej8MhE3H3nV8vkO8HjI872Bf48Vka8BzwO9VfVjnLGeO4ESnDk864DRqroi5PrxOPN8PgO+xJnn84t0K+l2CyKRlkIi82asxeFPyU4STdRHuz6KW26bzEXm1URc49EkU7+IN8nU64mFfq+PSVw2J4mWzS5DOfj3WBDWXLkmrXtnWrYCsHFXLk8yzQleTywM5+d5Mya2bGZV+WUB0XjyJa3Z5g0lxoJPHH4as/D7vBkTXbJZVcl8gOXKuEU+pDXnS4DNBAs+OcTmzaTPq79Kk2mdJPsBlisJBbmU1pzqz0k+BNhMseCTQ/yUxZaLvPyrNJnWSSofYKN6juKlcS+x5so1vDTuJd8FHsiP7sFcCrBes+CTQ/w2BpVrvPyrNJnWSUv9AEule9CLlmo6Pye5EmD9IFcnmeatXNiy2q+8/lBPNN25pU58TGX/otAMwWALJPRe2ZDOz4lXE3pzkQUfkzdy5UO9JX+AJTPfKFYLJJvBJ52fE5s3lDgLPiZv5MqHun2AObxqqab7c2ITehNjwcfkjVz6ULcPMO9aqrn0c5LLbIWDPN5G2xg/y6Wtw/NNJlY4sJaPMSYpbi2RYy2Qls2CjzEmYW5noOVD92O+rndn83xc5OauqMZkg83gz6x8Xo7Hgo9LgitSN1RXg2rTrqgWgEwu8XquVKJyZXHPfA7mFnxcYitSm5YgF2bw51JrIleCeTZY8HGJrUhtWoJcWEHbb62JWK2wXAjm2WLBxyW2IrVpCdxcQTvVrjM/tSbitcJyIZhni2W7uaTL1CkRdyG1FalNrnEjAy2drDo/LaN09z/ujrlEUD6nk1vwcUlwMdAt0x+koaaGViUldJk6xRYJNSaCdNZ188sySlUbq6jdVxvxWGgrLB/SySOx4OOilr4ide3ChRZcTUak03Xml9ZErDGmfBjTiceCj8mIYCp5sFsxmEoOWAAySUu368wPrYlYgTIfxnTi8SThQEQuE5FlIrJTRBrinHuKiFSJyL9FpFZEVojIBWHnbBKRehGpC3n0y+qbMM1YKrnJpJYwEB8tUHY8pKPngdEPvMp2+wJ4GJiSwLmdgLlAH+Bw4E7gaRE5Mey876vqYSGPdzJZYRObpZKbTHIzqy5bogXQm066yaMa+Ysn3W6q+iKAiAxN4Ny/hBX9UURWA4OBtzJeOZOSViUlzuoNEcqNSYUfus7S4ZexJ7/KuTEfEemG0wpaHXboARF5CPgY+LWqPhLl+knAJIAePXpks6p5Jd9TyfN1cUgTW64H0GzKqeAjIsXAc0CVqi4OOXQlsAL4EhgK/EFEiBSAVPVR4FFw9vPJeqXzRD6nkru90rMxLYGnm8kFut1eVtW4QVBE2gFVwOfApaq6L8a5PwHOVdXBse5pm8mZTBgxb0TEzKyS4hJeGveSBzUyJrsysZlcTiyvIyJHAIuBauDiWIEn4AAgWa+YMfhrORdjcoVXqdaFIlIEtAk8Lwo8DgoYgTGe14D3gPGq2hB2/GgRGRa4vlBEzgCm4mTIGZN1+bw4pDGp8qrl8x1gL/AiUBj4ei9wtIgMDszTCWYDXIOTYDAOqA2Zx3NL4Hgx8ACwFSeF+3+Bn6nqL917OyaftYQ5Kca4zdMxH6/ZmI/JFMt2M/kkE2M+OZXtZoxfWUqtMcnJiYQDY4wxLYsFH2OMMa6z4GOMMcZ1FnyMMca4zoKPMcYY11nwMcYY4zoLPsYYY1xnwccYY4zrLPgYY4xxnQUfY4wxrrPgY4wxxnUWfIwxxrjOgo8xxjVVG6sYMW8EZbPLGDFvBFUbq7yukvGIrWptjHFF1cYqKt+opL6xHoCa3TVUvlEJYCuC5yFr+RhjXDFj5YymwBNU31jPjJUzPKqR8ZIFH2OMKzbv3pxUuWnZLPgYY1zRrbhbUuWmZbPgY4xxxeQBkykqLGpWVlRYxOQBkz2qkfGSJRwYY1wRTCqYsXIGm3dvpltxNyYPmGzJBnnKgo8xxjWjeo6yYGMAj7rdROQyEVkmIjtFpCGB8weJyD9FZI+IfCAiE8KOdxGR+SKyS0S2isi9ImJdisYY41NefUB/ATwMTIl3ooh0AJ4HngMOB64FfiMip4acNifwb3fgZGAsMC2D9TXGGJNBngQfVX1RVZ8GNiZw+oXAHuAXqvqlqv4VWABMAhCRY4GzgGmqWquqG4F7cYKUMcYYH8qFMZ9y4G1V1ZCylcB3Qo7XquoHYcePEZH2qroz9GYiMolA4ALqRORfWar3kcC2LN07G3KtvmB1dkOu1Reszm74Rro3yIXg0w6oDSvbAbSPc5zAOc2Cj6o+Cjya0RpGICLLVXVQtl8nU3KtvmB1dkOu1Reszm4QkeXp3iMXBuV3AR3CyjryVVCJdjx4zBhjjM/kQvBZDVSElfUPlAePdxCRnmHHN6lqeIvIGGOMD3iVal0oIkVAm8DzosBDIpy+ACgWkWki0kZEhuMkITwKoKofAi8DvxCR9oEEhB8Dj7jyZqLLetdehuVafcHq7IZcqy9Ynd2Qdn2l+Ti+O0RkIvB4hEPHAl/DSa3uraofB84/EfhfoB9QA/xUVZ8MuV8X4DfA2cCXwO+Am1T1QBbfhjHGmBR5EnyMMcbkt1wY8zHGGNPCWPAxxhjjOgs+KQgkTNwXWEdul4g8JyJHxji/i4jMFpHPA+vZrRKRUp/X+cbAOnq7RGSDiFzncn0zuv6fG5Kps4icIiJVIvJvEakVkRUicoFLVQ3WIanvcch13xIRFZHfZrN+UV472Z+LYhF5SERqRKRORN4Tkf5u1DWkDsnW+QoReTfwu/eRiFRGScbKisDamO8G6lstIjNFpFOca84NXLNXRNaKyIh4r2PBJzU3AefjrCPXPVD2+0gnBrL6FgP7cGYFdwTGA3VZr2VzydR5DHAHMF5V2wFXAPeJyNluVDQg0+v/uSHhOgOdgLlAH5w63wk8HUiucUsy9QWavtczgL9lqU7xJPNzIcAfgWOAk1T1MGAUUJ296kWUTJ3LcRKmbsaZJH8OcA3w/SzWL1wjMAE4AmcFme7ArGgnB6a5zAfuxplzeTewQESOifkqqmqPJB/AR8D3Qp4fByhwdIRzrwE+AVrnUJ3/E3gjrOzvwI0e1Hso0BDnnKsC709Cyn4PPO7R9zpunaNc9ybwn36uL/AYzlSGWcBvvfj+JvFzcQ7OupCdvKpnCnW+EKgOK3sa+JWH9T4X2Bnj+B3AsrCyZcDtse5rLZ8kiUhHoAewIlimzrpyO3H+Sgg3DNgAzAp0u60Xkalu1DUohTr/AWgvIqeLSIGIDAaOB15wobqpiLb+X6T35ksi0g2nFbQ63rleEZFzcCZ83+9xVRI1DPgQuENEtgS6Y38uIq29rlgMLwLVIjI28LvXFxgC/NnDOg0n9s9lOSGfLQFxf/9yYW03v2kX+DfWenOhjsT5JZiC8xd6GfCCiGxR1TkRzs+GZOu8BZgHLOGrrtkpqro2K7VLX7z1/3xNRIpxugyrVHWx1/WJRETaA78GLlLVRheHINJxJNAbp0u2B84cwr8Au4H/9rBeUanqbhF5HHgCaAsUAveq6kte1EdELsLpxj4jxmnRfv/6xLq3tXySF1wvLtZ6c+Hnf6aqM1R1n6ouB57EGX9xS7J1vg34Ns5fua1x/oKZKiLfy1L90hVv/T/fEpF2OB+OW3DG1vzqfmCuqr7tdUWSsAtn/OInqlqvqhtwJqu7+buXFBG5CqjE2SamDdATGCoid3pQl4uBmcAYVV0Z49SUfv8s+CRJVXcAHwMDgmWBAbf2wJoIl6zCGVs56FZZqF5EKdR5ILBAVdep412cgdvR2a9tSuKt/+dLInIETjJKNXCxqu7zuEqxjACuFZFtIrINuAyYICKbvK1WTKuilPt5Zv1A4BVV/YeqHlBn+bA5uPy7FwiCjwCjVXVJnNNXE/LZEhD/98/rQbhcfAA/Af6FsxxQe+BZ4IUo5x6NM+j5A5wmdDmwFbjUx3W+OXBur8DzE4APgNtcrG8hUITzodcQ+LqIkKSCkHM7Br6n03D+WhyOk014qsvf42Tq3A1YC8wGCj36OU62vt1DHs8ATwGlPq5zO5zluO4J/FwcG/i5/rGP63w5Tit4YOD514B/AI+5WN8bgM+BExM8/7jAZ9zlOD0ll+N0bR4T8zo3/xNayiPww3Q/zuZPu3DSDI8MHBsP1IWdPxR4O/AfsgH4gZ/rjDMWeA+wKfAh/jHwP7iYsQdMxPkLNfxxDDA4UK8eIeefCPwT2IuzQ+4ED77HCdcZuD1wbHegPPi4xY/1jXDtLDzIdkvh56IMJ/NqN05GZCUuB/sU6jwt8DmxCyd4PgZ0cLG+CuwP+7kM/XyI9Bl3LvBu4PfvXWBEvNextd2MMca4zsZ8jDHGuM6CjzHGGNdZ8DHGGOM6Cz7GGGNcZ8HHGGOM6yz4GGOMcZ0FH2OMMa6z4GOMD4hIBxH5bWDl8y0i8kOv62RMNlnwMcZjgU3PFgGCs3TNRGCGiHzdy3oZk022pYIx3rsa6AWcpapfAn8Rkc9w9nH5P09rZkyWWMvHGO9NBmYHAk9QLc5+Lsa0SBZ8jPFQYGuL3hy8S2xbnK0WjGmRLPgY463TcFYRXiAiO4IPnOX/3xSRCSLy98BjmKc1NSaDbMzHGG+VAGtVtSxYICIjgLtwlqe/ETgFOAx4WUQGqOoBT2pqTAZZy8cYb+0NPELdADwEnAy8ps4W0NtwuuGOcbd6xmSHBR9jvLUU6Ccig0SkWETuAA5R1SeBI4AvQs79IlBmTM6z4GOMh1R1Dc7umi/i7BjbBRgbOPw5cHjI6R0DZcbkPNvJ1BifEpGOwCvAqUBx4Gsb8zEtgiUcGONTqrpDRB4EXg0U/cgCj2kprOVjjDHGdTbmY4wxxnUWfIwxxrjOgo8xxhjXWfAxxhjjOgs+xhhjXGfBxxhjjOss+BhjjHGdBR9jjDGu+//Uj808NasI7QAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Plot\n",
    "fig, ax = plt.subplots()\n",
    "ax.scatter(all_coef[0][0], all_coef[0][1], s=100, marker='*', label = 'Original')\n",
    "ax.scatter(mean_shift[:, 0], mean_shift[:, 1], marker='o', label = 'Mean shift')\n",
    "ax.scatter(cov_shift[:, 0], cov_shift[:, 1], marker='o', label = 'Cov shift')\n",
    "ax.scatter(both_shift[:, 0], both_shift[:, 1], marker='o', label = 'Both shift')\n",
    "\n",
    "ax.set(xlabel='$\\\\theta_{0}$', ylabel='$\\\\theta_{1}$')\n",
    "ax.legend(loc='upper right', frameon=False)\n",
    "\n",
    "ax.set_xlim(0.5, 2)\n",
    "ax.set_ylim(1, 2.75)\n",
    "\n",
    "if not os.path.exists('result/figure6/'):\n",
    "    os.makedirs('result/figure6/')\n",
    "plt.savefig('result/figure6/classifiers.pdf', dpi=500, transparent=True)\n",
    "plt.show()"
   ]
  }
 ],
 "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.7.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
