{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import os\n",
    "from itertools import product\n",
    "import math\n",
    "import numpy\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "from scipy.cluster.hierarchy import dendrogram, leaves_list\n",
    "from scipy.spatial.distance import pdist\n",
    "from sklearn.manifold import TSNE, MDS\n",
    "# import umap\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib import cm\n",
    "from mpl_toolkits.axes_grid1 import make_axes_locatable\n",
    "import time\n",
    "\n",
    "sys.path.append(os.path.abspath(\"../\"))\n",
    "\n",
    "from distance_functions import *\n",
    "import scipy.stats"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load distances and representations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "folder = \"distances/train/pretrained\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "# Load the ImageNet distances between train datasets\n",
    "filenames = os.listdir(folder)\n",
    "distnames = []\n",
    "for filename in filenames:\n",
    "    if filename.endswith(\"npy\"):\n",
    "        distnames.append(filename[:-4])\n",
    "distnames = np.sort(distnames)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['lin_cka_dist' 'lin_cka_prime_dist' 'mean_cca_e2e' 'mean_sq_cca_e2e'\n",
      " 'predictor_dist_0.0' 'predictor_dist_0.0001' 'predictor_dist_0.001'\n",
      " 'predictor_dist_0.01' 'predictor_dist_0.1' 'predictor_dist_1.0'\n",
      " 'predictor_dist_10.0' 'predictor_dist_100.0' 'predictor_dist_1000.0'\n",
      " 'predictor_dist_10000.0' 'predictor_dist_1e-05' 'predictor_dist_1e-06'\n",
      " 'predictor_dist_1e-07' 'predictor_dist_1e-08' 'predictor_dist_1e-09'\n",
      " 'predictor_dist_1e-10' 'predictor_dist_1e-11' 'predictor_dist_1e-12'\n",
      " 'predictor_dist_1e-13' 'predictor_dist_1e-14' 'predictor_dist_1e-15'\n",
      " 'predictor_dist_1e-16' 'predictor_dist_1e-17' 'predictor_dist_1e-18'\n",
      " 'predictor_dist_1e-19' 'predictor_dist_1e-20' 'procrustes'\n",
      " 'pwcca_dist_e2e']\n"
     ]
    }
   ],
   "source": [
    "print(distnames)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "stats = np.load(f\"{folder}/stats.npz\")\n",
    "model_names = stats[\"model_names\"]\n",
    "total_models = len(model_names)\n",
    "dist_pairs_saved = stats[\"dist_pairs_saved\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0\n",
      " 0 0 0 0 0 0 0]\n",
      "[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0\n",
      " 0 0 0 0 0 0 0]\n"
     ]
    }
   ],
   "source": [
    "# Check all distance pairs are saved. The output should be all-zeros\n",
    "print(np.arange(total_models) - np.sum(dist_pairs_saved, axis=0))\n",
    "print(np.arange(total_models) - np.flip(np.sum(dist_pairs_saved, axis=1)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['alexnet_pretrained_rep' 'convnext_base_pretrained_rep'\n",
      " 'convnext_large_pretrained_rep' 'convnext_small_pretrained_rep'\n",
      " 'convnext_tiny_pretrained_rep' 'densenet_pretrained_rep'\n",
      " 'efficientnet_b0_pretrained_rep' 'efficientnet_b1_pretrained_rep'\n",
      " 'efficientnet_b2_pretrained_rep' 'efficientnet_b3_pretrained_rep'\n",
      " 'efficientnet_b4_pretrained_rep' 'efficientnet_b5_pretrained_rep'\n",
      " 'efficientnet_b6_pretrained_rep' 'efficientnet_b7_pretrained_rep'\n",
      " 'googlenet_pretrained_rep' 'inception_pretrained_rep'\n",
      " 'mnasnet_pretrained_rep' 'mobilenet_v2_pretrained_rep'\n",
      " 'mobilenet_v3_large_pretrained_rep' 'mobilenet_v3_small_pretrained_rep'\n",
      " 'regnet_x_16gf_pretrained_rep' 'regnet_x_1_6gf_pretrained_rep'\n",
      " 'regnet_x_32gf_pretrained_rep' 'regnet_x_3_2gf_pretrained_rep'\n",
      " 'regnet_x_400mf_pretrained_rep' 'regnet_x_800mf_pretrained_rep'\n",
      " 'regnet_x_8gf_pretrained_rep' 'regnet_y_16gf_pretrained_rep'\n",
      " 'regnet_y_1_6gf_pretrained_rep' 'regnet_y_32gf_pretrained_rep'\n",
      " 'regnet_y_3_2gf_pretrained_rep' 'regnet_y_400mf_pretrained_rep'\n",
      " 'regnet_y_800mf_pretrained_rep' 'regnet_y_8gf_pretrained_rep'\n",
      " 'resnet18_pretrained_rep' 'resnext50_32x4d_pretrained_rep'\n",
      " 'shufflenet_pretrained_rep' 'squeezenet_pretrained_rep'\n",
      " 'vgg16_pretrained_rep' 'vit_b_16_pretrained_rep'\n",
      " 'vit_b_32_pretrained_rep' 'vit_l_16_pretrained_rep'\n",
      " 'vit_l_32_pretrained_rep' 'wide_resnet50_2_pretrained_rep']\n"
     ]
    }
   ],
   "source": [
    "print(model_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lin_cka_dist\n",
      "lin_cka_prime_dist\n",
      "mean_cca_e2e\n",
      "mean_sq_cca_e2e\n",
      "predictor_dist_0.0\n",
      "predictor_dist_0.0001\n",
      "predictor_dist_0.001\n",
      "predictor_dist_0.01\n",
      "predictor_dist_0.1\n",
      "predictor_dist_1.0\n",
      "predictor_dist_10.0\n",
      "predictor_dist_100.0\n",
      "predictor_dist_1000.0\n",
      "predictor_dist_10000.0\n",
      "predictor_dist_1e-05\n",
      "predictor_dist_1e-06\n",
      "predictor_dist_1e-07\n",
      "predictor_dist_1e-08\n",
      "predictor_dist_1e-09\n",
      "predictor_dist_1e-10\n",
      "predictor_dist_1e-11\n",
      "predictor_dist_1e-12\n",
      "predictor_dist_1e-13\n",
      "predictor_dist_1e-14\n",
      "predictor_dist_1e-15\n",
      "predictor_dist_1e-16\n",
      "predictor_dist_1e-17\n",
      "predictor_dist_1e-18\n",
      "predictor_dist_1e-19\n",
      "predictor_dist_1e-20\n",
      "procrustes\n",
      "pwcca_dist_e2e\n"
     ]
    }
   ],
   "source": [
    "distances = {}\n",
    "def symmetrize(A):\n",
    "    n = A.shape[0]\n",
    "    B = A.copy()\n",
    "    B[np.tril_indices(n)] = B.T[np.tril_indices(n)]\n",
    "    return B\n",
    "for distname in distnames:\n",
    "    print(distname)\n",
    "    curr_dist = np.load(f\"{folder}/{distname}.npy\")  \n",
    "    distances[distname] = symmetrize(curr_dist)\n",
    "#     print(distances[distname])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "alexnet_pretrained_rep\n",
      "WARNING: IN ORDER TO RUN THIS CODE, THE IMAGENET REPRESENTATIONS MUST COMPUTED. SEE README.\n"
     ]
    },
    {
     "ename": "FileNotFoundError",
     "evalue": "[Errno 2] No such file or directory: 'reps/train/10000_eval/alexnet_pretrained_rep.npy'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mFileNotFoundError\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-9-65e2a353f775>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m     25\u001b[0m \u001b[0;32mexcept\u001b[0m \u001b[0mFileNotFoundError\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     26\u001b[0m     \u001b[0mprint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'WARNING: IN ORDER TO RUN THIS CODE, THE IMAGENET REPRESENTATIONS MUST COMPUTED. SEE README.'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 27\u001b[0;31m     \u001b[0;32mraise\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m<ipython-input-9-65e2a353f775>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      9\u001b[0m     \u001b[0;32mfor\u001b[0m \u001b[0mmodel_name\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mmodel_names\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     10\u001b[0m         \u001b[0mprint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmodel_name\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 11\u001b[0;31m         \u001b[0mrep1\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtrain_reps_folder\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mmodel_name\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;34m\".npy\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     12\u001b[0m         \u001b[0;31m# center and normalize\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     13\u001b[0m         \u001b[0mrep1\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mrep1\u001b[0m \u001b[0;34m-\u001b[0m \u001b[0mrep1\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mmean\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0maxis\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mkeepdims\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mTrue\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/state/partition1/llgrid/pkg/anaconda/anaconda3-2021a/lib/python3.8/site-packages/numpy/lib/npyio.py\u001b[0m in \u001b[0;36mload\u001b[0;34m(file, mmap_mode, allow_pickle, fix_imports, encoding)\u001b[0m\n\u001b[1;32m    414\u001b[0m             \u001b[0mown_fid\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mFalse\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    415\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 416\u001b[0;31m             \u001b[0mfid\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mstack\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0menter_context\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mopen\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mos_fspath\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfile\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"rb\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    417\u001b[0m             \u001b[0mown_fid\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mTrue\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    418\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mFileNotFoundError\u001b[0m: [Errno 2] No such file or directory: 'reps/train/10000_eval/alexnet_pretrained_rep.npy'"
     ]
    }
   ],
   "source": [
    "train_reps_folder = 'reps/train/10000_eval/'\n",
    "val_reps_folder = 'reps/val/3000_eval/'\n",
    "\n",
    "# Load ImageNet representations\n",
    "reps_train = {} # Train dataset\n",
    "reps_val = {} # alidation dataset\n",
    "\n",
    "try:\n",
    "    for model_name in model_names:\n",
    "        print(model_name)\n",
    "        rep1 = np.load(train_reps_folder + model_name + \".npy\")\n",
    "        # center and normalize\n",
    "        rep1 = rep1 - rep1.mean(axis=1, keepdims=True)\n",
    "        rep1 = rep1 / np.linalg.norm(rep1)\n",
    "        rep1 = rep1 * np.sqrt(rep1.shape[1])\n",
    "        reps_train[model_name] = rep1\n",
    "\n",
    "        rep2 = np.load(val_reps_folder + model_name + \".npy\")\n",
    "        # center and normalize\n",
    "        rep2 = rep2 - rep2.mean(axis=1, keepdims=True)\n",
    "        rep2 = rep2 / np.linalg.norm(rep2)\n",
    "        rep2 = rep2 * np.sqrt(rep2.shape[1])\n",
    "        reps_val[model_name] = rep2\n",
    "    \n",
    "except FileNotFoundError as e:\n",
    "    print('WARNING: IN ORDER TO RUN THIS CODE, THE IMAGENET REPRESENTATIONS MUST COMPUTED. SEE README.')\n",
    "    raise e"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Predict generalization, random y's"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.datasets import load_iris\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "\n",
    "def find_best_pred(y, lmbda, reps, logistic=False):\n",
    "    # ridge regression \n",
    "    # assume reps is dimension x number datapoints \n",
    "\n",
    "    rep_dim = reps.shape[0]\n",
    "    numpts = reps.shape[1]\n",
    "    \n",
    "    if logistic is False:\n",
    "        return np.linalg.solve((lmbda*np.eye(rep_dim) + (reps @ reps.T) / numpts), reps@y)\n",
    "    else:     \n",
    "        clf = LogisticRegression(random_state=0).fit(reps.T, y)\n",
    "        return clf\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "def get_collected_correlations(lmbda, numtrials=50, numtrainsamples=10000):\n",
    "    collected_correlations = []\n",
    "\n",
    "    labels = []\n",
    "    for ky,val in distances.items():\n",
    "        if ky != 'predictor_dist_range':\n",
    "            labels.append(ky)\n",
    "\n",
    "    def flatten_upper_right_triangle(curr_mat):\n",
    "        cv = []\n",
    "        assert(curr_mat.shape[0] == curr_mat.shape[1])\n",
    "        assert(curr_mat.shape[0] == len(model_names))\n",
    "        for i in range(len(model_names)-1):\n",
    "            for j in range(i+1,len(model_names)):\n",
    "                cv.append(curr_mat[i,j])\n",
    "        cv = np.asarray(cv)\n",
    "        return cv\n",
    "\n",
    "    dist_vecs = {}\n",
    "\n",
    "    for distname in labels:\n",
    "        dist_vecs[distname] = flatten_upper_right_triangle(distances[distname])\n",
    "\n",
    "    numclasses = 1\n",
    "    log_reg = True\n",
    "    for tri in range(numtrials):\n",
    "        print(f'Trial {tri}')\n",
    "\n",
    "        y = np.random.randn(numtrainsamples,numclasses) \n",
    "        if log_reg:\n",
    "            y = y[:, 0]\n",
    "            y = (y > 0) \n",
    "        else:\n",
    "            log_reg += 1\n",
    "\n",
    "        preds = {}\n",
    "        count=0\n",
    "        for model_name in model_names:\n",
    "    #         print(model_name)\n",
    "            preds[model_name] = find_best_pred(y, lmbda, reps_train[model_name][:,0:numtrainsamples], logistic=log_reg)\n",
    "            if tri==0 and count==0 and not log_reg:\n",
    "                print('shape of prediction', preds[model_name].shape)\n",
    "                count+=1\n",
    "\n",
    "    #     # For each pair, compute the squared distance between predictions, averaged over test instances \n",
    "\n",
    "        errs = np.zeros((len(model_names), len(model_names)))\n",
    "        for ind1 in range(0, len(model_names)-1):\n",
    "            for ind2 in range(ind1+1, len(model_names)):\n",
    "                if not log_reg:\n",
    "                    cp1 = preds[model_names[ind1]].T @ reps_val[model_names[ind1]]\n",
    "                    cp2 = preds[model_names[ind2]].T @ reps_val[model_names[ind2]]\n",
    "                else:\n",
    "                    cp1 = preds[model_name].predict(reps_val[model_names[ind1]].T)\n",
    "                    cp2 = preds[model_name].predict(reps_val[model_names[ind2]].T)\n",
    "                if log_reg:\n",
    "                    errs[ind1, ind2] = (cp1 != cp2).sum()\n",
    "                else:\n",
    "                    errs[ind1, ind2] = np.linalg.norm(cp1 - cp2)\n",
    "                errs[ind2, ind1] = errs[ind1, ind2]\n",
    "        err_vec = flatten_upper_right_triangle(errs)\n",
    "\n",
    "        correlations = []\n",
    "\n",
    "        for distname in labels:\n",
    "            val = scipy.stats.spearmanr(err_vec, dist_vecs[distname]).correlation\n",
    "            correlations.append(val)\n",
    "\n",
    "        collected_correlations.append(correlations)\n",
    "        \n",
    "    return labels, collected_correlations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "label_names_dict = {'lin_cka_dist': 'CKA', 'mean_sq_cca_e2e' : 'CCA', 'pwcca_dist_e2e': 'PWCCA', 'procrustes': 'Procrustes'}\n",
    "\n",
    "label_names_dict['predictor_dist_0.0'] = 'GULP, $\\lambda = 0$'\n",
    "label_names_dict['predictor_dist_1e-07'] = 'GULP, $\\lambda = 10^{-7}$'\n",
    "label_names_dict['predictor_dist_1e-06'] = 'GULP, $\\lambda = 10^{-6}$'\n",
    "label_names_dict['predictor_dist_1e-05'] = 'GULP, $\\lambda = 10^{-5}$'\n",
    "label_names_dict['predictor_dist_0.0001'] = 'GULP, $\\lambda = 10^{-4}$'\n",
    "label_names_dict['predictor_dist_0.001'] = 'GULP, $\\lambda = 10^{-3}$'\n",
    "label_names_dict['predictor_dist_0.01'] = 'GULP, $\\lambda = 10^{-2}$'\n",
    "label_names_dict['predictor_dist_0.1'] = 'GULP, $\\lambda = 10^{-1}$'\n",
    "label_names_dict['predictor_dist_1.0'] = 'GULP, $\\lambda = 1$'\n",
    "label_names_dict['predictor_dist_10.0'] = 'GULP, $\\lambda = 10$'"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compute correlations over several trials"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial 0\n",
      "Trial 1\n",
      "Trial 2\n",
      "Trial 3\n",
      "Trial 4\n",
      "Trial 5\n",
      "Trial 6\n",
      "Trial 7\n",
      "Trial 8\n",
      "Trial 9\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import scipy\n",
    "import scipy.stats\n",
    "lmbda_vals = [0] #, 0.001, 0.01, 0.1, 1, 10]\n",
    "lmbda_names = ['0'] #, \"10^{-3}\", \"10^{-2}\", \"10^{-1}\", \"1\", \"10\"]\n",
    "# Lambda does not matter for logistic regression\n",
    "\n",
    "lmbda = 0\n",
    "labels, collected_correlations = get_collected_correlations(lmbda, numtrials=10,numtrainsamples=5000)\n",
    "\n",
    "std_devs = []\n",
    "means = []\n",
    "for i in range(len(collected_correlations[0])):\n",
    "    cvs = [collected_correlations[j][i] for j in range(len(collected_correlations))]\n",
    "    std_devs.append(scipy.stats.sem(cvs))\n",
    "    means.append(np.mean(cvs))\n",
    "std_devs = np.array(std_devs)\n",
    "means = np.array(means)\n",
    "subset_labels = ['lin_cka_dist', 'pwcca_dist_e2e', 'procrustes', 'mean_sq_cca_e2e', \\\n",
    "                 'predictor_dist_0.0', 'predictor_dist_1e-07', 'predictor_dist_1e-06',\\\n",
    "                 'predictor_dist_1e-05', 'predictor_dist_0.0001', 'predictor_dist_0.001', \\\n",
    "                 'predictor_dist_0.01', 'predictor_dist_0.1', 'predictor_dist_1.0', 'predictor_dist_10.0']\n",
    "subset_label_names = [label_names_dict[x] for x in subset_labels]\n",
    "subset_indices = []\n",
    "for i in range(len(subset_labels)):\n",
    "    subset_indices.append(labels.index(subset_labels[i]))\n",
    "lbels = labels\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "fig = plt.figure()\n",
    "ax = fig.add_axes([0,0,1,1])\n",
    "\n",
    "ax.bar(range(len(subset_labels)),means[subset_indices], yerr = std_devs[subset_indices])\n",
    "plt.xticks(range(len(subset_labels)), labels=subset_label_names, rotation='vertical', fontsize=15)\n",
    "plt.yticks(fontsize=15)\n",
    "plt.ylim(top=1)\n",
    "plt.title(f'Spearman $\\\\rho$ with prediction distance', fontsize=16)\n",
    "#plt.savefig('paper_figures/logistic_generalization.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:root] *",
   "language": "python",
   "name": "conda-root-py"
  },
  "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.5"
  },
  "latex_envs": {
   "LaTeX_envs_menu_present": true,
   "autoclose": false,
   "autocomplete": true,
   "bibliofile": "biblio.bib",
   "cite_by": "apalike",
   "current_citInitial": 1,
   "eqLabelWithNumbers": true,
   "eqNumInitial": 1,
   "hotkeys": {
    "equation": "Ctrl-E",
    "itemize": "Ctrl-I"
   },
   "labels_anchors": false,
   "latex_user_defs": false,
   "report_style_numbering": false,
   "user_envs_cfg": false
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": true,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {},
   "toc_section_display": true,
   "toc_window_display": false
  },
  "varInspector": {
   "cols": {
    "lenName": 16,
    "lenType": 16,
    "lenVar": 40
   },
   "kernels_config": {
    "python": {
     "delete_cmd_postfix": "",
     "delete_cmd_prefix": "del ",
     "library": "var_list.py",
     "varRefreshCmd": "print(var_dic_list())"
    },
    "r": {
     "delete_cmd_postfix": ") ",
     "delete_cmd_prefix": "rm(",
     "library": "var_list.r",
     "varRefreshCmd": "cat(var_dic_list()) "
    }
   },
   "types_to_exclude": [
    "module",
    "function",
    "builtin_function_or_method",
    "instance",
    "_Feature"
   ],
   "window_display": false
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
