{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Combine Good Twin-System Explanation with Clustering of FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import pickle\n",
    "import time\n",
    "import scipy\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from sklearn.neighbors import KNeighborsClassifier\n",
    "\n",
    "from functions import *\n",
    "\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "netC = load_cnn()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader, test_loader = load_dataloaders()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train, y_train, X_test, y_test = get_MNIST_data()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train_c = np.load(\"data/X_train_cont.npy\")\n",
    "X_test_c = np.load(\"data/X_test_cont.npy\")\n",
    "X_train_x = np.load(\"data/X_train_x.npy\")\n",
    "X_test_x = np.load(\"data/X_test_x.npy\")\n",
    "X_train_y = np.load(\"data/X_train_y.npy\")\n",
    "X_test_y = np.load(\"data/X_test_y.npy\")\n",
    "X_train_C = np.load(\"data/X_train_conv.npy\")\n",
    "X_test_C = np.load(\"data/X_test_conv.npy\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Sanity Checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "KNeighborsClassifier(algorithm='brute', n_neighbors=1)"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Fit COLE and DkNN\n",
    "KNN = KNeighborsClassifier(n_neighbors=1, algorithm=\"brute\") \n",
    "KNN.fit(X_train_c, X_train_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Examine Explanation with White Box"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "340\n",
      "445\n",
      "447\n",
      "582\n",
      "625\n",
      "659\n",
      "674\n",
      "1014\n",
      "1039\n",
      "1226\n",
      "1232\n",
      "1260\n",
      "1299\n",
      "1393\n",
      "1414\n",
      "1527\n",
      "1737\n",
      "1878\n",
      "1901\n",
      "2035\n",
      "2118\n",
      "2130\n",
      "2135\n",
      "2189\n",
      "2369\n",
      "2414\n",
      "2488\n",
      "2597\n",
      "2654\n",
      "2927\n",
      "2939\n",
      "2995\n",
      "3030\n",
      "3225\n",
      "3422\n",
      "3534\n",
      "4284\n",
      "4740\n",
      "4783\n",
      "4814\n",
      "4823\n",
      "4838\n",
      "4860\n",
      "5937\n",
      "5955\n",
      "6576\n",
      "6597\n",
      "8408\n",
      "8527\n",
      "9015\n",
      "9642\n",
      "9664\n",
      "9693\n",
      "9811\n",
      "9839\n"
     ]
    }
   ],
   "source": [
    "for i in range(len(X_test)):\n",
    "    \n",
    "    if X_test_y[i] != y_test[i]:\n",
    "        print(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# #### Find instance where explanation is learned as different label\n",
    "\n",
    "# for i in range(len(X_test_hp)):\n",
    "    \n",
    "#     idxs = COLE.kneighbors(X=[X_test_hp[i]], n_neighbors=3, return_distance=False)\n",
    "#     neighbours = idxs[0]\n",
    "\n",
    "#     if nn_preds_train[neighbours[0]] == nn_preds_test[i]:\n",
    "#         if nn_preds_train[neighbours[0]] != y_train[neighbours[0]]:\n",
    "#             print(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Prediction: 3    Label: 5\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x234a51d90>"
      ]
     },
     "execution_count": 90,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "test_instance = 2369\n",
    "pred = X_test_y[test_instance]\n",
    "label = y_test[test_instance]\n",
    "print(\"Prediction:\", pred, \"   Label:\", label.item())\n",
    "plt.imshow(X_test[test_instance])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Query Label: 5\n",
      "Prediction: 3\n",
      " \n",
      "Neighbors:\n",
      "tensor(3) Prediction 3\n",
      "tensor(3) Prediction 3\n",
      "tensor(3) Prediction 3\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "CHP_nns = plot_query_and_nns_CHP(X_test_c, X_test, X_train,\n",
    "                                         test_instance, KNN, y_test, \n",
    "                                         y_train, netC, pred, X_train_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2: Look at FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 93,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Most Powerful Feature Contributions:\n",
      "[ 25  30  81  20  40  93  49  58  57  15  29 121  91  79  74  95 127  75\n",
      " 116   5  86  78  71  72  92  73  90  88  80  76  87  85  77  84  94  89\n",
      "  83  99  97 125 124 123 122 120 119 118 117 115 114 113 112  96 111 109\n",
      " 108 107 106 105 104 103 102 101 100  69  98 110  68   0  66  33  32  31\n",
      "  28  27  26  24  23  22  21  19  18  34  16  13  12  11  10   9   8   7\n",
      "   6   4   3   2   1  14  67  35  37  65  64 126  62  61  60  59  56  55\n",
      "  54  53  52  36  63  51  50  48  47  46  45  44  43  42  41  38  17  39\n",
      "  82  70]\n",
      " \n",
      "[ 41  30  93  10  40  71  25  57  20  67  29  58  14  63  86  77  91  68\n",
      "  90  69  72  73  74  89  75  88  87  85  76  84  83  92  80  79  78  81\n",
      "  97  95 125 124 123 122 121 119 118 117 116 115 114 113  94 112 110 109\n",
      " 108 107 105 104 103 102 101  99  98  96 111  66   0  64  65  31  28  27\n",
      "  26  24  23  22  21  19  18  16  15  13  12  11   9   8   7   6   5   4\n",
      "   3   2   1  33  34  32  36 126  62  61  60  59  56  55  54  53  52  51\n",
      "  50  35  49  42  37  38  43  44 127  46  47  48  45 106  17 100 120  39\n",
      "  82  70]\n",
      "[ 40  30  81  96  93 116 127  29  25 117  57  91  20   0  65  95  86  66\n",
      " 111   5  90  89  88  87  85  84  83   6  14  79  78  77  76  75  74  73\n",
      "  72  71  13  69  80  92  94   3 125 124 123 122 121 119 118   1   2 114\n",
      "   4 113 110 109 108 107 103 102 101  99  98  97 112  68  63  64  67  35\n",
      "  34  33  32  31   9  10  28  27  37  26  24  23  22  21  12  19  18  17\n",
      "  16  15  11  38  36  51  61  60  59  58   7  56  55  54  53  52   8  62\n",
      "  50 126  48  46  45  44  43  42  41  49 106  47 104 105 100 115 120  39\n",
      "  82  70]\n",
      "[ 40  41  10   0  93  92  91  90  89  88  87  86  85  84  83  81  80  79\n",
      "  78  77  76  75  74  73  72  71  69  68  67  66  94  95  96  97 125 124\n",
      " 123 122 121 120 119 118 117 116 115 114 113  65 112 110 109 108 107 106\n",
      " 105 104 103 102 101 100  99  98 111  64  63  62  29  28  27  26  25  24\n",
      "  23  22  21  20  19  18  17  30  16  14  13  12  11   9   8   7   6   5\n",
      "   4   3   2   1  15 126  31  33  61  60  59  58  57  56  55  54  53  52\n",
      "  51  50  32  49  47  46  45  44  43  42  38  37  36  35  34  48 127  39\n",
      "  82  70]\n",
      " \n",
      "The Common Features Are...\n",
      "70 2.491656541824341\n",
      "82 1.613413691520691\n",
      "39 0.21515260636806488\n",
      "17 0.0005377514171414077\n",
      "38 -0.0\n",
      "41 -0.0\n",
      "42 -0.0\n",
      "43 -0.0\n",
      "44 -0.0\n",
      "45 -0.0\n",
      "46 0.0\n",
      "47 0.0\n",
      "48 0.0\n",
      "50 -0.0\n",
      "51 -0.0\n",
      "63 -0.0\n",
      "36 -0.0\n",
      "52 -0.0\n",
      "53 -0.0\n",
      "54 -0.0\n",
      "55 -0.0\n",
      "56 0.0\n",
      "59 -0.0\n",
      "60 -0.0\n",
      "61 -0.0\n",
      "62 -0.0\n",
      "126 -0.0\n",
      "64 -0.0\n",
      "65 -0.0\n",
      "37 -0.0\n",
      "35 -0.0\n",
      "67 -0.0\n",
      "14 -0.0\n",
      "1 -0.0\n",
      "2 -0.0\n",
      "3 -0.0\n",
      "4 -0.0\n",
      "6 -0.0\n",
      "7 -0.0\n",
      "8 0.0\n",
      "9 -0.0\n",
      "10 -0.0\n",
      "11 -0.0\n",
      "12 -0.0\n",
      "13 -0.0\n",
      "16 -0.0\n",
      "34 -0.0\n",
      "18 -0.0\n",
      "19 -0.0\n",
      "21 -0.0\n",
      "22 -0.0\n",
      "23 0.0\n",
      "24 -0.0\n",
      "26 -0.0\n",
      "27 -0.0\n",
      "28 -0.0\n",
      "31 0.0\n",
      "32 -0.0\n",
      "33 -0.0\n",
      "66 -0.0\n",
      "0 -0.0\n",
      "68 -0.0\n",
      "110 -0.0\n",
      "98 -0.0\n",
      "69 -0.0\n",
      "100 0.0\n",
      "101 -0.0\n",
      "102 -0.0\n",
      "103 -0.0\n",
      "104 0.0\n",
      "105 0.0\n",
      "106 0.0\n",
      "107 -0.0\n",
      "108 -0.0\n",
      "109 -0.0\n",
      "111 -0.0\n",
      "96 -0.0\n",
      "112 -0.0\n",
      "113 -0.0\n",
      "114 -0.0\n",
      "115 0.0\n",
      "117 -0.0\n",
      "118 -0.0\n",
      "119 -0.0\n",
      "120 0.0\n",
      "122 -0.0\n",
      "123 -0.0\n",
      "124 0.0\n",
      "125 -0.0\n",
      "97 -0.0\n",
      "99 -0.0\n",
      "83 -0.0\n",
      "89 -0.0\n",
      "94 -0.0\n",
      "84 -0.0\n",
      "77 -0.0\n",
      "85 -0.0\n",
      "87 -0.0\n",
      "76 -0.0\n",
      "80 -0.0\n",
      "88 0.0\n",
      "90 0.0\n",
      "73 -0.0\n",
      "92 -0.0\n",
      "72 -0.0\n",
      "71 -0.0\n",
      "78 -0.0\n",
      "86 -0.006733517628163099\n",
      "5 -0.007732613477855921\n",
      "116 -0.008312450721859932\n",
      "75 -0.01279214583337307\n",
      "127 -0.017065053805708885\n",
      "95 -0.02795979380607605\n",
      "74 -0.028568049892783165\n",
      "79 -0.028892353177070618\n",
      "91 -0.03459973633289337\n",
      "121 -0.037128522992134094\n",
      "29 -0.04561863839626312\n",
      "15 -0.05174746736884117\n",
      "57 -0.06696350872516632\n",
      "58 -0.11669515818357468\n",
      "49 -0.12829016149044037\n",
      "93 -0.13273945450782776\n",
      "40 -0.23415809869766235\n",
      "20 -0.2910068929195404\n",
      "81 -0.35082969069480896\n",
      "30 -0.719525933265686\n",
      "25 -1.5474086999893188\n"
     ]
    }
   ],
   "source": [
    "dominant_features = display_nb_features(X_test_c, X_train_c, CHP_nns, test_instance)\n",
    "nb_feature = dominant_features[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 94,
   "metadata": {},
   "outputs": [],
   "source": [
    "query_activations = X_test_C[test_instance]\n",
    "neigh_activations = X_train_C[CHP_nns]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 4: Upsample Using Bilinear:\n",
    "Bilinear Upsampling will probably be the best way to go here, we will try that first."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_FAM(image, activations):\n",
    "    plt.imshow(image, alpha=1)\n",
    "    FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "    plt.imshow(FAM, cmap='jet', alpha=0.4)\n",
    "    plt.axis('off')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Query Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 96,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 96,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, axarr = plt.subplots(2,2)\n",
    "\n",
    "image = X_test[test_instance]\n",
    "axarr[0,0].imshow(image, alpha=1)\n",
    "activations = query_activations[nb_feature]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[0,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[0]]   \n",
    "axarr[0,1].imshow(image, alpha=1)\n",
    "activations = neigh_activations[0][nb_feature]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[0,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[1]]   \n",
    "axarr[1,0].imshow(image, alpha=1)\n",
    "activations = neigh_activations[0][nb_feature]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[1,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[2]]   \n",
    "axarr[1,1].imshow(image, alpha=1)\n",
    "activations = neigh_activations[0][nb_feature]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[1,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "plt.axis('off')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Cluster FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(128, 7, 7)"
      ]
     },
     "execution_count": 80,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "query_activations.shape\n",
    "# neigh_activations = X_train_C[CHP_nns]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[0.  , 0.  , 0.  , 0.  , 0.  , 0.  , 0.  ],\n",
       "       [0.  , 0.  , 0.  , 0.  , 0.  , 0.  , 0.  ],\n",
       "       [0.  , 0.  , 0.02, 0.12, 0.25, 0.  , 0.  ],\n",
       "       [0.  , 0.13, 0.4 , 0.4 , 0.58, 0.3 , 0.  ],\n",
       "       [0.  , 0.  , 0.  , 0.  , 0.06, 0.26, 0.  ],\n",
       "       [0.  , 0.  , 0.  , 0.  , 0.  , 0.  , 0.  ],\n",
       "       [0.  , 0.  , 0.  , 0.  , 0.  , 0.  , 0.  ]])"
      ]
     },
     "execution_count": 81,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.round(query_activations[nb_feature], 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "X = query_activations.reshape(128, 7*7)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "X.shape\n",
    "from scipy.spatial import distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "temp = list()\n",
    "for i in range(len(X)):\n",
    "    for j in range(len(X)):\n",
    "        dist = distance.euclidean(X[i], X[j])\n",
    "        temp.append(dist)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(128, 49)"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([1.538e+04, 4.980e+02, 0.000e+00, 0.000e+00, 0.000e+00, 2.000e+00,\n",
       "        0.000e+00, 2.500e+02, 2.000e+00, 2.520e+02]),\n",
       " array([0.        , 0.27243803, 0.54487606, 0.81731409, 1.08975212,\n",
       "        1.36219015, 1.63462818, 1.90706621, 2.17950424, 2.45194227,\n",
       "        2.7243803 ]),\n",
       " <BarContainer object of 10 artists>)"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.hist(temp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.decomposition import PCA\n",
    "from sklearn.cluster import DBSCAN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "clustering = DBSCAN(eps=0.5, min_samples=1).fit(X)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n",
       "       0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n",
       "       0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n",
       "       0, 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n",
       "       0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n",
       "       0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0])"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "clustering.labels_"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "7\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "2\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "11\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "4\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "1\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n",
      "0\n"
     ]
    }
   ],
   "source": [
    "for fam in X:\n",
    "    print( 49 - (fam==0.0).sum() )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### What To Do:\n",
    "* Train this on more complex datasets.\n",
    "* Check the distribution of features (how many could be pruned?)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "44\n"
     ]
    }
   ],
   "source": [
    "num = 0\n",
    "for dist in X_train_x.T:\n",
    "    if dist.sum() == 0.0:\n",
    "         num += 1\n",
    "print(num)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "84"
      ]
     },
     "execution_count": 42,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "128 - 44"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "img_env",
   "language": "python",
   "name": "img_env"
  },
  "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
}
