{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "65210edb-7ab9-4452-a504-c803785e20a0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "60.0\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>algorithm</th>\n",
       "      <th>final_best_mean</th>\n",
       "      <th>final_best_std</th>\n",
       "      <th>auc_best_so_far_mean</th>\n",
       "      <th>auc_best_so_far_std</th>\n",
       "      <th>final_simple_regret_mean</th>\n",
       "      <th>final_simple_regret_std</th>\n",
       "      <th>evals_to_threshold_mean</th>\n",
       "      <th>evals_to_threshold_std</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>floorplan_botorch_mallows_EI_30</td>\n",
       "      <td>90.15</td>\n",
       "      <td>12.838516</td>\n",
       "      <td>8665.725</td>\n",
       "      <td>2638.886562</td>\n",
       "      <td>30.15</td>\n",
       "      <td>12.838516</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>floorplan_botorch_merge_EI_30</td>\n",
       "      <td>84.00</td>\n",
       "      <td>9.767292</td>\n",
       "      <td>8097.550</td>\n",
       "      <td>2167.350606</td>\n",
       "      <td>24.00</td>\n",
       "      <td>9.767292</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>floorplan_botorch_merge_EI_pairwise_pattern</td>\n",
       "      <td>86.20</td>\n",
       "      <td>10.547986</td>\n",
       "      <td>8379.900</td>\n",
       "      <td>2237.736594</td>\n",
       "      <td>26.20</td>\n",
       "      <td>10.547986</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>floorplan_botorch_merge_EI_shift_pairwise</td>\n",
       "      <td>86.95</td>\n",
       "      <td>11.508584</td>\n",
       "      <td>7940.675</td>\n",
       "      <td>1908.852569</td>\n",
       "      <td>26.95</td>\n",
       "      <td>11.508584</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>floorplan_botorch_merge_EI_shift_pairwise_pattern</td>\n",
       "      <td>83.30</td>\n",
       "      <td>11.576269</td>\n",
       "      <td>8140.800</td>\n",
       "      <td>2386.833799</td>\n",
       "      <td>23.30</td>\n",
       "      <td>11.576269</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>floorplan_botorch_merge_EI_shift_pattern</td>\n",
       "      <td>85.45</td>\n",
       "      <td>9.436498</td>\n",
       "      <td>8311.275</td>\n",
       "      <td>2202.002024</td>\n",
       "      <td>25.45</td>\n",
       "      <td>9.436498</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                           algorithm  final_best_mean  \\\n",
       "0                    floorplan_botorch_mallows_EI_30            90.15   \n",
       "1                      floorplan_botorch_merge_EI_30            84.00   \n",
       "2        floorplan_botorch_merge_EI_pairwise_pattern            86.20   \n",
       "3          floorplan_botorch_merge_EI_shift_pairwise            86.95   \n",
       "4  floorplan_botorch_merge_EI_shift_pairwise_pattern            83.30   \n",
       "5           floorplan_botorch_merge_EI_shift_pattern            85.45   \n",
       "\n",
       "   final_best_std  auc_best_so_far_mean  auc_best_so_far_std  \\\n",
       "0       12.838516              8665.725          2638.886562   \n",
       "1        9.767292              8097.550          2167.350606   \n",
       "2       10.547986              8379.900          2237.736594   \n",
       "3       11.508584              7940.675          1908.852569   \n",
       "4       11.576269              8140.800          2386.833799   \n",
       "5        9.436498              8311.275          2202.002024   \n",
       "\n",
       "   final_simple_regret_mean  final_simple_regret_std evals_to_threshold_mean  \\\n",
       "0                     30.15                12.838516                    None   \n",
       "1                     24.00                 9.767292                    None   \n",
       "2                     26.20                10.547986                    None   \n",
       "3                     26.95                11.508584                    None   \n",
       "4                     23.30                11.576269                    None   \n",
       "5                     25.45                 9.436498                    None   \n",
       "\n",
       "  evals_to_threshold_std  \n",
       "0                   None  \n",
       "1                   None  \n",
       "2                   None  \n",
       "3                   None  \n",
       "4                   None  \n",
       "5                   None  "
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "from copy import deepcopy\n",
    "\n",
    "def read_file(kernel_type, dim, benchmark_index, nruns):\n",
    "    file_name = 'tsp_botorch_'+kernel_type+'_EI_dim_'+str(dim)+'benchmark_index_'+str(benchmark_index)+'_nrun_'+str(nruns)+'.pkl'\n",
    "    data = torch.load(file_name, weights_only=False)\n",
    "    l = data['outputs']\n",
    "    l = [float(i) for i in l]\n",
    "    return l\n",
    "\n",
    "\n",
    "def read_file_filename(file_name):\n",
    "    data = torch.load(file_name, weights_only=False)\n",
    "    l = data['outputs']\n",
    "    l = [float(i) for i in l]\n",
    "    return l\n",
    "\n",
    "\n",
    "def read_file_no_anchor(kernel_type, dim, benchmark_index, nruns):\n",
    "    file_name = 'tsp_botorch_'+kernel_type+'_EI_dim_'+str(dim)+'benchmark_index_no_anchor_'+str(benchmark_index)+'_nrun_'+str(nruns)+'.pkl'\n",
    "    data = torch.load(file_name, weights_only=False)\n",
    "    l = data['outputs']\n",
    "    l = [float(i) for i in l]\n",
    "    return l\n",
    "\n",
    "\n",
    "import os\n",
    "\n",
    "def analyse_trial(dim=10, benchmark_index=0):\n",
    "    folders = os.listdir('./results')\n",
    "    nruns = 20\n",
    "    results_dict = {}\n",
    "\n",
    "    for folder in folders:\n",
    "        if '.' in folder:\n",
    "            continue\n",
    "        results_dict[folder] = []\n",
    "        for nrun in range(nruns):\n",
    "            results_dict[folder].append(read_file_filename(os.path.join('./results', folder, folder+f'_nrun_{nrun}.pkl')))\n",
    "\n",
    "    all_results = []\n",
    "    for key in results_dict.keys():\n",
    "        results_dict[key] = np.array(results_dict[key])\n",
    "        all_results.append(results_dict[key])\n",
    "    # print(all_results[0].shape)\n",
    "    # return all_results\n",
    "    global_minimum = np.min(all_results)\n",
    "    print(global_minimum)\n",
    "    best_so_far = [np.minimum.accumulate(res, axis=1) for res in all_results]\n",
    "    regrets = [bfs - global_minimum for bfs in best_so_far]\n",
    "    for i, key in enumerate(results_dict.keys()):\n",
    "        # results_dict[key] = regrets[i]\n",
    "        results_dict[key] = all_results[i]\n",
    "    return results_dict\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.metrics import auc\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.metrics import auc\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.metrics import auc\n",
    "\n",
    "def evaluate_algorithms(r: dict, f_opt=None, threshold=None):\n",
    "    \"\"\"\n",
    "    r: dict of {algorithm_name: np.ndarray of shape (n_repeats, n_iterations)}\n",
    "    f_opt: known global minimum value (float)\n",
    "    threshold: optional regret threshold to measure how many iterations are needed\n",
    "    \n",
    "    Returns:\n",
    "        pd.DataFrame with aggregated metrics for each algorithm\n",
    "    \"\"\"\n",
    "    results = []\n",
    "\n",
    "    for algo, outputs in r.items():\n",
    "        outputs = np.array(outputs)  # shape: (n_repeats, n_iterations)\n",
    "        n_repeats, n_iterations = outputs.shape\n",
    "\n",
    "        best_so_far = np.minimum.accumulate(outputs, axis=1)\n",
    "        \n",
    "        final_best = best_so_far[:, -1]\n",
    "        auc_vals = np.array([\n",
    "            auc(np.arange(1, n_iterations+1), best_so_far[i] - f_opt)\n",
    "            for i in range(n_repeats)\n",
    "        ])\n",
    "\n",
    "        metrics = {\n",
    "            \"algorithm\": algo,\n",
    "            \"final_best_mean\": np.mean(final_best),\n",
    "            \"final_best_std\": np.std(final_best),\n",
    "            \"auc_best_so_far_mean\": np.mean(auc_vals),\n",
    "            \"auc_best_so_far_std\": np.std(auc_vals),\n",
    "        }\n",
    "\n",
    "        if f_opt is not None:\n",
    "            simple_regrets       = best_so_far - f_opt       # ← 修正\n",
    "            instantaneous_regret = outputs - f_opt           # (可选) 如需两者都存\n",
    "            final_simple         = simple_regrets[:, -1]\n",
    "\n",
    "            metrics.update({\n",
    "                \"final_simple_regret_mean\": np.mean(final_simple),\n",
    "                \"final_simple_regret_std\": np.std(final_simple),\n",
    "            })\n",
    "        else:\n",
    "            metrics.update({\n",
    "                \"final_simple_regret_mean\": None,\n",
    "                \"final_simple_regret_std\": None,\n",
    "            })\n",
    "\n",
    "        if threshold is not None:\n",
    "            evals_to_threshold = []\n",
    "            for i in range(n_repeats):\n",
    "                for j in range(n_iterations):\n",
    "                    if best_so_far[i, j] <= threshold:\n",
    "                        evals_to_threshold.append(j + 1)\n",
    "                        break\n",
    "                else:\n",
    "                    evals_to_threshold.append(n_iterations)\n",
    "            evals_to_threshold = np.array(evals_to_threshold)\n",
    "            metrics.update({\n",
    "                \"evals_to_threshold_mean\": np.mean(evals_to_threshold),\n",
    "                \"evals_to_threshold_std\": np.std(evals_to_threshold),\n",
    "            })\n",
    "        else:\n",
    "            metrics.update({\n",
    "                \"evals_to_threshold_mean\": None,\n",
    "                \"evals_to_threshold_std\": None,\n",
    "            })\n",
    "\n",
    "        results.append(metrics)\n",
    "\n",
    "    return pd.DataFrame(results)\n",
    "\n",
    "\n",
    "r = analyse_trial()\n",
    "k = evaluate_algorithms(r, 60)\n",
    "k"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "530ccdc0-ea89-43ca-998b-90e80e78ddf9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "60.0\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAeoAAAGGCAYAAAC0W8IbAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/P9b71AAAACXBIWXMAAA9hAAAPYQGoP6dpAABrkUlEQVR4nO3dd3xT9f748dfJbtqm6R5QKBuVIfviABQUkKtexaty8SrqdYIDxcH9qSgOnLjH1avg5upXRC9evVeRIUNEhqIM2WW0QHfTNvv8/ogEQlfSJk1a3s/HIw+acz45eTcNeeezFVVVVYQQQggRkzTRDkAIIYQQ9ZNELYQQQsQwSdRCCCFEDJNELYQQQsQwSdRCCCFEDJNELYQQQsQwSdRCCCFEDJNELYQQQsQwXbQDiAVer5cDBw6QmJiIoijRDkcIIUQbp6oqlZWV5OTkoNE0XGeWRA0cOHCA3NzcaIchhBDiBLN3717at2/fYBlJ1EBiYiLge8EsFkuUoxFCCNHWVVRUkJub688/DZFEDf7mbovFIolaCCFEiwmmu1UGkwkhhBAxTBK1EEIIEcMkUQshhBAxTPqohRCiFfN6vTidzmiHIY6j1+vRarVhuZYkaiGEaKWcTie7du3C6/VGOxRRB6vVSlZWVrPX55BELYQQrZCqqhQUFKDVasnNzW100QzRclRVpbq6mkOHDgGQnZ3drOtJohZCiFbI7XZTXV1NTk4OZrM52uGI48TFxQFw6NAhMjIymtUMLl/BhBCiFfJ4PAAYDIYoRyLqc+QLlMvlatZ1JFELIUQrJvsTxK5w/W0kUQshhBAxTBK1EEKIVmn37t0oisKGDRsAWLJkCYqiUFZWFtW4wk0StRBCiBYzadIkFEXhxhtvrHVu8uTJKIrCpEmTWj6wGCaJOsxUmc8ohBANys3NZd68edTU1PiP2e12PvjgAzp06BDFyGKTJOow83jc0Q5BCCFiWv/+/cnNzWX+/Pn+Y/Pnz6dDhw7069fPf+yrr77ijDPOwGq1kpqayh//+Ed27NgR0nN98sknnHLKKRiNRvLy8njmmWf851566SV69erlv79gwQIUReG1117zHxs1ahT33XcfAD/99BNnnXUWiYmJWCwWBgwYwI8//hjy7x8qSdRhJisECSGiQVVVqp3uqNxUVQ053muuuYY5c+b477/11ltcffXVAWWqqqq44447+PHHH1m0aBEajYaLLroo6M/ZtWvXcumll3L55ZezceNGHnzwQe6//37mzp0LwPDhw9m0aROHDx8GYOnSpaSlpbFkyRLAN61q1apVjBgxAoCJEyfSvn171qxZw9q1a7n33nvR6/Uh/+6hkgVPwszr9UQ7BCHECajG5eHkB/4blefeNHM0ZkNo6eSKK65g+vTp7NmzB4AVK1Ywb948f5IEGD9+fMBj3nrrLdLT09m0aVNATbg+s2fPZuTIkdx///0AdO/enU2bNvHUU08xadIkevXqRUpKCkuXLuWSSy5hyZIl3HnnnTz//PMA/PDDD7hcLk477TQA8vPzueuuu+jZsycA3bp1C+l3biqpUYeZV5q+hRCiUenp6YwbN465c+cyZ84cxo0bR1paWkCZbdu2MWHCBDp37ozFYiEvLw/wJcxgbN68mdNPPz3g2Omnn862bdvweDwoisKwYcNYsmQJZWVlbNq0iZtvvhmHw8GWLVtYunQpgwYN8i9ccscdd/C3v/2NUaNG8fjjj4fcDN9UUqMOM2n6FkJEQ5xey6aZo6P23E1xzTXXMGXKFABefvnlWufPP/98OnbsyBtvvEFOTg5er5devXqFdbewESNG8Prrr/Pdd9/Rr18/LBaLP3kvXbqU4cOH+8s++OCD/OUvf+GLL77gyy+/ZMaMGcybN4+LLroobPHURRJ1mEmiFkJEg6IoITc/R9uYMWNwOp0oisLo0YFfMoqLi9m6dStvvPEGZ555JgDLly8P6fonnXQSK1asCDi2YsUKunfv7l97e/jw4dx+++18/PHH/r7oESNG8M0337BixQruvPPOgMd3796d7t27M3XqVCZMmMCcOXMkUbc2qjR9CyFEULRaLZs3b/b/fKzk5GRSU1N5/fXXyc7OJj8/n3vvvTek6995550MGjSIhx9+mMsuu4xVq1bx0ksv8corr/jL9OnTh+TkZD744AMWLlwI+BL1tGnTUBTF33ReU1PDXXfdxSWXXEKnTp3Yt28fa9asqdWPHgnSRx1mUqMWQojgWSwWLBZLreMajYZ58+axdu1aevXqxdSpU3nqqadCunb//v356KOPmDdvHr169eKBBx5g5syZAQuqKIrCmWeeiaIonHHGGYAveVssFgYOHEh8fDzg+yJRXFzMlVdeSffu3bn00ksZO3YsDz30UNN/+SApalPG1bcxFRUVJCUlUV5eXucbJhTFB/eRmtk+TJEJIUTd7HY7u3btolOnTphMpmiHI+rQ0N8olLwjNeowUz0yPUsIIUT4SKIOMxVVlhEVQggRNlFN1MuWLeP8888nJycHRVFYsGBBwHlFUeq8HdtPkZeXV+v8448/3sK/SSDppxZCCBEuUU3UVVVV9O3bt875cwAFBQUBt7feegtFUWqNsps5c2ZAuVtuuaUlwq+XrPcthBAiXKI6PWvs2LGMHTu23vNZWVkB9z/77DPOOussOnfuHHA8MTGxVtlokmVEhRBChEur6aM+ePAgX3zxBddee22tc48//jipqan069ePp556Crc7ujVa6aMWQggRLq1mwZO3336bxMRELr744oDjt956K/379yclJYWVK1cyffp0CgoKmD17dr3XcjgcOBwO//2KioqwxipN30IIIcKl1STqt956i4kTJ9aai3bHHXf4f+7Tpw8Gg4EbbriBWbNmYTQa67zWrFmzIjpJXQaTCSGECJdW0fT93XffsXXrVv72t781WnbIkCG43W52795db5np06dTXl7uv+3duzeM0UrTtxBCiPBpFYn6zTffZMCAAfTt27fRshs2bECj0ZCRkVFvGaPR6F+2rr7l65pDtroUQoi6TZo0CUVRuPHGG2udmzx5MoqiBCzxKaLc9G2z2di+fbv//q5du9iwYQMpKSl06NAB8PUff/zxxzzzzDO1Hr9q1SpWr17NWWedRWJiIqtWrWLq1KlcccUVJCcnt9jvcTwZ9S2EEPXLzc1l3rx5PPvss8TFxQG+5TY/+OAD/2d/U6iqisfjQadrNb26QYlqjfrHH3+kX79+9OvXD/D1N/fr148HHnjAX2bevHmoqsqECRNqPd5oNDJv3jyGDx/OKaecwqOPPsrUqVN5/fXXW+x3qIuqStO3EELUp3///uTm5jJ//nz/sfnz59OhQwd/PgDfeJ9Zs2bRqVMn4uLi6Nu3L//3f//nP79kyRIUReHLL79kwIABGI1Gli9fTmVlJRMnTiQ+Pp7s7GyeffZZRowYwe233+5/rMPhYNq0abRr1474+HiGDBnCkiVLWuLXD1lUv3aMGDGCxvYEuf7667n++uvrPNe/f3++//77SITWLF5Z71sIIRp0zTXXMGfOHCZOnAj4BgxfffXVAcly1qxZvPfee7z22mt069aNZcuWccUV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LheXXnopqqry6quvBpy74447/D/36dMHg8HADTfcwKxZs6K6mtbll1/u/7l379706dOHLl26sGTJEkaOHBm1uEL11ltvMXHiREwmU8DxWHjd6/tsjDZp+m6GtLQ0tFptrdF/Bw8eJCsrK0pRNWzKlCksXLiQxYsXN7pj2JAhQwDYvn17S4QWEqvVSvfu3dm+fTtZWVk4nU7KysoCysTa32HPnj188803/O1vf2uwXKy+7kdey4be71lZWbUGUrrdbkpKSmLmb3EkSe/Zs4evv/660V2QhgwZgtvtZvfu3S0TYJA6d+5MWlqa/33SGl777777jq1btzb6fwBa/nWv77MxmM+XrKysOv9fHDnXXJKom8FgMDBgwAAWLVrkP+b1elm0aBFDhw6NYmS1qarKlClT+PTTT/n222/p1KlTo4/ZsGEDANnZ2RGOLnQ2m40dO3aQnZ3NgAED0Ov1AX+HrVu3kp+fH1N/hzlz5pCRkcG4ceMaLBerr3unTp3IysoKeJ0rKipYvXq1/3UeOnQoZWVlrF271l/m22+/xev1+r+ARNORJL1t2za++eYbUlNTG33Mhg0b0Gg0tZqVo23fvn0UFxf73yex/tqDr0VpwIAB9O3bt9GyLfW6N/bZGMzny9ChQ9m4cWPAF6UjXwJPPvnksAQpmmHevHmq0WhU586dq27atEm9/vrrVavVGjD6LxbcdNNNalJSkrpkyRK1oKDAf6uurlZVVVW3b9+uzpw5U/3xxx/VXbt2qZ999pnauXNnddiwYVGO3OfOO+9UlyxZou7atUtdsWKFOmrUKDUtLU09dOiQqqqqeuONN6odOnRQv/32W/XHH39Uhw4dqg4dOjTKUR/l8XjUDh06qPfcc0/A8Vh73SsrK9X169er69evVwF19uzZ6vr16/2joh9//HHVarWqn332mfrzzz+rF154odqpUye1pqbGf40xY8ao/fr1U1evXq0uX75c7datmzphwoSox+90OtULLrhAbd++vbphw4aA/wdHRuauXLlSffbZZ9UNGzaoO3bsUN977z01PT1dvfLKK6Mae2VlpTpt2jR11apV6q5du9RvvvlG7d+/v9qtWzfVbrf7rxGt176x942qqmp5eblqNpvVV199tdbjo/m6N/bZqKqNf7643W61V69e6rnnnqtu2LBB/eqrr9T09HR1+vTpYYlREnUYvPjii2qHDh1Ug8GgDh48WP3++++jHVItQJ23OXPmqKqqqvn5+eqwYcPUlJQU1Wg0ql27dlXvuusutby8PLqB/+6yyy5Ts7OzVYPBoLZr10697LLL1O3bt/vP19TUqDfffLOanJysms1m9aKLLlILCgqiGHGg//73vyqgbt26NeB4rL3uixcvrvN9ctVVV6mq6puidf/996uZmZmq0WhUR44cWet3Ki4uVidMmKAmJCSoFotFvfrqq9XKysqox79r1656/x8sXrxYVVVVXbt2rTpkyBA1KSlJNZlM6kknnaQ+9thjAckwGrFXV1er5557rpqenq7q9Xq1Y8eO6nXXXVerQhCt176x942qquo//vEPNS4uTi0rK6v1+Gi+7o19NqpqcJ8vu3fvVseOHavGxcWpaWlp6p133qm6XK6wxCi7ZwkhhBAxTPqohRBCiBgmiVoIIYSIYZKohRBCiBgmiVoIIYSIYZKohRBCiBgmiVoIIYSIYZKohRBCiBgmiVoIIYSIYZKohRARl5eXx3PPPRftMIRolSRRC9HGTJo0iT/96U8AjBgxgttvv73Fnnvu3Lm19u0FWLNmTcB2nkKI4Ml+1EKIRjmdTgwGQ5Mfn56eHsZohDixSI1aiDZq0qRJLF26lOeffx5FUVAUxb+37y+//MLYsWNJSEggMzOTv/71rxQVFfkfO2LECKZMmcLtt99OWloao0ePBmD27Nn07t2b+Ph4cnNzufnmm7HZbAAsWbKEq6++mvLycv/zPfjgg0Dtpu/8/HwuvPBCEhISsFgsXHrppQH7+T744IOceuqpvPvuu+Tl5ZGUlMTll19OZWVlZF80IWKQJGoh2qjnn3+eoUOHct1111FQUEBBQQG5ubmUlZVx9tln069fP3788Ue++uorDh48yKWXXhrw+LfffhuDwcCKFSt47bXXANBoNLzwwgv8+uuvvP3223z77bfcfffdAJx22mk899xzWCwW//NNmzatVlxer5cLL7yQkpISli5dytdff83OnTu57LLLAsrt2LGDBQsWsHDhQhYuXMjSpUt5/PHHI/RqCRG7pOlbiDYqKSkJg8GA2WwmKyvLf/yll16iX79+PPbYY/5jb731Frm5ufz22290794dgG7duvHkk08GXPPY/u68vDweeeQRbrzxRl555RUMBgNJSUkoihLwfMdbtGgRGzduZNeuXeTm5gLwzjvvcMopp7BmzRoGDRoE+BL63LlzSUxMBOCvf/0rixYt4tFHH23eCyNEKyM1aiFOMD/99BOLFy8mISHBf+vZsyfgq8UeMWDAgFqP/eabbxg5ciTt2rUjMTGRv/71rxQXF1NdXR3082/evJnc3Fx/kgY4+eSTsVqtbN682X8sLy/Pn6QBsrOzOXToUEi/qxBtgdSohTjB2Gw2zj//fJ544ola57Kzs/0/x8fHB5zbvXs3f/zjH7npppt49NFHSUlJYfny5Vx77bU4nU7MZnNY49Tr9QH3FUXB6/WG9TmEaA0kUQvRhhkMBjweT8Cx/v3788knn5CXl4dOF/xHwNq1a/F6vTzzzDNoNL7GuI8++qjR5zveSSedxN69e9m7d6+/Vr1p0ybKyso4+eSTg45HiBOFNH0L0Ybl5eWxevVqdu/eTVFREV6vl8mTJ1NSUsKECRNYs2YNO3bs4L///S9XX311g0m2a9euuFwuXnzxRXbu3Mm7777rH2R27PPZbDYWLVpEUVFRnU3io0aNonfv3kycOJF169bxww8/cOWVVzJ8+HAGDhwY9tdAiNZOErUQbdi0adPQarWcfPLJpKenk5+fT05ODitWrMDj8XDuuefSu3dvbr/9dqxWq7+mXJe+ffsye/ZsnnjiCXr16sX777/PrFmzAsqcdtpp3HjjjVx22WWkp6fXGowGvibszz77jOTkZIYNG8aoUaPo3Lkz//rXv8L++wvRFiiqqqrRDkIIIYQQdZMatRBCCBHDJFELIYQQMUwStRBCCBHDJFELIYQQMUwStRBCCBHDJFELIYQQMUwStRBCCBHDJFELIYQQMUwStRBCCBHDJFELIYQQMUwStRBCCBHDJFELIYQQMez/Ax0KdOlEYQ6BAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 500x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# ---------- 1. 读取并整理数据 ----------\n",
    "r    = analyse_trial()   # 你的函数\n",
    "results = {}\n",
    "results['Mallows'] = r['floorplan_botorch_mallows_EI_30']\n",
    "results['Merge'] = r['floorplan_botorch_merge_EI_30']\n",
    "\n",
    "\n",
    "global_min = np.inf\n",
    "for arr in results.values():\n",
    "    global_min = min(global_min, arr[:, :].min())\n",
    "\n",
    "# 把每个算法的 simple regret（best‑so‑far – global_min）整理成均值±std\n",
    "mean_regret = {}\n",
    "std_regret  = {}\n",
    "\n",
    "for algo, outputs in results.items():\n",
    "    outputs        = outputs[:, :]                          \n",
    "    best_so_far    = np.minimum.accumulate(outputs, axis=1)    \n",
    "    regrets        = best_so_far - global_min                 \n",
    "    mean_regret[algo] = regrets.mean(axis=0)             \n",
    "    std_regret[algo]  = regrets.std(axis=0)\n",
    "\n",
    "# ---------- 2. 画图 ----------\n",
    "iters = np.arange(1, 201)\n",
    "\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "for algo in mean_regret:\n",
    "    mean = mean_regret[algo]\n",
    "    std  = std_regret[algo]\n",
    "    plt.plot(iters, mean, label=algo)\n",
    "    plt.fill_between(iters, mean - std, mean + std, alpha=0.2)\n",
    "\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Simple Regret (mean ± 1 std)\")\n",
    "# plt.yscale(\"log\")           # 若差距跨数量级，建议用对数轴；可去掉\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"fp_curve.pdf\", dpi=600, bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "5d203fcb-9f54-45cd-9657-d786a7849fcc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "65.0\n",
      "dict_keys(['floorplan_botorch_mallows_EI_30', 'floorplan_botorch_merge_EI_30'])\n",
      "win: 10, tie: 4, loss: 6\n",
      "65.0\n",
      "win: 10, tie: 1, loss: 9\n"
     ]
    }
   ],
   "source": [
    "\n",
    "def analyse_trial(dim=10, benchmark_index=0):\n",
    "    folders = os.listdir('./results')[:2]\n",
    "    nruns = 20\n",
    "    results_dict = {}\n",
    "\n",
    "    for folder in folders:\n",
    "        if '.' in folder:\n",
    "            continue\n",
    "        results_dict[folder] = []\n",
    "        for nrun in range(nruns):\n",
    "            results_dict[folder].append(read_file_filename(os.path.join('./results', folder, folder+f'_nrun_{nrun}.pkl')))\n",
    "\n",
    "    all_results = []\n",
    "    for key in results_dict.keys():\n",
    "        results_dict[key] = np.array(results_dict[key])\n",
    "        all_results.append(results_dict[key])\n",
    "    # print(all_results[0].shape)\n",
    "    # return all_results\n",
    "    global_minimum = np.min(all_results)\n",
    "    print(global_minimum)\n",
    "    best_so_far = [np.minimum.accumulate(res, axis=1) for res in all_results]\n",
    "    regrets = [bfs - global_minimum for bfs in best_so_far]\n",
    "    for i, key in enumerate(results_dict.keys()):\n",
    "        results_dict[key] = regrets[i]\n",
    "        # results_dict[key] = all_results[i]\n",
    "    return results_dict\n",
    "\n",
    "# analyse_trial()\n",
    "\n",
    "def final_regret_count(d):\n",
    "    print(d.keys())\n",
    "    mallows = d[list(d.keys())[0]]\n",
    "    merge = d[list(d.keys())[1]]\n",
    "    win = 0\n",
    "    tie = 0\n",
    "    loss = 0\n",
    "    for i in range(20):\n",
    "        if mallows[i, -1] > merge[i, -1]:\n",
    "            win += 1\n",
    "        elif mallows[i, -1] == merge[i, -1]:\n",
    "            tie += 1\n",
    "        else:\n",
    "            loss += 1\n",
    "    print(f'win: {win}, tie: {tie}, loss: {loss}')\n",
    "\n",
    "\n",
    "def cumulative_regret_count(d):\n",
    "    mallows = d[list(d.keys())[0]]\n",
    "    merge = d[list(d.keys())[1]]\n",
    "    win = 0\n",
    "    tie = 0\n",
    "    loss = 0\n",
    "    for i in range(20):\n",
    "        if mallows[i].mean() > merge[i].mean():\n",
    "            win += 1\n",
    "        elif mallows[i].mean() == merge[i].mean():\n",
    "            tie += 1\n",
    "        else:\n",
    "            loss += 1\n",
    "    print(f'win: {win}, tie: {tie}, loss: {loss}')\n",
    "\n",
    "final_regret_count(analyse_trial())\n",
    "cumulative_regret_count(analyse_trial())"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
