{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "3.24"
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from collections import defaultdict\n",
    "from addition import get_mnist_add\n",
    "\n",
    "def count_y_to_z(Z, Y):\n",
    "    y_to_z_count = defaultdict(set)\n",
    "    for y, z in zip(Y, Z):\n",
    "        y_to_z_count[y].add(tuple(z))\n",
    "    y_to_z_count = {y:   len(z_set) for y, z_set in y_to_z_count.items()}\n",
    "    y_to_z_count_dict = defaultdict(list)\n",
    "    for y, z in zip(Y, Z):\n",
    "        y_to_z_count_dict[y].append(tuple(z))\n",
    "    return y_to_z_count, {y: list(set(z_list)) for y, z_list in y_to_z_count_dict.items()}\n",
    "\n",
    "X, Z, Y = get_mnist_add(get_pseudo_label=True, n=2, z_list=[0,1,2], sequence_num=10000)\n",
    "res = count_y_to_z(Z, Y)\n",
    "sum(res[0].values()) / len(res[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 390625/390625 [15:24:27<00:00,  7.04it/s]t/s]\n",
      "  4%|▍         | 80538/1953125 [15:24:34<345:31:00,  1.51it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5 8 3544.704151217464\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  6%|▌         | 113084/1953125 [22:06:50<394:23:52,  1.30it/s]"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Input \u001b[0;32mIn [3]\u001b[0m, in \u001b[0;36m<cell line: 39>\u001b[0;34m()\u001b[0m\n\u001b[1;32m     36\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m count_sum_combinations_dp(z_list, n)\n\u001b[1;32m     38\u001b[0m \u001b[38;5;66;03m# Use Parallel to run the function in parallel\u001b[39;00m\n\u001b[0;32m---> 39\u001b[0m res \u001b[38;5;241m=\u001b[39m \u001b[43mParallel\u001b[49m\u001b[43m(\u001b[49m\u001b[43mn_jobs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m-\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdelayed\u001b[49m\u001b[43m(\u001b[49m\u001b[43mparallel_count_sum_combinations_dp\u001b[49m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mlist\u001b[39;49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mrange\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mi\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mj\u001b[49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mi\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[38;5;28;43mrange\u001b[39;49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m5\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m6\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mj\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[38;5;28;43mrange\u001b[39;49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m8\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m10\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/abl/lib/python3.8/site-packages/joblib/parallel.py:1056\u001b[0m, in \u001b[0;36mParallel.__call__\u001b[0;34m(self, iterable)\u001b[0m\n\u001b[1;32m   1053\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_iterating \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[1;32m   1055\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backend\u001b[38;5;241m.\u001b[39mretrieval_context():\n\u001b[0;32m-> 1056\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mretrieve\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1057\u001b[0m \u001b[38;5;66;03m# Make sure that we get a last message telling us we are done\u001b[39;00m\n\u001b[1;32m   1058\u001b[0m elapsed_time \u001b[38;5;241m=\u001b[39m time\u001b[38;5;241m.\u001b[39mtime() \u001b[38;5;241m-\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_start_time\n",
      "File \u001b[0;32m~/anaconda3/envs/abl/lib/python3.8/site-packages/joblib/parallel.py:935\u001b[0m, in \u001b[0;36mParallel.retrieve\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    933\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m    934\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mgetattr\u001b[39m(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backend, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124msupports_timeout\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;28;01mFalse\u001b[39;00m):\n\u001b[0;32m--> 935\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_output\u001b[38;5;241m.\u001b[39mextend(\u001b[43mjob\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mget\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m)\n\u001b[1;32m    936\u001b[0m     \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    937\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_output\u001b[38;5;241m.\u001b[39mextend(job\u001b[38;5;241m.\u001b[39mget())\n",
      "File \u001b[0;32m~/anaconda3/envs/abl/lib/python3.8/site-packages/joblib/_parallel_backends.py:542\u001b[0m, in \u001b[0;36mLokyBackend.wrap_future_result\u001b[0;34m(future, timeout)\u001b[0m\n\u001b[1;32m    539\u001b[0m \u001b[38;5;124;03m\"\"\"Wrapper for Future.result to implement the same behaviour as\u001b[39;00m\n\u001b[1;32m    540\u001b[0m \u001b[38;5;124;03mAsyncResults.get from multiprocessing.\"\"\"\u001b[39;00m\n\u001b[1;32m    541\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m--> 542\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mfuture\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mresult\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    543\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m CfTimeoutError \u001b[38;5;28;01mas\u001b[39;00m e:\n\u001b[1;32m    544\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mTimeoutError\u001b[39;00m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01me\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/abl/lib/python3.8/concurrent/futures/_base.py:439\u001b[0m, in \u001b[0;36mFuture.result\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m    436\u001b[0m \u001b[38;5;28;01melif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_state \u001b[38;5;241m==\u001b[39m FINISHED:\n\u001b[1;32m    437\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m__get_result()\n\u001b[0;32m--> 439\u001b[0m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_condition\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mwait\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    441\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_state \u001b[38;5;129;01min\u001b[39;00m [CANCELLED, CANCELLED_AND_NOTIFIED]:\n\u001b[1;32m    442\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m CancelledError()\n",
      "File \u001b[0;32m~/anaconda3/envs/abl/lib/python3.8/threading.py:302\u001b[0m, in \u001b[0;36mCondition.wait\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m    300\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:    \u001b[38;5;66;03m# restore state no matter what (e.g., KeyboardInterrupt)\u001b[39;00m\n\u001b[1;32m    301\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m timeout \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[0;32m--> 302\u001b[0m         \u001b[43mwaiter\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43macquire\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    303\u001b[0m         gotit \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[1;32m    304\u001b[0m     \u001b[38;5;28;01melse\u001b[39;00m:\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "from collections import defaultdict\n",
    "from itertools import product\n",
    "from addition import digits_to_number\n",
    "from tqdm import tqdm\n",
    "from joblib import Parallel, delayed\n",
    "\n",
    "# Using dynamic programming to speed up the calculation\n",
    "def count_sum_combinations_dp(z_list, n):\n",
    "    dp = defaultdict(int)\n",
    "    \n",
    "    # Initialize dp for single digit numbers\n",
    "    for digit in z_list:\n",
    "        dp[digit] = 1\n",
    "    \n",
    "    # Build dp for n-digit numbers\n",
    "    for _ in range(n - 1):\n",
    "        new_dp = defaultdict(int)\n",
    "        for num, count in dp.items():\n",
    "            for digit in z_list:\n",
    "                new_dp[num * 10 + digit] += count\n",
    "        dp = new_dp\n",
    "    \n",
    "    # Calculate the sum for each pair of numbers\n",
    "    sum_combinations = defaultdict(int)\n",
    "    for num1, count1 in tqdm(dp.items()):\n",
    "        for num2, count2 in dp.items():\n",
    "            sum_value = num1 + num2\n",
    "            sum_combinations[sum_value] += count1 * count2\n",
    "    \n",
    "    # Count the number of unique combinations for each sum\n",
    "    sum_count = {sum_value: count for sum_value, count in sum_combinations.items()}\n",
    "    print(len(z_list), n, sum(sum_count.values()) / len(sum_count))\n",
    "    return sum(sum_count.values()) / len(sum_count)\n",
    "\n",
    "def parallel_count_sum_combinations_dp(z_list, n):\n",
    "    return count_sum_combinations_dp(z_list, n)\n",
    "\n",
    "# Use Parallel to run the function in parallel\n",
    "res = Parallel(n_jobs=-1)(delayed(parallel_count_sum_combinations_dp)(list(range(i)), j) for i in range(5, 6) for j in range(8, 10))\n",
    "# res = [[count_sum_combinations_dp(list(range(i)), j) for i in range(5,6)] for j in range(8,10)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "390625it [00:00, 742936.80it/s]\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "from itertools import product\n",
    "from collections import defaultdict\n",
    "from tqdm import tqdm\n",
    "\n",
    "def digits_to_number(digits):\n",
    "    return int(''.join(map(str, digits)))\n",
    "\n",
    "def count_sum_combinations(z_list, n):\n",
    "    # 生成所有可能的 n 位数\n",
    "    numbers = [digits_to_number(num) for num in tqdm(product(z_list, repeat=n))]\n",
    "    \n",
    "    if not numbers:\n",
    "        return 0\n",
    "    \n",
    "    max_num = max(numbers)\n",
    "    \n",
    "    # 构建频率向量\n",
    "    a = np.zeros(max_num + 1, dtype=np.int64)\n",
    "    a[numbers] = 1\n",
    "    \n",
    "    # 进行卷积运算\n",
    "    c = np.convolve(a, a)\n",
    "    \n",
    "    # 计算所有和的频率\n",
    "    sum_count = c[:2 * max_num + 1]\n",
    "    \n",
    "    # 计算平均值\n",
    "    average = sum_count.sum() / len(sum_count)\n",
    "    \n",
    "    print(len(z_list), n, average)\n",
    "    return average\n",
    "\n",
    "# 生成 res 矩阵\n",
    "res = [[count_sum_combinations(list(range(i)), j) for i in range(5,6)] for j in range(8,10)]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0],\n",
       " [1.0,\n",
       "  1.3333333333333333,\n",
       "  1.8,\n",
       "  2.2857142857142856,\n",
       "  2.7777777777777777,\n",
       "  3.272727272727273,\n",
       "  3.769230769230769,\n",
       "  4.266666666666667,\n",
       "  4.764705882352941,\n",
       "  5.2631578947368425],\n",
       " [1.0,\n",
       "  1.7777777777777777,\n",
       "  3.24,\n",
       "  5.224489795918367,\n",
       "  7.716049382716049,\n",
       "  11.675675675675675,\n",
       "  18.05263157894737,\n",
       "  26.425806451612903,\n",
       "  37.067796610169495,\n",
       "  50.25125628140704],\n",
       " [1.0,\n",
       "  2.3703703703703702,\n",
       "  5.832,\n",
       "  11.941690962099125,\n",
       "  21.43347050754458,\n",
       "  41.994599459945995,\n",
       "  88.25881470367592,\n",
       "  168.58135048231512,\n",
       "  299.0664040517727,\n",
       "  500.25012506253125],\n",
       " [1.0,\n",
       "  3.1604938271604937,\n",
       "  10.4976,\n",
       "  27.295293627655145,\n",
       "  59.53741807651273,\n",
       "  151.1669516695167,\n",
       "  432.3708842721068,\n",
       "  1078.5738347798135,\n",
       "  2421.4839961748326,\n",
       "  5000.250012500625],\n",
       " [1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0],\n",
       " [1.0,\n",
       "  1.3333333333333333,\n",
       "  1.8,\n",
       "  2.2857142857142856,\n",
       "  2.7777777777777777,\n",
       "  3.272727272727273,\n",
       "  3.769230769230769,\n",
       "  4.266666666666667,\n",
       "  4.764705882352941,\n",
       "  5.2631578947368425],\n",
       " [1.0,\n",
       "  1.7777777777777777,\n",
       "  3.24,\n",
       "  5.224489795918367,\n",
       "  7.716049382716049,\n",
       "  11.675675675675675,\n",
       "  18.05263157894737,\n",
       "  26.425806451612903,\n",
       "  37.067796610169495,\n",
       "  50.25125628140704],\n",
       " [1.0,\n",
       "  2.3703703703703702,\n",
       "  5.832,\n",
       "  11.941690962099125,\n",
       "  21.43347050754458,\n",
       "  41.994599459945995,\n",
       "  88.25881470367592,\n",
       "  168.58135048231512,\n",
       "  299.0664040517727,\n",
       "  500.25012506253125],\n",
       " [1.0,\n",
       "  3.1604938271604937,\n",
       "  10.4976,\n",
       "  27.295293627655145,\n",
       "  59.53741807651273,\n",
       "  151.1669516695167,\n",
       "  432.3708842721068,\n",
       "  1078.5738347798135,\n",
       "  2421.4839961748326,\n",
       "  5000.250012500625]]"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "res + res"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5000.250012500625\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "17.296384270256006"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from collections import defaultdict\n",
    "from itertools import product\n",
    "\n",
    "def count_sum_combinations(z_list, n):\n",
    "    sum_combinations = defaultdict(set)\n",
    "    \n",
    "    # Generate all possible n-digit numbers from z_list\n",
    "    numbers = list(product(z_list, repeat=n))\n",
    "    \n",
    "    # Convert tuples to integers\n",
    "    numbers = [digits_to_number(num) for num in numbers]\n",
    "    \n",
    "    # Calculate the sum for each pair of numbers\n",
    "    for num1 in numbers:\n",
    "        for num2 in numbers:\n",
    "            sum_value = num1 + num2\n",
    "            sum_combinations[sum_value].add((num1, num2))\n",
    "    \n",
    "    # Count the number of unique combinations for each sum\n",
    "    sum_count = {sum_value: len(combinations) for sum_value, combinations in sum_combinations.items()}\n",
    "    \n",
    "    return sum_count\n",
    "\n",
    "# Example usage\n",
    "z_list = list(range(10))\n",
    "n = 4\n",
    "sum_count = count_sum_combinations(z_list, n)\n",
    "known_z_list = list(range(7))\n",
    "print(sum(sum_count.values()) / len(sum_count))\n",
    "def normalize_number(num, known_z_list):\n",
    "    return tuple(0 if digit in known_z_list else digit for digit in num)\n",
    "\n",
    "normalized_numbers = [normalize_number(num, known_z_list) for num in product(z_list, repeat=n)]\n",
    "normalized_numbers = [digits_to_number(num) for num in normalized_numbers]\n",
    "\n",
    "sum_combinations = defaultdict(set)\n",
    "\n",
    "for num1 in normalized_numbers:\n",
    "    for num2 in normalized_numbers:\n",
    "        sum_value = num1 + num2\n",
    "        sum_combinations[sum_value].add((num1, num2))\n",
    "\n",
    "sum_count2 = {sum_value: len(combinations) for sum_value, combinations in sum_combinations.items()}\n",
    "\n",
    "sum(sum_count2.values()) / len(sum_count2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import defaultdict\n",
    "from itertools import product\n",
    "from addition import digits_to_number\n",
    "\n",
    "def count_sum_combinations(z_list, n):\n",
    "    sum_combinations = defaultdict(set)\n",
    "    \n",
    "    # Generate all possible n-digit numbers from z_list\n",
    "    numbers = list(product(z_list, repeat=n))\n",
    "    \n",
    "    # Convert tuples to integers\n",
    "    numbers = [digits_to_number(num) for num in numbers]\n",
    "    \n",
    "    # Calculate the sum for each pair of numbers\n",
    "    for num1 in numbers:\n",
    "        for num2 in numbers:\n",
    "            sum_value = num1 + num2\n",
    "            sum_combinations[sum_value].add((num1, num2))\n",
    "    \n",
    "    # Count the number of unique combinations for each sum\n",
    "    sum_count = {sum_value: len(combinations) for sum_value, combinations in sum_combinations.items()}\n",
    "    \n",
    "    return sum_count\n",
    "\n",
    "# Example usage\n",
    "z_list = list(range(2))\n",
    "n = 4\n",
    "sum_count = count_sum_combinations(z_list, n)\n",
    "sum(sum_count.values()) / len(sum_count)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2.56e-06"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "def count_sublist(sub_z_list, z_list, n):\n",
    "    n1 = len(list(product(z_list, repeat=n)))\n",
    "    n2 = len(list(product(sub_z_list, repeat=n)))\n",
    "    return n2 / n1\n",
    "import math\n",
    "count_sublist(list(range(int(math.log(100)))), list(range(100)), 4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1854131/4134461403.py:9: RuntimeWarning: divide by zero encountered in divide\n",
      "  n_log_n_values = n_values / np.log2(n_values)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA1IAAAIjCAYAAAAJLyrXAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAABtmklEQVR4nO3deVxU9eLG8WeGHRFwY1NQFARxLU1zyw33TLP1tplZ3UoztdKsrLRFMyvbt1tZv/Ja3dLMLXHPMrcyRRT3XcCNXWCYOb8/vHIjlxwEzgCf9+vl6945czjzDHzTeTjnfL8WwzAMAQAAAAAumdXsAAAAAABQ0VCkAAAAAMBJFCkAAAAAcBJFCgAAAACcRJECAAAAACdRpAAAAADASRQpAAAAAHASRQoAAAAAnESRAgAAAAAnUaQAAOVixYoVslgsWrFiRZm/1sGDB+Xt7a2ff/7Z6a9NSkqSu7u7EhMTyyAZAKCyoEgBAMrMW2+9pYCAANlstnJ93UmTJqldu3bq2LGj018bFxen/v3765lnnimDZACAyoIiBQAoM/Pnz1evXr3k4eFRbq957NgxffbZZ3rggQdKfIwHHnhAs2fP1u7du0sxGQCgMqFIAQDKRG5urlauXKn+/fuX6+t+8cUXcnd314ABA0p8jPj4eNWoUUOfffZZKSYrGcMwdPr0abNjAAD+giIFALgkzz33nCwWi3bt2qW7775bgYGBCggI0NChQ5Wbm3vO/kuXLlV+fr769u170eN+8803at26tXx8fFS7dm3dcccdOnz48Hn3i4uLk7e3t5o1a6bZs2fr7rvvVoMGDYrtN2fOHLVr105+fn7Ftnft2lXNmjVTUlKSunXrJl9fX9WtW1dTp04957U8PDzUtWtXff/995fwnZEOHz6sYcOGKSwsTF5eXoqMjNSDDz6ogoICSf/73v3VjBkzZLFYtG/fvqJtDRo00LXXXqsff/xRbdq0kY+Pjz744AM1a9ZM3bp1O+cYDodDdevW1Y033lhs2/Tp09W0aVN5e3srODhY//znP3Xq1KlLej8AgL9HkQIAOOXmm29WVlaWJk+erJtvvlkzZszQxIkTz9lvwYIFat26tYKDgy94rBkzZujmm2+Wm5ubJk+erPvuu0/fffedOnXqpPT09KL95s+fr1tuuUUeHh6aPHmyBg8erGHDhmnjxo3Fjmez2bR+/XpdeeWV5329U6dOqU+fPmrZsqVeffVVxcbGaty4cVq4cOE5+7Zu3VqJiYnKzMy86PfjyJEjatu2rWbNmqVbbrlFb775pu68806tXLnyvAXzUiQnJ+sf//iHevbsqTfeeEOtWrXSLbfcolWrViklJaXYvqtXr9aRI0d06623Fm375z//qccff1wdO3bUG2+8oaFDh+rLL79U7969y/1+NQCotAwAAC7Bs88+a0gy7rnnnmLbr7/+eqNWrVrn7B8REWE8++yzRY+XL19uSDKWL19uGIZhFBQUGEFBQUazZs2M06dPF+03b948Q5LxzDPPFG1r3ry5Ua9ePSMrK6to24oVKwxJRv369Yu27dq1y5BkvPXWW+fk6dKliyHJ+Pzzz4u25efnGyEhIcYNN9xwzv4zZ840JBlr16698DfFMIy77rrLsFqtxvr16895zuFwGIbxv+/dX3366aeGJGPv3r1F2+rXr29IMhYtWlRs3+Tk5PO+t4ceesjw8/MzcnNzDcMwjJ9++smQZHz55ZfF9lu0aNF5twMASoYzUgAAp/x1EofOnTvrxIkTxc7cJCYm6sCBAxe9P2rDhg1KS0vTQw89JG9v76Lt/fv3V2xsrObPny/pzBmfLVu26K677ip2uV6XLl3UvHnzYsc8ceKEJKlGjRrnfU0/Pz/dcccdRY89PT3Vtm1b7dmz55x9zx7j+PHjF3wPDodDc+bM0YABA9SmTZtznj/f5XyXIjIyUr179y62rXHjxmrVqpW++uqrom12u13/+c9/NGDAAPn4+Eg6cwlkQECAevbsqePHjxf9ad26tfz8/LR8+fISZQIAFEeRAgA4JSIiotjjs4Xjz/ffzJ8/X8HBwectF2ft379fkhQTE3POc7GxsUXPn/3fqKioc/Y73zbpzAQN51OvXr1zyk2NGjXOe+/Q2WNcrAwdO3ZMmZmZatas2QX3KYnIyMjzbr/lllv0888/F91DtmLFCqWlpemWW24p2mfnzp3KyMhQUFCQ6tSpU+xPdna20tLSSjUrAFRV7mYHAABULG5ubufd/ufysmDBAvXp06fEZ2RKqlatWpJ0wUkVLiX7WWePUbt27cvOdaHvg91uP+/2s2eX/uqWW27R+PHj9c0332jUqFH6+uuvFRAQoD59+hTt43A4FBQUpC+//PK8x6hTp46T6QEA50ORAgCUqvT0dP3yyy8aMWLERferX7++pDMTK3Tv3r3Yc8nJyUXPn/3fXbt2nXOMv26LiIiQj4+P9u7dW+L8Z+3du1dWq1WNGze+4D516tSRv7+/EhMTL3qss2ft0tPTFRgYWLT97Nm2SxUZGam2bdvqq6++0ogRI/Tdd99p0KBB8vLyKtqnUaNGWrJkiTp27HjBQgYAuHxc2gcAKFWLFy+WJPXq1eui+7Vp00ZBQUF6//33lZ+fX7R94cKF2rZtW9H9VWFhYWrWrJk+//xzZWdnF+23cuVKbdmypdgxPTw81KZNG23YsOGy38fGjRvVtGlTBQQEXHAfq9WqQYMG6Ycffjjva54909WoUSNJ0qpVq4qey8nJKdE6Vbfccot+/fVXffLJJzp+/Hixy/qkM7Mq2u12Pf/88+d8bWFhYbHZEAEAJUeRAgCUqvnz56tTp04XLSDSmdLz8ssva/PmzerSpYveeOMNPfnkk7rxxhvVoEEDjR49umjfl156SYcPH1bHjh01ffp0Pfvssxo8eLCaNWt2zmVzAwcO1Lp16/522vKLsdlsWrlypQYOHPi3+7700ksKCgpSly5dNHr0aH344YeaOHGimjVrpoyMDElnSmVERISGDRumqVOn6tVXX1Xbtm1LdJndzTffLIvFoscee0w1a9ZUfHx8see7dOmif/7zn5o8ebL69eun6dOn65133tGoUaNUv359LVmyxOnXBACciyIFACg1hmFo0aJF6tev3yXtf/fdd+urr75SQUGBxo0bpw8++EDXX3+9Vq9eXewSuAEDBujf//63CgoK9MQTT+i7777TjBkzFBMTU2zGP0m68847ZbfbNXfu3BK/j6VLl+rkyZMaMmTI3+5bt25drV27VjfeeKO+/PJLjRw5Up9//rm6du0qX19fSWdK4+zZs9WoUSNNmDBBb775pu69996/vfzxfOrVq6cOHTooKytLgwcPloeHxzn7vP/++/rwww+VlpamJ598UuPHj9eyZct0xx13qGPHjk6/JgDgXBbjQlMbAQDgpHXr1qldu3baunWr4uLiyvz1WrVqpTp16ighIaHY9mHDhmnHjh366aefSnTcQYMGyWKxaPbs2aUREwBQCXFGCgBQql566aVSL1E2m02FhYXFtq1YsUJ//PGHunbtes7+zz77rNavX6+ff/7Z6dfatm2b5s2bd957jAAAOIszUgAAl7dv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",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# 定义 n 的范围\n",
    "n_values = np.arange(1, 1001)\n",
    "\n",
    "# 计算 n/log(n)\n",
    "n_log_n_values = n_values / np.log2(n_values)\n",
    "\n",
    "# 绘制曲线图\n",
    "plt.figure(figsize=(10, 6))\n",
    "plt.plot(n_values, n_log_n_values, label='n/log(n)')\n",
    "plt.xlabel('n')\n",
    "plt.ylabel('n/log(n)')\n",
    "plt.title('n/log(n) curve')\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Downloading zip archive\n",
      "https://biometrics.nist.gov/cs_links/EMNIST/gzip.zip\n"
     ]
    }
   ],
   "source": [
    "import torchvision\n",
    "from torchvision.datasets.utils import download_url\n",
    "import os\n",
    "\n",
    "raw_folder = 'data/EMNIST/raw'\n",
    "\n",
    "url = 'https://biometrics.nist.gov/cs_links/EMNIST/gzip.zip'\n",
    "md5 = \"58c8d27c78d21e728a6bc7b3cc06412e\"\n",
    "\n",
    "version_numbers = list(map(int, torchvision.__version__.split('+')[0].split('.')))\n",
    "if version_numbers[0] == 0 and version_numbers[1] < 10:\n",
    "    filename = \"emnist.zip\"\n",
    "else:\n",
    "    filename = None\n",
    "\n",
    "os.makedirs(raw_folder, exist_ok=True)\n",
    "\n",
    "# download files\n",
    "print('Downloading zip archive')\n",
    "download_url(url, root=raw_folder, filename=filename, md5=md5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [],
   "source": [
    "import itertools\n",
    "import json\n",
    "import random\n",
    "from pathlib import Path\n",
    "from typing import Callable, List, Iterable, Tuple\n",
    "\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "from torch.utils.data import Dataset as TorchDataset\n",
    "from collections import Counter, defaultdict\n",
    "from torch.utils.data import Subset\n",
    "\n",
    "_DATA_ROOT = './'\n",
    "\n",
    "transform = transforms.Compose(\n",
    "    [transforms.ToTensor(), transforms.Normalize((0.1307,), (0.3081,))]\n",
    ")\n",
    "\n",
    "target_labels = list(range(16))\n",
    "\n",
    "datasets_full = {\n",
    "    \"train\": torchvision.datasets.EMNIST(\n",
    "        root=_DATA_ROOT, split='bymerge', train=True, download=True, transform=transform\n",
    "    ),\n",
    "    \"test\": torchvision.datasets.EMNIST(\n",
    "        root=_DATA_ROOT, split='bymerge', train=False, download=True, transform=transform\n",
    "    ),\n",
    "}\n",
    "\n",
    "datasets = {\n",
    "    \"train\": Subset(datasets_full[\"train\"], [\n",
    "        i for i, (_, label) in enumerate(datasets_full[\"train\"]) if label in target_labels\n",
    "    ]),\n",
    "    \"test\": Subset(datasets_full[\"test\"], [\n",
    "        i for i, (_, label) in enumerate(datasets_full[\"test\"]) if label in target_labels\n",
    "    ])\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {},
   "outputs": [],
   "source": [
    "import itertools\n",
    "import json\n",
    "import random\n",
    "from pathlib import Path\n",
    "from typing import Callable, List, Iterable, Tuple\n",
    "\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "from torch.utils.data import Dataset as TorchDataset\n",
    "from collections import Counter, defaultdict\n",
    "from torch.utils.data import Subset\n",
    "\n",
    "_DATA_ROOT = './'\n",
    "\n",
    "transform = transforms.Compose(\n",
    "    [transforms.ToTensor(), transforms.Normalize((0.1307,), (0.3081,))]\n",
    ")\n",
    "\n",
    "# target_labels = list(range(16))\n",
    "\n",
    "datasets = {\n",
    "    \"train\": torchvision.datasets.MNIST(\n",
    "        root=_DATA_ROOT, train=True, download=True, transform=transform\n",
    "    ),\n",
    "    \"test\": torchvision.datasets.MNIST(\n",
    "        root=_DATA_ROOT, train=False, download=True, transform=transform\n",
    "    ),\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "60000"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset = datasets[\"train\"]\n",
    "len(dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.utils.data import Subset\n",
    "target_labels = list(range(16))\n",
    "indices = [i for i, (_, label) in enumerate(dataset) if label in target_labels]\n",
    "\n",
    "# 创建子集\n",
    "emnist_16 = Subset(dataset, indices)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "abl",
   "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.8.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
