{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import util\n",
    "from sklearn.cluster import KMeans\n",
    "import numpy as np\n",
    "import time\n",
    "from numba import jit\n",
    "\n",
    "data = util.load_normalized_datasets()\n",
    "\n",
    "opt_centers_10 = KMeans(n_clusters=10, n_init=2, max_iter=40).fit(data[3][0]).cluster_centers_\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "opt_centers_50 = KMeans(n_clusters=50, n_init=2, max_iter=40).fit(data[3][0]).cluster_centers_  \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_min_distances1(data, centers):\n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    for i in range(data.shape[0]):\n",
    "        min_distances[i] = np.min(np.linalg.norm(data[i] - centers, axis=1))\n",
    "    return min_distances\n",
    "\n",
    "def calculate_min_distances2(data, centers):\n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    batch_size = 5000\n",
    "    num_batches = int(data.shape[0]/batch_size)+1\n",
    "    for i in range(num_batches):\n",
    "        norms = np.linalg.norm(data[i*batch_size:(i+1)*batch_size] - centers[:, np.newaxis], axis=2)\n",
    "        min_distances[i*batch_size:(i+1)*batch_size] = np.min(norms, axis=0)\n",
    "    return min_distances\n",
    "\n",
    "def calculate_min_distances3(data, centers):\n",
    "    data_float32 = data.astype(np.float32)\n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    for i in range(data.shape[0]):\n",
    "        min_distances[i] = np.min(np.linalg.norm(data_float32[i] - centers, axis=1))\n",
    "    return min_distances\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for calculate_min_distances1:  0.15000152587890625\n"
     ]
    }
   ],
   "source": [
    "start_time = time.time()\n",
    "calculate_min_distances1(data[0][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances1: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for calculate_min_distances2:  0.3720829486846924\n"
     ]
    }
   ],
   "source": [
    "start_time = time.time()\n",
    "calculate_min_distances2(data[0][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances2: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for calculate_min_distances3:  0.13802528381347656\n"
     ]
    }
   ],
   "source": [
    "start_time = time.time()\n",
    "calculate_min_distances3(data[0][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances3: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "@jit(nopython=True, parallel=True)\n",
    "def calculate_min_distances7(data, centers):\n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    for i in range(data.shape[0]):\n",
    "            diff = data[i]-centers\n",
    "            min_distances[i] = np.min(np.sum(diff*diff, axis=1))\n",
    "    return min_distances"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'time' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[1], line 1\u001b[0m\n\u001b[1;32m----> 1\u001b[0m start_time \u001b[39m=\u001b[39m time\u001b[39m.\u001b[39mtime()\n\u001b[0;32m      2\u001b[0m calculate_min_distances7(data[\u001b[39m0\u001b[39m][\u001b[39m0\u001b[39m], opt_centers_50)\n\u001b[0;32m      3\u001b[0m \u001b[39mprint\u001b[39m(\u001b[39m\"\u001b[39m\u001b[39mtime for calculate_min_distances3: \u001b[39m\u001b[39m\"\u001b[39m, time\u001b[39m.\u001b[39mtime() \u001b[39m-\u001b[39m start_time)\n",
      "\u001b[1;31mNameError\u001b[0m: name 'time' is not defined"
     ]
    }
   ],
   "source": [
    "start_time = time.time()\n",
    "calculate_min_distances7(data[0][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances3: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for calculate_min_distances4:  1.5090446472167969\n"
     ]
    }
   ],
   "source": [
    "@jit(nopython=True)\n",
    "def calculate_min_distances4(data, centers):\n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    for i in range(data.shape[0]):\n",
    "        diff = data[i]-centers\n",
    "        min_distances[i] = np.min(np.sum(diff*diff, axis=1))\n",
    "    min_distances = np.sqrt(min_distances)\n",
    "    return min_distances\n",
    "start_time = time.time()\n",
    "calculate_min_distances4(data[0][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances4: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for calculate_min_distances4:  0.26200032234191895\n"
     ]
    }
   ],
   "source": [
    "@jit(nopython=True,parallel=True)\n",
    "def calculate_min_distances5(data, centers):\n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    for point,i in enumerate(data):\n",
    "        diff = point-centers\n",
    "        min_distances[i] = np.min(np.sum(diff*diff, axis=1))\n",
    "    min_distances = np.sqrt(min_distances)\n",
    "    return min_distances\n",
    "\n",
    "start_time = time.time()\n",
    "calculate_min_distances4(data[1][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances4: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "TypingError",
     "evalue": "Failed in nopython mode pipeline (step: nopython frontend)\n\u001b[1m\u001b[1m\u001b[1mNo implementation of function Function(<function norm at 0x0000015697E72C20>) found for signature:\n \n >>> norm(array(float64, 2d, C), axis=Literal[int](1))\n \nThere are 2 candidate implementations:\n\u001b[1m  - Of which 2 did not match due to:\n  Overload in function 'norm_impl': File: numba\\np\\linalg.py: Line 2351.\n    With argument(s): '(array(float64, 2d, C), axis=int64)':\u001b[0m\n\u001b[1m   Rejected as the implementation raised a specific error:\n     TypingError: \u001b[1mgot an unexpected keyword argument 'axis'\u001b[0m\u001b[0m\n  raised from c:\\Users\\au616584\\Anaconda3\\envs\\subspace\\lib\\site-packages\\numba\\core\\typing\\templates.py:791\n\u001b[0m\n\u001b[0m\u001b[1mDuring: resolving callee type: Function(<function norm at 0x0000015697E72C20>)\u001b[0m\n\u001b[0m\u001b[1mDuring: typing of call at c:\\Users\\au616584\\OneDrive - Aarhus Universitet\\Datalogi\\PhD\\Implementation_subspace_clustering\\util.py (85)\n\u001b[0m\n\u001b[1m\nFile \"util.py\", line 85:\u001b[0m\n\u001b[1mdef calculate_min_distances1(data, centers):\n    <source elided>\n    for i in range(data.shape[0]):\n\u001b[1m        min_distances[i] = np.min(np.linalg.norm(data[i] - centers, axis=1))\n\u001b[0m        \u001b[1m^\u001b[0m\u001b[0m\n",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mTypingError\u001b[0m                               Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[3], line 2\u001b[0m\n\u001b[0;32m      1\u001b[0m start_time \u001b[39m=\u001b[39m time\u001b[39m.\u001b[39mtime()\n\u001b[1;32m----> 2\u001b[0m util\u001b[39m.\u001b[39;49mcalculate_min_distances1(data[\u001b[39m0\u001b[39;49m][\u001b[39m0\u001b[39;49m], opt_centers_50)\n\u001b[0;32m      3\u001b[0m \u001b[39mprint\u001b[39m(\u001b[39m\"\u001b[39m\u001b[39mtime for calculate_min_distances3: \u001b[39m\u001b[39m\"\u001b[39m, time\u001b[39m.\u001b[39mtime() \u001b[39m-\u001b[39m start_time)\n",
      "File \u001b[1;32mc:\\Users\\au616584\\Anaconda3\\envs\\subspace\\lib\\site-packages\\numba\\core\\dispatcher.py:468\u001b[0m, in \u001b[0;36m_DispatcherBase._compile_for_args\u001b[1;34m(self, *args, **kws)\u001b[0m\n\u001b[0;32m    464\u001b[0m         msg \u001b[39m=\u001b[39m (\u001b[39mf\u001b[39m\u001b[39m\"\u001b[39m\u001b[39m{\u001b[39;00m\u001b[39mstr\u001b[39m(e)\u001b[39m.\u001b[39mrstrip()\u001b[39m}\u001b[39;00m\u001b[39m \u001b[39m\u001b[39m\\n\u001b[39;00m\u001b[39m\\n\u001b[39;00m\u001b[39mThis error may have been caused \u001b[39m\u001b[39m\"\u001b[39m\n\u001b[0;32m    465\u001b[0m                \u001b[39mf\u001b[39m\u001b[39m\"\u001b[39m\u001b[39mby the following argument(s):\u001b[39m\u001b[39m\\n\u001b[39;00m\u001b[39m{\u001b[39;00margs_str\u001b[39m}\u001b[39;00m\u001b[39m\\n\u001b[39;00m\u001b[39m\"\u001b[39m)\n\u001b[0;32m    466\u001b[0m         e\u001b[39m.\u001b[39mpatch_message(msg)\n\u001b[1;32m--> 468\u001b[0m     error_rewrite(e, \u001b[39m'\u001b[39;49m\u001b[39mtyping\u001b[39;49m\u001b[39m'\u001b[39;49m)\n\u001b[0;32m    469\u001b[0m \u001b[39mexcept\u001b[39;00m errors\u001b[39m.\u001b[39mUnsupportedError \u001b[39mas\u001b[39;00m e:\n\u001b[0;32m    470\u001b[0m     \u001b[39m# Something unsupported is present in the user code, add help info\u001b[39;00m\n\u001b[0;32m    471\u001b[0m     error_rewrite(e, \u001b[39m'\u001b[39m\u001b[39munsupported_error\u001b[39m\u001b[39m'\u001b[39m)\n",
      "File \u001b[1;32mc:\\Users\\au616584\\Anaconda3\\envs\\subspace\\lib\\site-packages\\numba\\core\\dispatcher.py:409\u001b[0m, in \u001b[0;36m_DispatcherBase._compile_for_args.<locals>.error_rewrite\u001b[1;34m(e, issue_type)\u001b[0m\n\u001b[0;32m    407\u001b[0m     \u001b[39mraise\u001b[39;00m e\n\u001b[0;32m    408\u001b[0m \u001b[39melse\u001b[39;00m:\n\u001b[1;32m--> 409\u001b[0m     \u001b[39mraise\u001b[39;00m e\u001b[39m.\u001b[39mwith_traceback(\u001b[39mNone\u001b[39;00m)\n",
      "\u001b[1;31mTypingError\u001b[0m: Failed in nopython mode pipeline (step: nopython frontend)\n\u001b[1m\u001b[1m\u001b[1mNo implementation of function Function(<function norm at 0x0000015697E72C20>) found for signature:\n \n >>> norm(array(float64, 2d, C), axis=Literal[int](1))\n \nThere are 2 candidate implementations:\n\u001b[1m  - Of which 2 did not match due to:\n  Overload in function 'norm_impl': File: numba\\np\\linalg.py: Line 2351.\n    With argument(s): '(array(float64, 2d, C), axis=int64)':\u001b[0m\n\u001b[1m   Rejected as the implementation raised a specific error:\n     TypingError: \u001b[1mgot an unexpected keyword argument 'axis'\u001b[0m\u001b[0m\n  raised from c:\\Users\\au616584\\Anaconda3\\envs\\subspace\\lib\\site-packages\\numba\\core\\typing\\templates.py:791\n\u001b[0m\n\u001b[0m\u001b[1mDuring: resolving callee type: Function(<function norm at 0x0000015697E72C20>)\u001b[0m\n\u001b[0m\u001b[1mDuring: typing of call at c:\\Users\\au616584\\OneDrive - Aarhus Universitet\\Datalogi\\PhD\\Implementation_subspace_clustering\\util.py (85)\n\u001b[0m\n\u001b[1m\nFile \"util.py\", line 85:\u001b[0m\n\u001b[1mdef calculate_min_distances1(data, centers):\n    <source elided>\n    for i in range(data.shape[0]):\n\u001b[1m        min_distances[i] = np.min(np.linalg.norm(data[i] - centers, axis=1))\n\u001b[0m        \u001b[1m^\u001b[0m\u001b[0m\n"
     ]
    }
   ],
   "source": [
    "start_time = time.time()\n",
    "util.calculate_min_distances1(data[1][0], opt_centers_50)\n",
    "print(\"time for calculate_min_distances3: \", time.time() - start_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for total_cost_kmeans2:  15.12880825996399\n",
      "time for total_cost_kmeans2:  4.66021466255188\n",
      "1442.9618278714925 1442.9618278714925\n"
     ]
    }
   ],
   "source": [
    "def total_cost_kmeans1(data, centers):\n",
    "    print(\"data shape: \", data.shape[0])   \n",
    "    min_distances = np.zeros(data.shape[0])\n",
    "    for i,point in enumerate(data):\n",
    "        min_distances[i] = np.min(np.linalg.norm(data[i] - centers, axis=1))\n",
    "    return np.sum(min_distances**2)\n",
    "\n",
    "def total_cost_kmeans2(data, centers):\n",
    "    cost = 0\n",
    "    for i in range(data.shape[0]):\n",
    "        cost += np.min(np.linalg.norm(data[i] - centers, axis=1))**2\n",
    "    return cost\n",
    "\n",
    "@jit(nopython=True)\n",
    "def total_cost_kmeans3(data, centers):\n",
    "    cost = 0\n",
    "    for i in range(data.shape[0]):\n",
    "        diff = data[i]-centers\n",
    "        cost += np.sqrt(np.min(np.sum(diff*diff, axis=1)))**2\n",
    "    return cost\n",
    "\n",
    "\n",
    "\n",
    "# start_time = time.time()\n",
    "# cost1 = total_cost_kmeans1(data[3][0], opt_centers_50)\n",
    "# print(\"time for total_cost_kmeans1: \", time.time() - start_time)\n",
    "start_time = time.time()\n",
    "cost2 = total_cost_kmeans2(data[3][0], opt_centers_50)\n",
    "print(\"time for total_cost_kmeans2: \", time.time() - start_time)\n",
    "start_time = time.time()\n",
    "cost3 = total_cost_kmeans3(data[3][0], opt_centers_50)\n",
    "print(\"time for total_cost_kmeans2: \", time.time() - start_time)\n",
    "print(cost2, cost3)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "time for total_cost_kmeans2:  11.76407766342163\n",
      "[30. 30.  4. ... 20. 20. 20.]\n"
     ]
    }
   ],
   "source": [
    "# @jit(nopython=True)\n",
    "def reassign_points(data, centers):\n",
    "    clusters = np.zeros(data.shape[0])  \n",
    "    for i in range(data.shape[0]):\n",
    "        diff = data[i]-centers\n",
    "        clusters[i] = np.argmin(np.sum(diff*diff, axis=1))\n",
    "    return clusters\n",
    "\n",
    "start_time = time.time()\n",
    "cluster = reassign_points(data[3][0], opt_centers_50)\n",
    "print(\"time for total_cost_kmeans2: \", time.time() - start_time)\n",
    "print(cluster)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "fitting kmedian\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x16f059f8c70>"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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NWPGDGVnZ9s9FNtwDepxHbgGPlwHroHENfBJ1nVwfGIvAABomZiJAQAAjkQQAwAAHIkgBgAAOBJBDAAAcCSCmDDefPNNuVyugD8ffPBB0POGDx/e7Pgbb7wxiSO3Xo8ePZp9pvvuuy/kOYcOHdLEiRPVsWNHHXvssfr5z3+unTt3JmnE1tq0aZPGjRunkpIStW7dWr169dKMGTN0+PDhkOelw3fh4YcfVo8ePZSTk6PBgwdr5cqVIY9/9tlndeKJJyonJ0ennHKKXnnllSSNNDHKysp02mmnKTc3VwUFBRo7dqzWr18f8pz58+c3+7vn5OQkacTWu+uuu5p9nhNPPDHkOen2PZAC/zvocrk0ceLEgMenw/fg7bff1oUXXqguXbrI5XLphRde8HveGKM777xTRUVFat26tUaOHKkvvvgi7OtG++9KIAQxYQwdOlRVVVV+P7/+9a9VUlKigQMHhjx3/Pjxfufdf//9SRp14tx9991+n+mWW24Jefytt96qRYsW6dlnn9Vbb72lHTt26JJLLknSaK21bt06eb1ePfroo1qzZo0efPBBPfLII/rd734X9lwnfxf+8Y9/aMqUKZoxY4YqKirUt29fjRo1Srt27Qp4/IoVK3TllVdq3Lhx+uijjzR27FiNHTtWlZWVSR65dd566y1NnDhR7733npYsWaIjR47o/PPP18GDB0Oel5eX5/d337x5c5JGnBgnn3yy3+dZvnx50GPT8XsgSR988IHfNViyZIkk6dJLLw16jtO/BwcPHlTfvn318MMPB3z+/vvv1//+7//qkUce0fvvv6+2bdtq1KhROnToUNDXjPbflaAMonL48GGTn59v7r777pDHnX322WbSpEnJGVSSdO/e3Tz44IMRH79v3z6TlZVlnn322YbHPvvsMyPJlJeXJ2CEyXf//febkpKSkMc4/bswaNAgM3HixIbf6+rqTJcuXUxZWVnA4y+77DIzZswYv8cGDx5s/vu//zuh40ymXbt2GUnmrbfeCnrMvHnzjNvtTt6gEmzGjBmmb9++ER/fEr4HxhgzadIk06tXL+P1egM+n27fA0nm3//+d8PvXq/XFBYWmtmzZzc8tm/fPpOdnW2efvrpoK8T7b8rwTATE6UXX3xRe/bs0fXXXx/22CeffFKdOnVSnz59NH36dH377bdJGGFi3XffferYsaP69++v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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import k_median_alg\n",
    "import sklearn.datasets\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "data = sklearn.datasets.make_blobs(n_samples=100, n_features=2, centers=3)\n",
    "alg = k_median_alg.K_median_alg(4, 50,1)\n",
    "alg.fit(data[0])\n",
    "\"\"\"plot data and centers\"\"\"\n",
    "plt.scatter(data[0][:,0], data[0][:,1])\n",
    "plt.scatter(alg.centers[:,0], alg.centers[:,1], c='r')\n",
    "\n",
    "\n",
    "\n",
    "\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "subspace",
   "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.8 | packaged by conda-forge | (main, Nov 24 2022, 14:07:00) [MSC v.1916 64 bit (AMD64)]"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "3f94b3ea8b08becd1766c9f3a63569b610389292faace546d6478dffb17875b9"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
