{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU",
    "gpuClass": "standard"
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "FuEFM_tPC7F5"
      },
      "outputs": [],
      "source": [
        "import torch\n",
        "import torch.nn as nn\n",
        "import torch.nn.functional as F\n",
        "import numpy as np\n",
        "import random\n",
        "import torch.optim as optim\n",
        "import matplotlib.pyplot as plt\n",
        "import os\n",
        "from torch.optim.optimizer import Optimizer, required\n",
        "## two layer neural network\n",
        "\n",
        "from scipy.linalg import svdvals\n",
        "\n",
        "import scipy as sp"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "torch.manual_seed(10)\n",
        "N = 300\n",
        "d = 2\n",
        "X = torch.zeros([N, d]).normal_(0,1).cuda()\n",
        "Xtest = torch.zeros([1000*N, d]).normal_(0,1).cuda()\n",
        "\n",
        "eps = torch.zeros([N, 1]).normal_(0, 0.7).cuda()\n",
        "eps2 = torch.zeros([N, d]).normal_(0, 0.2).cuda()\n",
        "\n",
        "\n",
        "G = torch.tensor([0.1, 0]).cuda()\n",
        "mask = torch.tensor([0.05, 1]).cuda()\n",
        "Y = X.mv(G).unsqueeze(1) + eps\n",
        "X = X + eps2 * mask\n",
        "print(X.shape)\n",
        "print(Y.shape)\n",
        "Ytest = Xtest.mv(G).unsqueeze(1)\n",
        "\n",
        "print((Y * X).mean(dim=0))\n",
        "print( (Y * X).mean(dim=0).norm())\n",
        "\n",
        "Xprime = X.cpu().numpy()\n",
        "sin = np.linalg.eigvals(np.matmul(Xprime.T, Xprime)/N)\n",
        "print(sin)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "1sYAWqTJC-Nc",
        "outputId": "73f0cc44-3e8d-47f2-f17f-421ac1bd3e7c"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "torch.Size([300, 2])\n",
            "torch.Size([300, 1])\n",
            "tensor([ 0.0845, -0.0242], device='cuda:0')\n",
            "tensor(0.0879, device='cuda:0')\n",
            "[0.81168836 1.0896001 ]\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "class Net(nn.Module):\n",
        "\n",
        "    def __init__(self, width=10, depth=0):\n",
        "        super(Net, self).__init__()\n",
        "        \n",
        "        #self.w3 = nn.Linear(width, 1)\n",
        "        self.layers = nn.ModuleList()\n",
        "        if depth == 0:\n",
        "          weight = nn.Linear(d, d, bias=False)\n",
        "          weight.weight.data.normal_(0, 0.02)\n",
        "          self.layers.append(weight)\n",
        "          return\n",
        "        \n",
        "        self.layers.append(nn.Linear(d, width, bias=False))\n",
        "        for _ in range(depth-1):\n",
        "            weight = nn.Linear(width, width, bias=False)\n",
        "            #print(weight.weight.var())\n",
        "            #weight.weight.data.zero_()\n",
        "            self.layers.append(weight)\n",
        "        self.layers.append(nn.Linear(width, 32, bias=False))\n",
        "\n",
        "    def forward(self, x):\n",
        "\n",
        "        for layer in self.layers:\n",
        "          x = layer(x)\n",
        "        return x\n",
        "    def get_norm(self):\n",
        "      norm = 0\n",
        "      for layer in self.layers:\n",
        "          norm = layer.weight.data.norm() ** 2 + norm\n",
        "      return norm\n",
        "\n",
        "    def get_inner(self):\n",
        "      #if \n",
        "      ws = []\n",
        "      #norm = 0\n",
        "      for layer in self.layers:\n",
        "          ws.append(layer.weight.data)\n",
        "          \n",
        "      return (ws[0] * ws[1])#.sum()\n",
        "    \n",
        "    def noisify(self, T=1):\n",
        "      for layer in self.layers:\n",
        "          layer.weight.data += T * torch.zeros_like(layer.weight.data).normal_(0, 1)\n",
        "model = Net()"
      ],
      "metadata": {
        "id": "6KOSKMdEDCp1"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "## generalization of beta infonce\n",
        "\n",
        "\n",
        "exp = np.arange(1, 14)\n",
        "#width_range = 2 ** exp\n",
        "\n",
        "\n",
        "losses = []\n",
        "norms = []\n",
        "weights = []\n",
        "test_losses = []\n",
        "\n",
        "losses2 = []\n",
        "norms2 = []\n",
        "weights2 = []\n",
        "test_losses2 = []\n",
        "\n",
        "\n",
        "sigma_list =  np.arange(0, 1.5, 0.05) ** 0.5\n",
        "#T_list =  np.arange(0.1, 1,0.1)\n",
        "#sigma = 1\n",
        "T = 0.5\n",
        "\n",
        "for sigma in sigma_list:\n",
        "  torch.manual_seed(2)\n",
        "  model2 = Net(depth=0).cuda()\n",
        "  torch.manual_seed(2)\n",
        "  model = Net(depth=0).cuda()\n",
        "\n",
        "  lr = 0.0015 #/ (width ** 0.2)\n",
        "  #opt = optim.SGD(model.parameters(), lr=lr, momentum=0)\n",
        "  opt2 = optim.Adam(model2.parameters(), lr=4e-3)\n",
        "  opt1 = optim.Adam(model.parameters(), lr=4e-3)\n",
        "\n",
        "\n",
        "  STEP = 2000\n",
        "  criterion = F.mse_loss\n",
        "  sum_loss = 0\n",
        "\n",
        "  theta = 0.99\n",
        "\n",
        "  mask = torch.tensor([1.414 * (1-theta) ** 0.5, 1.414 * (theta) **0.5]).cuda()\n",
        "  for i in range(STEP):\n",
        "    #print(i)\n",
        "    #if i > 2000:\n",
        "    #  opt = optim.SGD(model.parameters(), lr=2e-4, momentum=0)\n",
        "\n",
        "    opt1.zero_grad()\n",
        "    opt2.zero_grad()\n",
        "\n",
        "    \n",
        "    eps1 = sigma * torch.zeros([N, d]).normal_(0,1).cuda() \n",
        "    output1 = model2((X + eps1 *mask))#/((1 + sigma **2) **2))\n",
        "\n",
        "    eps2 = sigma * torch.zeros([N, d]).normal_(0,1).cuda() \n",
        "    output2 = model2((X + eps2* mask))#/((1 + sigma **2) **2))\n",
        "\n",
        "    entropy = - ((output1.unsqueeze(1) - output2.unsqueeze(0)) ** 2).sum(dim=-1)\n",
        "    #print(entropy.shape)\n",
        "    loss = ((output1 - output2) **2).sum(dim=-1).mean()/T + torch.logsumexp(entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "    #loss = torch.logsumexp(((output1 - output2) **2).sum(dim=-1)/(T * N) + entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "    #print(loss)\n",
        "    loss.backward()\n",
        "    #torch.nn.utils.clip_grad_norm_(model.parameters(), 1)\n",
        "    opt2.step()\n",
        "\n",
        "    if True:\n",
        "      output1 = model((X + eps1))#/((1 + sigma **2) **2))\n",
        "      output2 = model((X + eps2))#/((1 + sigma **2) **2))\n",
        "\n",
        "      entropy = - ((output1.unsqueeze(1) - output2.unsqueeze(0)) ** 2).sum(dim=-1)\n",
        "      #print(entropy.shape)\n",
        "      loss = ((output1 - output2) **2).sum(dim=-1).mean()/T + torch.logsumexp(entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "      #loss = torch.logsumexp(((output1 - output2) **2).sum(dim=-1)/(T * N) + entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "\n",
        "      loss.backward()\n",
        "      #torch.nn.utils.clip_grad_norm_(model.parameters(), 1)\n",
        "      opt1.step()\n",
        "      #if i % 100 ==0:\n",
        "  print(sigma ** 2)\n",
        "\n",
        "  weights.append(model.layers[0].weight.data.cpu().numpy())\n",
        "  losses.append(loss.item())\n",
        "\n",
        "  weights2.append(model2.layers[0].weight.data.cpu().numpy())\n",
        "  losses2.append(loss.item())\n",
        "  #print(weights[-1].shape)\n",
        "  #smax = np.linalg.eigvals(np.matmul(weights[-1], weights[-1].T))\n",
        "  #print(smax)\n",
        "  #smax = svdvals((weights[-1])) ** 2\n",
        "  #print(smax)\n",
        "  with torch.no_grad():\n",
        "    representation = X.cpu().numpy().dot(weights2[-1].T)\n",
        "    representation = representation / (N **0.5) #/ representation.std()\n",
        "    gamma = np.array([[1,0],[0,1]])\n",
        "    A0 = np.matmul(representation.T, representation)  + 1e-3 * gamma\n",
        "\n",
        "    \n",
        "    A0inv = np.linalg.pinv(A0)\n",
        "    \n",
        "    Exy = (representation.T.dot(Y.cpu().numpy())/ (N **0.5))\n",
        "    Ghat = A0inv.dot(Exy)\n",
        "    #print(Ghat)\n",
        "    #plt.hist(smax)\n",
        "    #plt.show()\n",
        "    #plt.close()\n",
        "    #print(((model(Xtest).cpu().numpy().dot(Ghat) - Ytest.cpu().numpy()) ** 2).shape)\n",
        "    testloss = 0#(((model(Xtest)).cpu().numpy().dot(Ghat) - Ytest.cpu().numpy()) ** 2).mean()\n",
        "    #testloss = ((model(Xtest).cpu().numpy().dot(Ghat) - Ytest) ** 2).mean()\n",
        "\n",
        "    Gnp = G.cpu().numpy()\n",
        "    Az = np.matmul(weights2[-1], weights2[-1].T)\n",
        "    test_loss = (Ghat.T.dot(Az.dot(Ghat))) - 2 * ((Ghat.T).dot(weights2[-1])).dot(Gnp)   +  Gnp.T.dot(Gnp)\n",
        "    \n",
        "    test_losses2.append([testloss, test_loss])\n",
        "    norms2.append(np.linalg.norm(Ghat))\n",
        "    print(\"test\", testloss, norms2[-1],test_loss)\n",
        "\n",
        "\n",
        "\n",
        "    representation = X.cpu().numpy().dot(weights[-1].T)\n",
        "    representation = representation /(N **0.5)#/ representation.std()\n",
        "    gamma = np.array([[1,0],[0,1]])\n",
        "    A0 = np.matmul(representation.T, representation)  + 1e-3 * gamma\n",
        "\n",
        "    \n",
        "    A0inv = np.linalg.pinv(A0)\n",
        "    \n",
        "    Exy = (representation.T.dot(Y.cpu().numpy())/((N **0.5)))\n",
        "    Ghat = A0inv.dot(Exy)\n",
        "    #print(Ghat)\n",
        "    #plt.hist(smax)\n",
        "    #plt.show()\n",
        "    #plt.close()\n",
        "    #print(((model(Xtest).cpu().numpy().dot(Ghat) - Ytest.cpu().numpy()) ** 2).shape)\n",
        "    testloss = 0#(((model(Xtest)).cpu().numpy().dot(Ghat) - Ytest.cpu().numpy()) ** 2).mean()\n",
        "    #testloss = ((model(Xtest).cpu().numpy().dot(Ghat) - Ytest) ** 2).mean()\n",
        "\n",
        "    Gnp = G.cpu().numpy()\n",
        "    Az = np.matmul(weights[-1], weights[-1].T)\n",
        "    test_loss = (Ghat.T.dot(Az.dot(Ghat))) - 2 * ((Ghat.T).dot(weights[-1])).dot(Gnp)   +  Gnp.T.dot(Gnp)\n",
        "    \n",
        "    test_losses.append([testloss, test_loss])\n",
        "    norms.append(np.linalg.norm(Ghat))\n",
        "    print(\"test\", testloss, norms[-1],test_loss)\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "gR82VGgcDGsw",
        "outputId": "4930d989-87de-4aba-da07-c917b24fb811"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "0.0\n",
            "test 0 0.016441414566553965 [[0.00156729]]\n",
            "test 0 0.016441414566553965 [[0.00156729]]\n",
            "0.049999999999999996\n",
            "test 0 0.0409128623015865 [[0.00156433]]\n",
            "test 0 0.06626569805213584 [[0.00156597]]\n",
            "0.1\n",
            "test 0 0.06039165320680816 [[0.00156022]]\n",
            "test 0 0.09649236551326226 [[0.00156433]]\n",
            "0.15000000000000002\n",
            "test 0 0.08062783725173628 [[0.00155427]]\n",
            "test 0 0.12289628549396141 [[0.00156229]]\n",
            "0.19999999999999998\n",
            "test 0 0.10329007873465418 [[0.00154578]]\n",
            "test 0 0.1483805723986746 [[0.00155974]]\n",
            "0.25\n",
            "test 0 0.13016950777924227 [[0.00153237]]\n",
            "test 0 0.17424130161661952 [[0.00155655]]\n",
            "0.3000000000000001\n",
            "test 0 0.16450345391079488 [[0.00151092]]\n",
            "test 0 0.20134259214639202 [[0.00155255]]\n",
            "0.35\n",
            "test 0 0.21335863872648292 [[0.00147151]]\n",
            "test 0 0.2304315545950047 [[0.00154748]]\n",
            "0.4\n",
            "test 0 0.2940348323426142 [[0.00138038]]\n",
            "test 0 0.2623006290747428 [[0.00154098]]\n",
            "0.45\n",
            "test 0 0.5215132142927816 [[0.00090517]]\n",
            "test 0 0.29792562416213914 [[0.00153252]]\n",
            "0.5000000000000001\n",
            "test 0 0.025316806210510162 [[0.00012144]]\n",
            "test 0 0.33863828176710314 [[0.00152128]]\n",
            "0.55\n",
            "test 0 0.026504080082858126 [[0.00012146]]\n",
            "test 0 0.3864961372276331 [[0.0015059]]\n",
            "0.6000000000000001\n",
            "test 0 0.02765353386442374 [[0.00012149]]\n",
            "test 0 0.4447266909001329 [[0.00148437]]\n",
            "0.6500000000000001\n",
            "test 0 0.02876135407516216 [[0.00012152]]\n",
            "test 0 0.5189846936604737 [[0.00145252]]\n",
            "0.7000000000000001\n",
            "test 0 0.029834965220187682 [[0.00012155]]\n",
            "test 0 0.620030396010021 [[0.00139912]]\n",
            "0.7499999999999999\n",
            "test 0 0.03087258992968445 [[0.00012158]]\n",
            "test 0 0.7723830860225147 [[0.00130131]]\n",
            "0.7999999999999999\n",
            "test 0 0.03187850351862407 [[0.00012161]]\n",
            "test 0 1.217508054374261 [[0.00105206]]\n",
            "0.8500000000000001\n",
            "test 0 0.0328552234658237 [[0.00012163]]\n",
            "test 0 0.2637728557246028 [[0.00602968]]\n",
            "0.8999999999999999\n",
            "test 0 0.03380579671051083 [[0.00012165]]\n",
            "test 0 0.30200472485936264 [[0.00604241]]\n",
            "0.9500000000000001\n",
            "test 0 0.03473306780700717 [[0.00012166]]\n",
            "test 0 0.36755955705629284 [[0.00606346]]\n",
            "1.0\n",
            "test 0 0.0356372083217915 [[0.00012167]]\n",
            "test 0 0.5637012482926547 [[0.005997]]\n",
            "1.0500000000000003\n",
            "test 0 0.036521382563093834 [[0.00012167]]\n",
            "test 0 0.789171575975573 [[0.00640541]]\n",
            "1.1\n",
            "test 0 0.03738658604620365 [[0.00012167]]\n",
            "test 0 0.021803208015054386 [[0.00999841]]\n",
            "1.1500000000000001\n",
            "test 0 0.038233587716293115 [[0.00012166]]\n",
            "test 0 6.904315264545438e-11 [[0.01]]\n",
            "1.2000000000000004\n",
            "test 0 0.0390646674007975 [[0.00012165]]\n",
            "test 0 0.24973384515588185 [[0.00986354]]\n",
            "1.2500000000000002\n",
            "test 0 0.03988010925091702 [[0.00012163]]\n",
            "test 0 1.4681124699325133e-10 [[0.01]]\n",
            "1.3000000000000003\n",
            "test 0 0.04068091597434184 [[0.00012161]]\n",
            "test 0 4.2294119926686915e-19 [[0.01]]\n",
            "1.35\n",
            "test 0 0.04146810256286932 [[0.00012158]]\n",
            "test 0 9.372164536400684e-21 [[0.01]]\n",
            "1.4\n",
            "test 0 0.042242436113901924 [[0.00012155]]\n",
            "test 0 2.80570174810415e-21 [[0.01]]\n",
            "1.4500000000000002\n",
            "test 0 0.043004587253157096 [[0.00012151]]\n",
            "test 0 2.7949143528753504e-24 [[0.01]]\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## generalization of beta infonce\n",
        "\n",
        "\n",
        "exp = np.arange(1, 14)\n",
        "#width_range = 2 ** exp\n",
        "\n",
        "\n",
        "losses = []\n",
        "norms = []\n",
        "weights = []\n",
        "test_losses3 = []\n",
        "\n",
        "\n",
        "\n",
        "sigma_list =  np.arange(0, 1.5, 0.05) ** 0.5\n",
        "#T_list =  np.arange(0.1, 1,0.1)\n",
        "#sigma = 1\n",
        "T = 0.5\n",
        "\n",
        "for sigma in sigma_list:\n",
        "  torch.manual_seed(2)\n",
        "  model2 = Net(depth=0).cuda()\n",
        "  torch.manual_seed(2)\n",
        "  model = Net(depth=0).cuda()\n",
        "\n",
        "  lr = 0.0015 #/ (width ** 0.2)\n",
        "  #opt = optim.SGD(model.parameters(), lr=lr, momentum=0)\n",
        "  opt2 = optim.Adam(model2.parameters(), lr=4e-3)\n",
        "  opt1 = optim.Adam(model.parameters(), lr=4e-3)\n",
        "\n",
        "\n",
        "  STEP = 2000\n",
        "  criterion = F.mse_loss\n",
        "  sum_loss = 0\n",
        "\n",
        "  theta = 0.63\n",
        "\n",
        "  mask = torch.tensor([1.414 * (1-theta) ** 0.5, 1.414 * (theta) **0.5]).cuda()\n",
        "  for i in range(STEP):\n",
        "    #print(i)\n",
        "    #if i > 2000:\n",
        "    #  opt = optim.SGD(model.parameters(), lr=2e-4, momentum=0)\n",
        "\n",
        "    opt1.zero_grad()\n",
        "    opt2.zero_grad()\n",
        "\n",
        "    \n",
        "    eps1 = sigma * torch.zeros([N, d]).normal_(0,1).cuda() \n",
        "    output1 = model2((X + eps1 *mask))#/((1 + sigma **2) **2))\n",
        "\n",
        "    eps2 = sigma * torch.zeros([N, d]).normal_(0,1).cuda() \n",
        "    output2 = model2((X + eps2* mask))#/((1 + sigma **2) **2))\n",
        "\n",
        "    entropy = - ((output1.unsqueeze(1) - output2.unsqueeze(0)) ** 2).sum(dim=-1)\n",
        "    #print(entropy.shape)\n",
        "    loss = ((output1 - output2) **2).sum(dim=-1).mean()/T + torch.logsumexp(entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "    #loss = torch.logsumexp(((output1 - output2) **2).sum(dim=-1)/(T * N) + entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "    #print(loss)\n",
        "    loss.backward()\n",
        "    #torch.nn.utils.clip_grad_norm_(model.parameters(), 1)\n",
        "    opt2.step()\n",
        "\n",
        "    if False:\n",
        "      output1 = model((X + eps1))#/((1 + sigma **2) **2))\n",
        "      output2 = model((X + eps2))#/((1 + sigma **2) **2))\n",
        "\n",
        "      entropy = - ((output1.unsqueeze(1) - output2.unsqueeze(0)) ** 2).sum(dim=-1)\n",
        "      #print(entropy.shape)\n",
        "      loss = ((output1 - output2) **2).sum(dim=-1).mean()/T + torch.logsumexp(entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "      #loss = torch.logsumexp(((output1 - output2) **2).sum(dim=-1)/(T * N) + entropy, dim=-1).mean()# (entropy.exp().mean(dim=-1) + 1e-8).log().mean()\n",
        "\n",
        "      loss.backward()\n",
        "      #torch.nn.utils.clip_grad_norm_(model.parameters(), 1)\n",
        "      opt1.step()\n",
        "      #if i % 100 ==0:\n",
        "  print(sigma ** 2)\n",
        "\n",
        "  weights.append(model.layers[0].weight.data.cpu().numpy())\n",
        "  losses.append(loss.item())\n",
        "\n",
        "  weights2.append(model2.layers[0].weight.data.cpu().numpy())\n",
        "  losses2.append(loss.item())\n",
        "  #print(weights[-1].shape)\n",
        "  #smax = np.linalg.eigvals(np.matmul(weights[-1], weights[-1].T))\n",
        "  #print(smax)\n",
        "  #smax = svdvals((weights[-1])) ** 2\n",
        "  #print(smax)\n",
        "  with torch.no_grad():\n",
        "    representation = X.cpu().numpy().dot(weights2[-1].T)\n",
        "    representation = representation / (N **0.5) #/ representation.std()\n",
        "    gamma = np.array([[1,0],[0,1]])\n",
        "    A0 = np.matmul(representation.T, representation)  + 1e-3 * gamma\n",
        "\n",
        "    \n",
        "    A0inv = np.linalg.pinv(A0)\n",
        "    \n",
        "    Exy = (representation.T.dot(Y.cpu().numpy())/ (N **0.5))\n",
        "    Ghat = A0inv.dot(Exy)\n",
        "    #print(Ghat)\n",
        "    #plt.hist(smax)\n",
        "    #plt.show()\n",
        "    #plt.close()\n",
        "    #print(((model(Xtest).cpu().numpy().dot(Ghat) - Ytest.cpu().numpy()) ** 2).shape)\n",
        "    testloss = 0#(((model(Xtest)).cpu().numpy().dot(Ghat) - Ytest.cpu().numpy()) ** 2).mean()\n",
        "    #testloss = ((model(Xtest).cpu().numpy().dot(Ghat) - Ytest) ** 2).mean()\n",
        "\n",
        "    Gnp = G.cpu().numpy()\n",
        "    Az = np.matmul(weights2[-1], weights2[-1].T)\n",
        "    test_loss = (Ghat.T.dot(Az.dot(Ghat))) - 2 * ((Ghat.T).dot(weights2[-1])).dot(Gnp)   +  Gnp.T.dot(Gnp)\n",
        "    \n",
        "    test_losses3.append([testloss, test_loss])\n",
        "    norms2.append(np.linalg.norm(Ghat))\n",
        "    print(\"test\", testloss, norms2[-1],test_loss)\n",
        "\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "ImaMzbUsEyWK",
        "outputId": "63ec05a3-98f2-4cf2-cc57-69fc02b8d528"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "0.0\n",
            "test 0 0.016441414566553965 [[0.00156729]]\n",
            "0.049999999999999996\n",
            "test 0 0.05962565043102194 [[0.0015656]]\n",
            "0.1\n",
            "test 0 0.0865870929030142 [[0.00156346]]\n",
            "0.15000000000000002\n",
            "test 0 0.10993527520649322 [[0.00156068]]\n",
            "0.19999999999999998\n",
            "test 0 0.1322181293763771 [[0.0015571]]\n",
            "0.25\n",
            "test 0 0.15458719403789853 [[0.00155248]]\n",
            "0.3000000000000001\n",
            "test 0 0.17782059489599836 [[0.00154649]]\n",
            "0.35\n",
            "test 0 0.20260500390685451 [[0.00153863]]\n",
            "0.4\n",
            "test 0 0.2297061428814167 [[0.00152812]]\n",
            "0.45\n",
            "test 0 0.26016914721219525 [[0.00151367]]\n",
            "0.5000000000000001\n",
            "test 0 0.29573297905522017 [[0.00149294]]\n",
            "0.55\n",
            "test 0 0.3391990077170186 [[0.00146117]]\n",
            "0.6000000000000001\n",
            "test 0 0.39659079860303775 [[0.00140757]]\n",
            "0.6500000000000001\n",
            "test 0 0.48112171144636295 [[0.00130315]]\n",
            "0.7000000000000001\n",
            "test 0 0.6675633215260633 [[0.00095226]]\n",
            "0.7499999999999999\n",
            "test 0 0.28770499070473227 [[0.00052666]]\n",
            "0.7999999999999999\n",
            "test 0 0.31204817788691325 [[0.00051382]]\n",
            "0.8500000000000001\n",
            "test 0 0.34237045103292046 [[0.0005008]]\n",
            "0.8999999999999999\n",
            "test 0 0.378744941525258 [[0.00048946]]\n",
            "0.9500000000000001\n",
            "test 0 0.42339973198957215 [[0.00048323]]\n",
            "1.0\n",
            "test 0 0.47187629041879137 [[0.00049537]]\n",
            "1.0500000000000003\n",
            "test 0 0.516357052826165 [[0.00051547]]\n",
            "1.1\n",
            "test 0 0.5818788470376342 [[0.00054293]]\n",
            "1.1500000000000001\n",
            "test 0 0.6733035392097485 [[0.00058134]]\n",
            "1.2000000000000004\n",
            "test 0 0.8001359026356915 [[0.00069029]]\n",
            "1.2500000000000002\n",
            "test 0 1.004510000126765 [[0.00104892]]\n",
            "1.3000000000000003\n",
            "test 0 1.1146370942247659 [[0.00627827]]\n",
            "1.35\n",
            "test 0 0.060430954621370644 [[0.00999122]]\n",
            "1.4\n",
            "test 0 0.00037706381369932875 [[0.01]]\n",
            "1.4500000000000002\n",
            "test 0 0.00023409096020765872 [[0.01]]\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "test_losses = np.array(test_losses)\n",
        "test_losses2 = np.array(test_losses2)\n",
        "test_losses3 = np.array(test_losses3)\n",
        "\n",
        "plt.figure(figsize=(3,2.5))\n",
        "for s in sin:\n",
        "  print(s)\n",
        "  plt.axvline(x=2 * s, linestyle='--', c='black')\n",
        "plt.plot(2 * sigma_list ** 2, (test_losses2[:, 1]), marker=\">\", lw=2, label=r'$\\theta=1$', markersize=4)\n",
        "plt.plot(2 * sigma_list ** 2, (test_losses3[:, 1]), marker=\"<\", lw=2, label=r'$\\theta=0.65$', markersize=4)\n",
        "\n",
        "plt.plot(2 * sigma_list ** 2, (test_losses[:, 1]), marker=\"^\", lw=2, label=r'$\\theta=0.5$', markersize=4)\n",
        "plt.axvline(x=s, linestyle='--', c='gray')\n",
        "\n",
        "plt.yscale('log')\n",
        "#plt.xlim(1.0, 1.1)\n",
        "plt.ylim(1e-4, 2e-2)\n",
        "plt.ylabel('test loss')\n",
        "plt.xlabel(r'$\\sigma^2$')\n",
        "plt.legend(loc='upper left')\n",
        "plt.savefig(\"test_loss_comparison3.png\", dpi=250, bbox_inches='tight')"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 294
        },
        "id": "R5Lpa_reDLbC",
        "outputId": "fed62d6b-ecda-435b-f089-c2acbcd1d05d"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "0.81168836\n",
            "1.0896001\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.7/dist-packages/ipykernel_launcher.py:3: VisibleDeprecationWarning: Creating an ndarray from ragged nested sequences (which is a list-or-tuple of lists-or-tuples-or ndarrays with different lengths or shapes) is deprecated. If you meant to do this, you must specify 'dtype=object' when creating the ndarray.\n",
            "  This is separate from the ipykernel package so we can avoid doing imports until\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 216x180 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {
            "needs_background": "light"
          }
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "dkinrtDlExCS"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}