{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.cluster import KMeans, MiniBatchKMeans\n",
    "import numpy as np\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.datasets import fetch_covtype\n",
    "\"\"\"import covtype dataset from sklearn.datasets as np.array\"\"\"\n",
    "def get_covtype():\n",
    "    covtype = fetch_covtype()\n",
    "    return covtype.data, covtype.target\n",
    "data, labels = get_covtype()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"run kmeans on covtype dataset\"\"\"\n",
    "def run_kmeans(data, k, max_iter):\n",
    "    kmeans = KMeans(n_clusters=k, max_iter=max_iter, n_init=10).fit(data)\n",
    "    return kmeans\n",
    "kmeans_res = run_kmeans(data, 10, 100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_batch_kmeans(data, k, max_iter):\n",
    "    kmeans = MiniBatchKMeans(n_clusters=k, max_iter=max_iter, n_init=10).fit(data)\n",
    "    return kmeans\n",
    "kmeans_batch = run_batch_kmeans(data, 10, 100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_experiment_kmeans(data, k_values, subset_sizes, max_iter, n_runs, n_init=10, batch=False):\n",
    "    res_list_subsetcost = []\n",
    "    res_list_originalcost = []\n",
    "    original_opt = []\n",
    "    for k in k_values:\n",
    "        original_opt_kmeans = KMeans(n_clusters=k, max_iter=max_iter, n_init=n_init).fit(data)\n",
    "        original_opt.append((k, original_opt_kmeans.inertia_))\n",
    "        for subset_size in subset_sizes:\n",
    "            for run in range(n_runs):\n",
    "                print(\"new run: k={}, subset_size={}, run={}\".format(k, subset_size, run))\n",
    "                subset = data[np.random.choice(data.shape[0], subset_size, replace=False), :]\n",
    "                if batch:\n",
    "                    kmeans = MiniBatchKMeans(n_clusters=k, max_iter=max_iter, n_init=n_init).fit(subset)\n",
    "                else:\n",
    "                    kmeans = KMeans(n_clusters=k, max_iter=max_iter, n_init=n_init).fit(subset)\n",
    "                res_list_subsetcost.append((k, subset_size, run, kmeans.inertia_))\n",
    "                cost_on_original = kmeans.score(data)\n",
    "                res_list_originalcost.append((k, subset_size, run, cost_on_original))\n",
    "    return res_list_subsetcost, res_list_originalcost, original_opt\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "19\n",
      "[64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384] 581012\n",
      "new run: k=5, subset_size=64, run=0\n",
      "new run: k=5, subset_size=64, run=1\n",
      "new run: k=5, subset_size=64, run=2\n",
      "new run: k=5, subset_size=64, run=3\n",
      "new run: k=5, subset_size=64, run=4\n",
      "new run: k=5, subset_size=64, run=5\n",
      "new run: k=5, subset_size=64, run=6\n",
      "new run: k=5, subset_size=64, run=7\n",
      "new run: k=5, subset_size=64, run=8\n",
      "new run: k=5, subset_size=64, run=9\n",
      "new run: k=5, subset_size=128, run=0\n",
      "new run: k=5, subset_size=128, run=1\n",
      "new run: k=5, subset_size=128, run=2\n",
      "new run: k=5, subset_size=128, run=3\n",
      "new run: k=5, subset_size=128, run=4\n",
      "new run: k=5, subset_size=128, run=5\n",
      "new run: k=5, subset_size=128, run=6\n",
      "new run: k=5, subset_size=128, run=7\n",
      "new run: k=5, subset_size=128, run=8\n",
      "new run: k=5, subset_size=128, run=9\n",
      "new run: k=5, subset_size=256, run=0\n",
      "new run: k=5, subset_size=256, run=1\n",
      "new run: k=5, subset_size=256, run=2\n",
      "new run: k=5, subset_size=256, run=3\n",
      "new run: k=5, subset_size=256, run=4\n",
      "new run: k=5, subset_size=256, run=5\n",
      "new run: k=5, subset_size=256, run=6\n",
      "new run: k=5, subset_size=256, run=7\n",
      "new run: k=5, subset_size=256, run=8\n",
      "new run: k=5, subset_size=256, run=9\n",
      "new run: k=5, subset_size=512, run=0\n",
      "new run: k=5, subset_size=512, run=1\n",
      "new run: k=5, subset_size=512, run=2\n",
      "new run: k=5, subset_size=512, run=3\n",
      "new run: k=5, subset_size=512, run=4\n",
      "new run: k=5, subset_size=512, run=5\n",
      "new run: k=5, subset_size=512, run=6\n",
      "new run: k=5, subset_size=512, run=7\n",
      "new run: k=5, subset_size=512, run=8\n",
      "new run: k=5, subset_size=512, run=9\n",
      "new run: k=5, subset_size=1024, run=0\n",
      "new run: k=5, subset_size=1024, run=1\n",
      "new run: k=5, subset_size=1024, run=2\n",
      "new run: k=5, subset_size=1024, run=3\n",
      "new run: k=5, subset_size=1024, run=4\n",
      "new run: k=5, subset_size=1024, run=5\n",
      "new run: k=5, subset_size=1024, run=6\n",
      "new run: k=5, subset_size=1024, run=7\n",
      "new run: k=5, subset_size=1024, run=8\n",
      "new run: k=5, subset_size=1024, run=9\n",
      "new run: k=5, subset_size=2048, run=0\n",
      "new run: k=5, subset_size=2048, run=1\n",
      "new run: k=5, subset_size=2048, run=2\n",
      "new run: k=5, subset_size=2048, run=3\n",
      "new run: k=5, subset_size=2048, run=4\n",
      "new run: k=5, subset_size=2048, run=5\n",
      "new run: k=5, subset_size=2048, run=6\n",
      "new run: k=5, subset_size=2048, run=7\n",
      "new run: k=5, subset_size=2048, run=8\n",
      "new run: k=5, subset_size=2048, run=9\n",
      "new run: k=5, subset_size=4096, run=0\n",
      "new run: k=5, subset_size=4096, run=1\n",
      "new run: k=5, subset_size=4096, run=2\n",
      "new run: k=5, subset_size=4096, run=3\n",
      "new run: k=5, subset_size=4096, run=4\n",
      "new run: k=5, subset_size=4096, run=5\n",
      "new run: k=5, subset_size=4096, run=6\n",
      "new run: k=5, subset_size=4096, run=7\n",
      "new run: k=5, subset_size=4096, run=8\n",
      "new run: k=5, subset_size=4096, run=9\n",
      "new run: k=5, subset_size=8192, run=0\n",
      "new run: k=5, subset_size=8192, run=1\n",
      "new run: k=5, subset_size=8192, run=2\n",
      "new run: k=5, subset_size=8192, run=3\n",
      "new run: k=5, subset_size=8192, run=4\n",
      "new run: k=5, subset_size=8192, run=5\n",
      "new run: k=5, subset_size=8192, run=6\n",
      "new run: k=5, subset_size=8192, run=7\n",
      "new run: k=5, subset_size=8192, run=8\n",
      "new run: k=5, subset_size=8192, run=9\n",
      "new run: k=5, subset_size=16384, run=0\n",
      "new run: k=5, subset_size=16384, run=1\n",
      "new run: k=5, subset_size=16384, run=2\n",
      "new run: k=5, subset_size=16384, run=3\n",
      "new run: k=5, subset_size=16384, run=4\n",
      "new run: k=5, subset_size=16384, run=5\n",
      "new run: k=5, subset_size=16384, run=6\n",
      "new run: k=5, subset_size=16384, run=7\n",
      "new run: k=5, subset_size=16384, run=8\n",
      "new run: k=5, subset_size=16384, run=9\n"
     ]
    }
   ],
   "source": [
    "k_values = [5,10,20]\n",
    "subset_sizes = []\n",
    "print(int(np.log2(data.shape[0])))\n",
    "for i in range(6,int(np.log2(data.shape[0]))-4):\n",
    "    subset_sizes.append(2**i)\n",
    "# subset_sizes.append(data.shape[0])\n",
    "print(subset_sizes, data.shape[0])\n",
    "res_list_subsetcost, res_list_originalcost, original_opt = run_experiment_kmeans(data, k_values, subset_sizes , 100, 10, n_init=10, batch=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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F2vWrFk6fPiw7r//fh0/flyjRo1S69attXnzZr399ttavXq1MjIyPOquWbNGOTk5WrVqlVasWHHe/8bbtm3T8OHD9dOf/lQvvfSSbDab8vLy1LJly1ofv/71rz32s3fvXiUmJqpbt25KT09XXl7eed+7vrht5CVrtBGxD4APTlQ41e/RfwXlvXfNG6UW4XX7s3/zzTd7PP/zn/+s+Ph47dq1S/3799f48eN177336uOPP7bCytKlSzVx4kTrJLho0SLl5eUpMTFRknT//fcrMzNTixYtsk6AFRUV+v3vf6+BAwda7/XDH/7Q471feeUVxcbG6t///reuv/56rVq1Svv379e6deuspWWefPJJXXvttVadN998Uy6XS6+99pp1pXzRokWKjY3VunXrdN1111X5zAsXLlRMTIyWLVtmzUHSq1cv6/UFCxZo9uzZmjBhgiTp6aef1ocffqjnn39eCxcuVF5ennr27Knhw4fLZrN53PZxz04fGxtrtbk6CQkJCg0NVcuWLa1yr776qsrKyvT6668rKipKkvTSSy/phhtu0NNPP20taBwVFaXXXnutTreLPvnkE11//fV66KGHNGvWLGt7YmKiNalsTeLi4qzfU1JStHjxYvXu3VuHDx/WE088oREjRmjHjh1q1arVedvhK8KLl1hVGkBzsHfvXj366KPauHGjvv32W+uKS15envr376/4+Hhdd911WrJkiUaMGKHc3FxlZWXpj3/8o6RTtzmcTqfHyV86dSupTZs21vPw8HANGDDAo8yRI0f08MMPa926dTp69KicTqdKS0ut/6PPyclRUlKSRwi47LLLPPbxxRdfaN++fVVOoGVlZR63v862detWjRgxotrJ04qLi3Xo0CFdeeWVHtuvvPJKffHFF5JO3bq59tpr1bt3b6Wlpen666+vNiR5a/fu3Ro4cKAVXNzv63K5lJOTY4WX5OTkOgWXvLw8XXvttXryySd17733erwWGhqqHj161Llto0ePtn4fMGCAUlJS1LlzZ7311luaOnVqnffjLcKLl1wsDwCgHiLDQrRr3qigvXdd3XDDDercubNeffVVJSYmyuVyqX///h63XNLT0zVjxgy9+OKLWrp0qZKTk5WcnCxJKikpUUhIiLKzsxUS4vm+7iViJCkyMrJKH8LJkyfru+++0+9+9zt17txZDodDw4YN82pV7pKSEg0ZMkRLliyp8lpNa/RFRkbWef/VGTx4sHJzc/XBBx9o9erVGj9+vFJTU/X3v/+9Xvutq7PDTW3i4+OVmJioN954Q7fffruio6Ot1/Ly8tSvX79a6z/44IN68MEHq30tNjZWvXr10r59++recB8QXrzkcjHaCIDvbDZbnW/dBMt3332nnJwcvfrqq9YtoY8//rhKuRtvvFHTp09XZmamli5dqkmTJlmvDRo0SE6nU0ePHrX2UVcbNmzQ73//e40ZM0bSqY633377rfV67969deDAAR05csS66nBuH5LBgwfrzTffVLt27TxOzrUZMGCA/vKXv6iioqLK1Zfo6GglJiZqw4YNuvrqqz3aevZVn+joaN1666269dZb9ZOf/ERpaWkqKChQXFycwsLCrOUivNG3b18tXrxYx48ftwLKhg0bZLfb69wx92yRkZFasWKFxowZo1GjRun//u//rCtU3t42OldJSYn279+v2267zet2eYOeG15ihl0ATV3r1q3Vpk0bvfLKK9q3b5/Wrl2rmTNnVikXFRWlcePG6ZFHHtHu3bs1ceJE67VevXopPT1dkyZN0jvvvKPc3Fxt2rRJ8+fP1/vvv1/r+/fs2VN//etftXv3bm3cuFHp6ekeV0WuvfZade/eXZMnT9a2bdu0YcMGPfzww5LOjARNT09X27ZtdeONN+qjjz5Sbm6u1q1bpxkzZujrr7+u9n0zMjJUXFysCRMmaMuWLdq7d6/++te/KicnR5L0wAMP6Omnn9abb76pnJwczZkzR1u3btU999wjSfrtb3+rN954Q3v27NF//vMfvf3220pISFBsbKykUyOO1qxZo/z8fH3//fd1/K9x6rNERERo8uTJ2rFjhz788EPdfffduu2226zw5q2oqCi9//77Cg0N1ejRo62RS+7bRrU9zg4v999/v/7973/ryy+/1CeffKKbbrpJISEhHt8FfyC8eKlfYrTu+kF3jUmuucMVADRmdrtdy5YtU3Z2tvr376/77rtPv/nNb6otm56eri+++EIjRoxQp06dPF5btGiRJk2apFmzZql3794aN26cNm/eXKXcuf70pz/p+++/1+DBg3XbbbdpxowZateunfV6SEiIli9frpKSEl166aW64447rNFG7oV8W7RoofXr16tTp0768Y9/rL59+2rq1KkqKyur8UpMmzZttHbtWpWUlOjqq6/WkCFD9Oqrr1pXYWbMmKGZM2dq1qxZSk5OVmZmpt577z317NlTktSqVSs988wzGjp0qC699FJ9+eWXWrlypTVvzbPPPqtVq1YpKSnJWqy4Llq0aKF//etfKigo0KWXXqqf/OQnGjlypF566aU676M6LVu21AcffCBjjMaOHavjx497vY+vv/5aEydOVO/evTV+/Hi1adNGn376aY235hqKzfg6sPsCVVxcrJiYGBUVFdX5UiEA+EtZWZlyc3PVtWtX68SKhrdhwwYNHz5c+/btU/fu3YPdHNSgtn8P3py/L+wbrwAAVOPdd99Vy5Yt1bNnT+3bt0/33HOPrrzySoJLM0F4AQA0OseOHdPs2bOVl5entm3bKjU1Vc8++2ywm4UAIbwAABqdSZMmeYxuQvNCh10AANCoEF4AAECjQngBgABoYgM7AZ801L8D+rwAgB+FhYXJZrPpm2++UXx8fJWp8IHmwhijb775Rjabrdr1o7xBeAEAPwoJCVHHjh319ddf68svvwx2c4Cgstls6tixY5X1rrxFeAEAP3PPR1JRURHspgBBFRYWVu/gIhFeACAgQkJCGuSPNgA67AIAgEaG8AIAABoVwgsAAGhUmlyfF/cY8uLi4iC3BAAA1JX7vF2XuWCaXHg5duyYJCkpKSnILQEAAN46duyYYmJiai1jM01s2keXy6VDhw6pVatWDT4ZVHFxsZKSknTgwAFFR0c36L4bK46JJ45HVRyTqjgmVXFMqmpux8QYo2PHjikxMVF2e+29WprclRe73a6OHTv69T2io6ObxRfJGxwTTxyPqjgmVXFMquKYVNWcjsn5rri40WEXAAA0KoQXAADQqBBevOBwOPTYY4/J4XAEuykXDI6JJ45HVRyTqjgmVXFMquKY1KzJddgFAABNG1deAABAo0J4AQAAjQrhBQAANCqEFwAA0KgQXupo4cKF6tKliyIiIpSSkqJNmzYFu0kNYv78+br00kvVqlUrtWvXTuPGjVNOTo5HmWuuuUY2m83j8fOf/9yjTF5ensaOHasWLVqoXbt2euCBB1RZWelRZt26dRo8eLAcDod69OihxYsX+/vj+eTxxx+v8nn79OljvV5WVqa77rpLbdq0UcuWLXXzzTfryJEjHvtoSsdDkrp06VLlmNhsNt11112Smsd3ZP369brhhhuUmJgom82m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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\"\"\"average cost for each k and subset size\"\"\"\n",
    "def get_average_cost(res_list_subsetcost, num_runs):\n",
    "    avg_cost_subset = {}\n",
    "    for k, subset_size, run, cost in res_list_subsetcost:\n",
    "        if (k, subset_size) not in avg_cost_subset:\n",
    "            avg_cost_subset[(k, subset_size)] = 0\n",
    "        avg_cost_subset[(k, subset_size)] += cost/subset_size\n",
    "    for key in avg_cost_subset:\n",
    "        avg_cost_subset[key] = avg_cost_subset[key]/num_runs\n",
    "    return avg_cost_subset\n",
    "\n",
    "\"\"\"plot average cost and original cost for each subset size and fixed k\"\"\"\n",
    "def plot_average_cost(average_costs, original_opt, subset_sizes, k):\n",
    "    import matplotlib.pyplot as plt\n",
    "    x = []\n",
    "    y = []\n",
    "    for subset_size in subset_sizes:\n",
    "        x.append(subset_size)\n",
    "        y.append(average_costs[(k, subset_size)])\n",
    "    plt.plot(x, y, label=\"average cost for k={}\".format(k))\n",
    "    plt.plot(original_opt[0][1]/data.shape[0])\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "average_costs = get_average_cost(res_list_originalcost, 10)\n",
    "plot_average_cost(average_costs, original_opt, subset_sizes, 5)\n",
    "    \n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "26.196249243850627\n",
      "<class 'int'>\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "num = np.log2(76887687)\n",
    "print(num)\n",
    "print(type(int(num)))\n",
    "    "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "py11",
   "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.11.0"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "9562cf3746306604c7816703dc302112a03af910ce9afbe2e8d4eefeff37f6c7"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
