{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.cm as cm\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "ename": "FileNotFoundError",
     "evalue": "[Errno 2] No such file or directory: '../4096/page_topk_accu_score.xlsx'",
     "output_type": "error",
     "traceback": [
      "\u001b[31m---------------------------------------------------------------------------\u001b[39m",
      "\u001b[31mFileNotFoundError\u001b[39m                         Traceback (most recent call last)",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[3]\u001b[39m\u001b[32m, line 21\u001b[39m\n\u001b[32m     19\u001b[39m \u001b[38;5;66;03m# ========== 1. 按组顺序画图 ==========\u001b[39;00m\n\u001b[32m     20\u001b[39m \u001b[38;5;28;01mfor\u001b[39;00m idx, file_key \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28menumerate\u001b[39m((ARK_NAME, GT_NAME)):\n\u001b[32m---> \u001b[39m\u001b[32m21\u001b[39m     df = \u001b[43mpd\u001b[49m\u001b[43m.\u001b[49m\u001b[43mread_excel\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfiles_to_process\u001b[49m\u001b[43m[\u001b[49m\u001b[43mfile_key\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mindex_col\u001b[49m\u001b[43m=\u001b[49m\u001b[32;43m0\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[32m     22\u001b[39m     df.index = df.index.str.replace(\u001b[33m'\u001b[39m\u001b[33mToken_\u001b[39m\u001b[33m'\u001b[39m, \u001b[33m'\u001b[39m\u001b[33m'\u001b[39m, regex=\u001b[38;5;28;01mFalse\u001b[39;00m).astype(\u001b[38;5;28mint\u001b[39m)\n\u001b[32m     24\u001b[39m     \u001b[38;5;66;03m# 计算每一行（每个 Token_Index）所有 Layer 的均值\u001b[39;00m\n",
      "\u001b[36mFile \u001b[39m\u001b[32m~/anaconda3/envs/Arkvale/lib/python3.13/site-packages/pandas/io/excel/_base.py:495\u001b[39m, in \u001b[36mread_excel\u001b[39m\u001b[34m(io, sheet_name, header, names, index_col, usecols, dtype, engine, converters, true_values, false_values, skiprows, nrows, na_values, keep_default_na, na_filter, verbose, parse_dates, date_parser, date_format, thousands, decimal, comment, skipfooter, storage_options, dtype_backend, engine_kwargs)\u001b[39m\n\u001b[32m    493\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(io, ExcelFile):\n\u001b[32m    494\u001b[39m     should_close = \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[32m--> \u001b[39m\u001b[32m495\u001b[39m     io = \u001b[43mExcelFile\u001b[49m\u001b[43m(\u001b[49m\n\u001b[32m    496\u001b[39m \u001b[43m        \u001b[49m\u001b[43mio\u001b[49m\u001b[43m,\u001b[49m\n\u001b[32m    497\u001b[39m \u001b[43m        \u001b[49m\u001b[43mstorage_options\u001b[49m\u001b[43m=\u001b[49m\u001b[43mstorage_options\u001b[49m\u001b[43m,\u001b[49m\n\u001b[32m    498\u001b[39m \u001b[43m        \u001b[49m\u001b[43mengine\u001b[49m\u001b[43m=\u001b[49m\u001b[43mengine\u001b[49m\u001b[43m,\u001b[49m\n\u001b[32m    499\u001b[39m \u001b[43m        \u001b[49m\u001b[43mengine_kwargs\u001b[49m\u001b[43m=\u001b[49m\u001b[43mengine_kwargs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[32m    500\u001b[39m \u001b[43m    \u001b[49m\u001b[43m)\u001b[49m\n\u001b[32m    501\u001b[39m \u001b[38;5;28;01melif\u001b[39;00m engine \u001b[38;5;129;01mand\u001b[39;00m engine != io.engine:\n\u001b[32m    502\u001b[39m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\n\u001b[32m    503\u001b[39m         \u001b[33m\"\u001b[39m\u001b[33mEngine should not be specified when passing \u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m    504\u001b[39m         \u001b[33m\"\u001b[39m\u001b[33man ExcelFile - ExcelFile already has the engine set\u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m    505\u001b[39m     )\n",
      "\u001b[36mFile \u001b[39m\u001b[32m~/anaconda3/envs/Arkvale/lib/python3.13/site-packages/pandas/io/excel/_base.py:1550\u001b[39m, in \u001b[36mExcelFile.__init__\u001b[39m\u001b[34m(self, path_or_buffer, engine, storage_options, engine_kwargs)\u001b[39m\n\u001b[32m   1548\u001b[39m     ext = \u001b[33m\"\u001b[39m\u001b[33mxls\u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m   1549\u001b[39m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[32m-> \u001b[39m\u001b[32m1550\u001b[39m     ext = \u001b[43minspect_excel_format\u001b[49m\u001b[43m(\u001b[49m\n\u001b[32m   1551\u001b[39m \u001b[43m        \u001b[49m\u001b[43mcontent_or_path\u001b[49m\u001b[43m=\u001b[49m\u001b[43mpath_or_buffer\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstorage_options\u001b[49m\u001b[43m=\u001b[49m\u001b[43mstorage_options\u001b[49m\n\u001b[32m   1552\u001b[39m \u001b[43m    \u001b[49m\u001b[43m)\u001b[49m\n\u001b[32m   1553\u001b[39m     \u001b[38;5;28;01mif\u001b[39;00m ext \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[32m   1554\u001b[39m         \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\n\u001b[32m   1555\u001b[39m             \u001b[33m\"\u001b[39m\u001b[33mExcel file format cannot be determined, you must specify \u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m   1556\u001b[39m             \u001b[33m\"\u001b[39m\u001b[33man engine manually.\u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m   1557\u001b[39m         )\n",
      "\u001b[36mFile \u001b[39m\u001b[32m~/anaconda3/envs/Arkvale/lib/python3.13/site-packages/pandas/io/excel/_base.py:1402\u001b[39m, in \u001b[36minspect_excel_format\u001b[39m\u001b[34m(content_or_path, storage_options)\u001b[39m\n\u001b[32m   1399\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(content_or_path, \u001b[38;5;28mbytes\u001b[39m):\n\u001b[32m   1400\u001b[39m     content_or_path = BytesIO(content_or_path)\n\u001b[32m-> \u001b[39m\u001b[32m1402\u001b[39m \u001b[38;5;28;01mwith\u001b[39;00m \u001b[43mget_handle\u001b[49m\u001b[43m(\u001b[49m\n\u001b[32m   1403\u001b[39m \u001b[43m    \u001b[49m\u001b[43mcontent_or_path\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[33;43m\"\u001b[39;49m\u001b[33;43mrb\u001b[39;49m\u001b[33;43m\"\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstorage_options\u001b[49m\u001b[43m=\u001b[49m\u001b[43mstorage_options\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mis_text\u001b[49m\u001b[43m=\u001b[49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\n\u001b[32m   1404\u001b[39m \u001b[43m\u001b[49m\u001b[43m)\u001b[49m \u001b[38;5;28;01mas\u001b[39;00m handle:\n\u001b[32m   1405\u001b[39m     stream = handle.handle\n\u001b[32m   1406\u001b[39m     stream.seek(\u001b[32m0\u001b[39m)\n",
      "\u001b[36mFile \u001b[39m\u001b[32m~/anaconda3/envs/Arkvale/lib/python3.13/site-packages/pandas/io/common.py:882\u001b[39m, in \u001b[36mget_handle\u001b[39m\u001b[34m(path_or_buf, mode, encoding, compression, memory_map, is_text, errors, storage_options)\u001b[39m\n\u001b[32m    873\u001b[39m         handle = \u001b[38;5;28mopen\u001b[39m(\n\u001b[32m    874\u001b[39m             handle,\n\u001b[32m    875\u001b[39m             ioargs.mode,\n\u001b[32m   (...)\u001b[39m\u001b[32m    878\u001b[39m             newline=\u001b[33m\"\u001b[39m\u001b[33m\"\u001b[39m,\n\u001b[32m    879\u001b[39m         )\n\u001b[32m    880\u001b[39m     \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[32m    881\u001b[39m         \u001b[38;5;66;03m# Binary mode\u001b[39;00m\n\u001b[32m--> \u001b[39m\u001b[32m882\u001b[39m         handle = \u001b[38;5;28;43mopen\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mhandle\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mioargs\u001b[49m\u001b[43m.\u001b[49m\u001b[43mmode\u001b[49m\u001b[43m)\u001b[49m\n\u001b[32m    883\u001b[39m     handles.append(handle)\n\u001b[32m    885\u001b[39m \u001b[38;5;66;03m# Convert BytesIO or file objects passed with an encoding\u001b[39;00m\n",
      "\u001b[31mFileNotFoundError\u001b[39m: [Errno 2] No such file or directory: '../4096/page_topk_accu_score.xlsx'"
     ]
    },
    {
     "data": {
      "image/png": "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tm1b7NixIw4fPhzf+9734uzZszFz5syIiKitrY2VK1d27T937tx488034+///u/j8OHD8bd/+7exf//+qKmp6bt7AQDAFa3g15ROnz49Tp06FatXr47m5uYYN25crF+/vuvp+OPHj3d7Leidd94Zzz77bDz//PPx3HPPxa233hpr1qyJ0aNH9929AADgitardxLV1NTEf/zHf8T+/fvjn//5n+Nzn/tc1882b94cK1as6Lb/n/7pn8a//uu/xv79++Nf/uVf4tixY74RqogU8g1g27Zti4ceeijuuuuuuOuuu2LevHl/8PHB5aXQb3y74JVXXokxY8bEN77xjX5eIX2p0Hn/9re/jbq6upg8eXJUVFTEvffe6//0K0ih837hhRfi3nvvjQkTJsSUKVPi6aef7vqUHS5fv/jFL2LRokUxefLkGDNmzEd+jOf/b/fu3fHggw9GRUVFfPnLX46XXnqp4PMO+NvbfSNUcSl03rt374777rsvNm3aFFu3bo2bbropHnnkkThx4sQAr5zeKHTeFxw7dix+8IMfxOc///kBWil9odB5t7W1xfz58+Pdd9+NVatWxWuvvRbLli2LkSNHDvDK6Y1C5/3yyy/HypUrY+nSpdHQ0BDLly+PhoaGeO655wZ45RTqzJkzMWbMmHjyySc/1v5NTU2xcOHCuPvuu+OnP/1pfO1rX4vvfOc78eabbxZ24s4B9pWvfKWzrq6u6+/t7e2dkydP7ly7dm2P+//FX/xF59e//vVu26qrqzv/5m/+pl/XSd8odN7/1/nz5zsnTZrUuWPHjn5aIX2pN/M+f/5855/92Z91btu2rfOJJ57oXLx48UAslT5Q6Lz/8R//sfOLX/xiZ1tb20AtkT5U6Lzr6uo6586d221bfX1955w5c/p1nfSt0aNHd/7bv/3bJfd55plnOu+7775u2771rW91PvLIIwWda0CvlF74RqiqqqqubR/nG6G+8IUvdNs2efLkePvtt/tzqfSB3sz7/zp79mycP3++Tz8Hjf7R23mvWbMmysrKorq6eiCWSR/pzbxff/31mDhxYnz/+9+PqqqquP/+++Pv/u7vor29faCWTS/1Zt6TJk2KAwcOdD3F39TUFLt27YopU6YMyJoZOH3VagP64fkD8Y1QXD56M+//69lnn41Pf/rT3f4j5PLUm3n/8pe/jJ/85Cexc+fOAVghfak3825qaoq33norZsyYEevWrYujR49GXV1dnD9/PpYuXToQy6aXejPvGTNmxOnTp+Ohhx6Kzs7OOH/+fMyZMycWLVo0EEtmAPXUauXl5dHa2hoffvhhDB069GPdjq9M4rK1bt26aGhoiB/+8IcxZMiQ7OXQx1pbW6O2tjaWLVsWI0aMyF4OA6CzszPKyspi2bJlUVFREdOnT49FixbF1q1bs5dGP9i9e3esXbs2nnzyyXjppZfihz/8YezatSvWrFmTvTQuUwN6pXQgvhGKy0dv5n3Bhg0bYt26dbFx48YYO3Zsfy6TPlLovJuamuLdd9+NxYsXd23r6OiIiIg77rgjXnvttfjsZz/bv4um13rz7/uGG26IQYMGRWlpade22267LZqbm6OtrS0GDx7cr2um93oz71WrVsUDDzzQ9dKcMWPGxJkzZ+K73/1uLF682FeJX0V6arWTJ0/GsGHDPvZV0ogBvlLqG6GKS2/mHRHx4x//OH70ox/F+vXro7KyciCWSh8odN633XZbvPzyy7Fz586uP9OmTYu77747du7cGTfeeONALp8C9ebf95133hlHjx7t+uUjIuKdd96JG264QZBe5noz7w8//PCi8LzwC0lnZ2f/LZYB11etNuC/pvhGqOJS6LzXrVsXq1atiqeffjo+85nPRHNzczQ3N8cHH3yQdRcoQCHzHjJkSIwePbrbn+uuuy4+9alPxejRo0XKFaDQf99f/epX4ze/+U0sX748fv3rX8cbb7wRa9eujYcffjjrLlCAQuc9derU+Kd/+qd45ZVXoqmpKf7zP/8zVq1aFVOnTu12tZzLzwcffBAHDx6MgwcPRsTvP7bv4MGD8d5770VExMqVK6O2trZr/zlz5kRTU1M888wzcfjw4diyZUu8+uqrMW/evILOO6BP30f4RqhiU+i8t27dGr/73e/im9/8ZrfbWbp0aTz22GMDunYKV+i8ubIVOu+bbropNmzYEPX19fHAAw/EyJEjY+7cufHoo49m3QUKUOi8Fy9eHNdcc008//zzceLEiRgxYkRMnTo1vv3tb2fdBT6m/fv3x9y5c7v+Xl9fHxERDz74YKxYsSKam5vj+PHjXT+/5ZZbYu3atVFfXx+bNm2KG2+8MZ566qm45557CjrvNZ2uoQMAkMwlCwAA0olSAADSiVIAANKJUgAA0olSAADSiVIAANKJUgAA0olSAADSiVIAANKJUgAA0olSAADSiVIAANL9L5+8WXCJwazhAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 800x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ========== 0. 可配置参数 ==========\n",
    "ARK_NAME   = 'Arkvale'        # <-- 想换名字只改这里\n",
    "GT_NAME    = 'Ground Truth'   # <-- 想换名字只改这里\n",
    "\n",
    "files_to_process = {\n",
    "    ARK_NAME: '../4096/page_topk_accu_score.xlsx',\n",
    "    GT_NAME : '../4096/token_topk_accu_score.xlsx'\n",
    "}\n",
    "# ====================================\n",
    "\n",
    "plt.style.use('seaborn-v0_8-whitegrid')\n",
    "fig, ax = plt.subplots(figsize=(8, 6))\n",
    "\n",
    "# 颜色、线型、标记\n",
    "colors   = cm.viridis(np.linspace(0, 1, 2))          # 2 条线\n",
    "linestyles = {ARK_NAME: '-', GT_NAME: '--'}\n",
    "markers    = {ARK_NAME: 'o', GT_NAME: '^'}\n",
    "\n",
    "# ========== 1. 按组顺序画图 ==========\n",
    "for idx, file_key in enumerate((ARK_NAME, GT_NAME)):\n",
    "    df = pd.read_excel(files_to_process[file_key], index_col=0)\n",
    "    df.index = df.index.str.replace('Token_', '', regex=False).astype(int)\n",
    "\n",
    "    # 计算每一行（每个 Token_Index）所有 Layer 的均值\n",
    "    mean_series = df.mean(axis=1)   # axis=1 按行求均值\n",
    "\n",
    "    ax.plot(mean_series.index, mean_series.values,\n",
    "            marker=markers[file_key],\n",
    "            markersize=6,\n",
    "            linestyle=linestyles[file_key],\n",
    "            color=colors[idx],\n",
    "            label=file_key)          # 图例只显示名字即可\n",
    "\n",
    "# ========== 2. 图例：单行即可 ==========\n",
    "ax.legend(title=None,\n",
    "          loc='upper center',\n",
    "          bbox_to_anchor=(0.5, 1.10),\n",
    "          ncol=2,\n",
    "          fontsize=11,\n",
    "          frameon=False,\n",
    "          shadow=False,\n",
    "          columnspacing=1.0,\n",
    "          handletextpad=0.5)\n",
    "\n",
    "# ========== 3. 其余不变 ==========\n",
    "ax.set_xlabel('Generated Token Index', fontsize=12)\n",
    "ax.set_ylabel('Average Attention Recall Rate', fontsize=12)\n",
    "ax.tick_params(axis='both', labelsize=12)\n",
    "ax.grid(True, which='both', linestyle='--', linewidth=0.5)\n",
    "plt.tight_layout(rect=[0, 0.03, 1, 0.92])\n",
    "plt.savefig('topk_accuracy_mean_comparison.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Arkvale",
   "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.13.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
