{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "4cd71d5b",
   "metadata": {},
   "source": [
    "# Code for  \"Three ways that non-differentiability affects neural network training\"\n",
    "\n",
    "The graphs will be slightly different on each run since the results are being demonstrated for randomly generated data. Nevertheless, the key features described in the paper will be see on every run.\n",
    "\n",
    "We run most pieces on CPU because the runs are so fast as is. The slowest code to run is the histogram computation in Figure 2, which has been commented out because it is not essential for the verification of correctness. This code is slow because of the cost of bin computation of the histogram, and would not benefit from the use of GPUs "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "fef3d67c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: matplotlib in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (3.3.4)\n",
      "Requirement already satisfied: cycler>=0.10 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from matplotlib) (0.10.0)\n",
      "Requirement already satisfied: kiwisolver>=1.0.1 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from matplotlib) (1.3.1)\n",
      "Requirement already satisfied: python-dateutil>=2.1 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from matplotlib) (2.8.1)\n",
      "Requirement already satisfied: pillow>=6.2.0 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from matplotlib) (8.4.0)\n",
      "Requirement already satisfied: numpy>=1.15 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from matplotlib) (1.19.5)\n",
      "Requirement already satisfied: pyparsing!=2.0.4,!=2.1.2,!=2.1.6,>=2.0.3 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from matplotlib) (2.4.7)\n",
      "Requirement already satisfied: six in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from cycler>=0.10->matplotlib) (1.15.0)\n",
      "Requirement already satisfied: numpy in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (1.19.5)\n",
      "Requirement already satisfied: pandas in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (1.1.5)\n",
      "Requirement already satisfied: python-dateutil>=2.7.3 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from pandas) (2.8.1)\n",
      "Requirement already satisfied: pytz>=2017.2 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from pandas) (2021.1)\n",
      "Requirement already satisfied: numpy>=1.15.4 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from pandas) (1.19.5)\n",
      "Requirement already satisfied: six>=1.5 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from python-dateutil>=2.7.3->pandas) (1.15.0)\n",
      "Requirement already satisfied: torch in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (1.10.1)\n",
      "Requirement already satisfied: typing-extensions in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torch) (4.0.1)\n",
      "Requirement already satisfied: dataclasses in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torch) (0.8)\n",
      "Requirement already satisfied: torchvision in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (0.11.2)\n",
      "Requirement already satisfied: torch==1.10.1 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torchvision) (1.10.1)\n",
      "Requirement already satisfied: pillow!=8.3.0,>=5.3.0 in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torchvision) (8.4.0)\n",
      "Requirement already satisfied: numpy in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torchvision) (1.19.5)\n",
      "Requirement already satisfied: typing-extensions in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torch==1.10.1->torchvision) (4.0.1)\n",
      "Requirement already satisfied: dataclasses in /home/ec2-user/anaconda3/envs/python3/lib/python3.6/site-packages (from torch==1.10.1->torchvision) (0.8)\n"
     ]
    }
   ],
   "source": [
    "!pip install matplotlib\n",
    "!pip install numpy\n",
    "!pip install pandas\n",
    "!pip install torch\n",
    "!pip install torchvision"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "994d52f6",
   "metadata": {},
   "source": [
    "# Utils. \n",
    "\n",
    "1) `generate_dummy_data` - Code to generate data with randomly initialized entries in a specified range\n",
    "2) `runner` - Code to run SGD on a regularized NN of given specications\n",
    "3) `BaseNN` - Single layer FC NN with ReLU activations described in the preliminaries of the paper\n",
    "4) `LinearReg` - Basic linear regression model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "d60eca4f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from torch.utils.data import TensorDataset, DataLoader\n",
    "import torch.optim as optim\n",
    "\n",
    "# Set the seed for weight initialization (you can use any seed value)\n",
    "seed = 42\n",
    "torch.manual_seed(seed)\n",
    "\n",
    "\n",
    "def generate_dummy_data(x_shape=(20,500), x_range=(-1, 1), y_range=(-1, 1)):\n",
    "    '''\n",
    "    Generates training data based on shape of data matrix, and range of values for\n",
    "    data matrix and response vector. The shape of the response vector is (x_shape[0],1)\n",
    "\n",
    "    :param x_shape: Tuple describing shape of data matrix\n",
    "    :param x_range: Range of values for entries in data matrix\n",
    "    :param y_range: Range of values for response vector\n",
    "    :return:\n",
    "    '''\n",
    "    x_train = np.random.uniform(*x_range, size=x_shape)\n",
    "    y_train = np.random.uniform(*y_range, size=(x_shape[0], 1))\n",
    "    return torch.Tensor(x_train), torch.Tensor(y_train)\n",
    "\n",
    "def runner(model, x, y, lr=0.01, max_iter=200, l2_penalty=0.001, l1_penalty=0.01, criterion = nn.MSELoss(reduction = 'sum')):\n",
    "    '''\n",
    "    Returns the list of weights and losses for each iteration when running SGD on the model with given x and y\n",
    "\n",
    "    :param model: model\n",
    "    :param x: data matrix\n",
    "    :param y: response vector\n",
    "    :param lr: learning rate\n",
    "    :param max_iter: number of iterations\n",
    "    :param l2_penalty: lasso penalty\n",
    "    :param l1_penalty: ridge penalty\n",
    "    :param criterion: loss criterion to be used in training\n",
    "    :return: (losses, weights)\n",
    "    losses -> List[Float]\n",
    "    weights -> List[List[Float]]\n",
    "    '''\n",
    "    optimizer = optim.SGD(model.parameters(), lr=lr)\n",
    "    losses = []\n",
    "    weights = []\n",
    "    for it in range(max_iter):\n",
    "        out = model(x)\n",
    "        loss = criterion(out, y)\n",
    "        wt_sqrd = model.linear.weight.pow(2).sum()\n",
    "        l2_norm = l2_penalty * wt_sqrd\n",
    "        l1_norm = l1_penalty * torch.norm(model.linear.weight, p=1)\n",
    "        loss += l2_norm + l1_norm\n",
    "        losses.append(loss.detach().item())\n",
    "        weights.append(np.copy(model.linear.weight.detach().numpy()))\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        optimizer.zero_grad()\n",
    "    return losses, weights\n",
    "\n",
    "\n",
    "\n",
    "class BaseNN(torch.nn.Module):\n",
    "    '''\n",
    "    Simple single layer NN used throughout our analysis. The weights are initialized as U[init_low, init_high] \n",
    "    \n",
    "    :param: input_size - number of neurons in hidden layer\n",
    "    :param: init_low - lower bound of weight init\n",
    "    :param: init_high - upper bound of weight init\n",
    "    \n",
    "    '''\n",
    "    def __init__(self, input_size,init_low = -0.99, init_high = 0.99):\n",
    "        super(BaseNN, self).__init__()\n",
    "        self.linear = torch.nn.Linear(input_size, 1, bias=False)\n",
    "        torch.nn.init.uniform_(self.linear.weight,init_low,init_high)\n",
    "        self.relu = torch.nn.ReLU()\n",
    "\n",
    "    def forward(self,x):\n",
    "        return self.relu(self.linear(x))\n",
    "\n",
    "    \n",
    "class LinearReg(torch.nn.Module):\n",
    "    '''\n",
    "    Basic linear regression module\n",
    "\n",
    "    '''\n",
    "    def __init__(self, input_size,init_low = -0.99, init_high = 0.99):\n",
    "        super(LinearReg, self).__init__()\n",
    "        self.linear = torch.nn.Linear(input_size, 1, bias=False)\n",
    "        torch.nn.init.uniform_(self.linear.weight,init_low,init_high)\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.linear(x)    "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d19fc496",
   "metadata": {},
   "source": [
    "## Section 3 - NGD and Convergence Analysis - Code for Figure 1 "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1ddc0905",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[Text(0.5, 0, 'Iteration number'), Text(0, 0.5, 'Infinity norm')]"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib\n",
    "\n",
    "nrows = 20\n",
    "ncols = 500\n",
    "l1 = 0\n",
    "l2 = 0.01\n",
    "\n",
    "x,y=generate_dummy_data(x_shape = (nrows,ncols), x_range = (-1,1), y_range = (-5,-4))\n",
    "\n",
    "model = BaseNN(x.shape[1],init_low = -0.002, init_high = -.001) \n",
    "\n",
    "losses, weights  = runner(model = model,x = x,y = y,l1_penalty = 0, l2_penalty = l2, max_iter = 200)\n",
    "\n",
    "inf_norm = [np.linalg.norm(x[0],np.inf) for x in weights]\n",
    "\n",
    "ticks = list(range(0,len(losses)))\n",
    "\n",
    "fig, (ax1, ax2) = plt.subplots(2)\n",
    "matplotlib.rcParams.update({'font.size': 14})\n",
    "ax1.plot(ticks,losses, 'bo')\n",
    "ax1.set(xlabel=\"Iteration number\", ylabel=\"Loss\")\n",
    "ax2.plot(ticks, inf_norm, 'ro')\n",
    "ax2.set(xlabel=\"Iteration number\", ylabel=\"Infinity norm\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eb1739c1",
   "metadata": {},
   "source": [
    "### Section 3 - Code for figure 2\n",
    "\n",
    "As mentioned in paper, our simulations are performed using the code provided in https://github.com/locuslab/edge-of-stability/blob/github/src/archs.py. We add a few lines of code to theirs, and run the script provided in their documentation to generate the training curves. We provide the steps for how to generate the training curves for FC tanh and relu networks of depth 6. The implementations for the rest of the curves are similar.\n",
    "\n",
    "1) Add the following funtions to  https://github.com/locuslab/edge-of-stability/blob/github/src/utilities.py\n",
    "\n",
    "```\n",
    "def generate_fully_connected_nn(activation, depth = 5):\n",
    "    return fully_connected_net(dataset_name, [200]*depth, activation, bias=True)\n",
    "\n",
    "def generate_mini_vgg(activation, depth = 5):\n",
    "    return convnet(dataset_name, [32] * depth, activation=activation, pooling='average', bias=True)\n",
    "```\n",
    "\n",
    "2) Add the architecture to the `load_architecture` function in https://github.com/locuslab/edge-of-stability/blob/github/src/archs.py#L119 by adding another elif branch\n",
    "\n",
    "```\n",
    "def load_architecture(arch_id: str, dataset_name: str) -> nn.Module:\n",
    "    #  ======   fully-connected networks =======\n",
    "    if arch_id == 'fc-relu':\n",
    "        return fully_connected_net(dataset_name, [200, 200], 'relu', bias=True)\n",
    "    elif arch_id == 'fc-elu':\n",
    "        return fully_connected_net(dataset_name, [200, 200], 'elu', bias=True)\n",
    "    ...\n",
    "    ...\n",
    "    elif arch_id == 'fc-relu-depth6':\n",
    "        generate_fully_connected_nn('relu',6)\n",
    "    elif arch_id == 'fc-tanh-depth6':\n",
    "        generate_fully_connected_nn('tanh',6)\n",
    "    ...\n",
    "    ...\n",
    "```\n",
    "\n",
    "3) Run their scripts, and plot the training losses. The scripts print the training losses on the command line, which we then plot. \n",
    "\n",
    "```\n",
    "> python3 src/gd.py cifar10-2k fc-relu-depth6  ce 0.05 200 --acc_goal 0.99 --neigs 2  --eig_freq 200 --save_freq 200 --seed=123             \n",
    "\n",
    "\n",
    "> python3 src/gd.py cifar10-2k fc-tanh-depth6  ce 0.05 200 --acc_goal 0.99 --neigs 2  --eig_freq 200 --save_freq 200 --seed=123             \n",
    "\n",
    "```\n",
    "\n",
    "      "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "61fff7c6",
   "metadata": {},
   "source": [
    "## Section 4 - NGD and the LASSO penalty - larger L1 penalty results in larger L1 norm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "afbd8f54",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "L1 norm of wts with (lambda1 = 0.01) = 1.9065666198730469, and (lambda1 = 10) = 25.81937026977539\n"
     ]
    }
   ],
   "source": [
    "x,y = generate_dummy_data(x_shape = (20,500), x_range = (-1,1), y_range = (-1,1))\n",
    "\n",
    "model = LinearReg(x.shape[1])\n",
    "\n",
    "def get_l1_norm_of_wts(lambda1, model = model, x = x, y = y, max_iter = 20000):\n",
    "    '''\n",
    "    Computes the L1 norm of the weights at the end of a training run with the specified LASSO penalty,\n",
    "    model, data and total number of iterations\n",
    "    '''\n",
    "    l , w = runner(model = model, \n",
    "                   x=x, \n",
    "                   y=y, \n",
    "                   l2_penalty = 0, \n",
    "                   l1_penalty = lambda1,\n",
    "                   max_iter = max_iter, \n",
    "                   criterion = nn.MSELoss(reduction = 'mean'))\n",
    "    return np.linalg.norm(w[-1][0], 1)\n",
    "    \n",
    "    \n",
    "print(\"L1 norm of wts with (lambda1 = 0.01) = {0}, and (lambda1 = 10) = {1}\".format(\n",
    "    get_l1_norm_of_wts(0.01),\n",
    "    get_l1_norm_of_wts(10)))\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e5f23fb2",
   "metadata": {},
   "source": [
    "## Section 4 - NGD and the LASSO - Code for comparing L1 norm of solution with larger v/s smaller LASSO penalty\n",
    "\n",
    "First we run training with a LASSO penalty of 0.001., and then we re-run the same code with a LASSO penalty of 0.1 We have setup the code for the GD and Adam. The only things which needs to be changed for the two runs is the LASSO penalty, and the learning rate. A sample run with Adam is provided below"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "668fd29e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Epoch [1/25], Loss: 28049.0172\n",
      "Epoch [2/25], Loss: 3626.7292\n",
      "Epoch [3/25], Loss: 1181.9200\n",
      "Epoch [4/25], Loss: 769.9305\n",
      "Epoch [5/25], Loss: 653.6696\n",
      "Epoch [6/25], Loss: 604.2276\n",
      "Epoch [7/25], Loss: 578.6502\n",
      "Epoch [8/25], Loss: 563.2060\n",
      "Epoch [9/25], Loss: 553.2772\n",
      "Epoch [10/25], Loss: 546.5567\n",
      "Epoch [11/25], Loss: 541.7991\n",
      "Epoch [12/25], Loss: 538.1229\n",
      "Epoch [13/25], Loss: 535.2629\n",
      "Epoch [14/25], Loss: 533.1936\n",
      "Epoch [15/25], Loss: 531.6098\n",
      "Epoch [16/25], Loss: 529.8622\n",
      "Epoch [17/25], Loss: 528.6344\n",
      "Epoch [18/25], Loss: 527.7127\n",
      "Epoch [19/25], Loss: 526.9920\n",
      "Epoch [20/25], Loss: 525.8468\n",
      "Epoch [21/25], Loss: 525.5014\n",
      "Epoch [22/25], Loss: 524.8779\n",
      "Epoch [23/25], Loss: 524.4542\n",
      "Epoch [24/25], Loss: 523.8086\n",
      "Epoch [25/25], Loss: 523.4718\n",
      "Finished Training, L1 norm: 5151.16552734375\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torchvision\n",
    "from torchvision import transforms\n",
    "from torch.utils.data import DataLoader\n",
    "from torchvision.datasets import CIFAR10\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "\n",
    "\n",
    "torch.manual_seed(0)\n",
    "\n",
    "# Define transforms for preprocessing\n",
    "transform = transforms.Compose([\n",
    "    transforms.Resize(256),\n",
    "    transforms.CenterCrop(224),\n",
    "    transforms.ToTensor(),\n",
    "    transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))  # Assuming CIFAR-10 has 3 channels (RGB)\n",
    "])\n",
    "\n",
    "cifar10_train = CIFAR10(root='./data', train=True, download=True, transform=transform)\n",
    "cifar10_subset = torch.utils.data.Subset(cifar10_train, range(2000))\n",
    "\n",
    "# Create DataLoader\n",
    "batch_size = 32\n",
    "train_loader = DataLoader(cifar10_subset, batch_size=batch_size, shuffle=False)\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "# Load pre-trained VGG16 model\n",
    "vgg16 = torchvision.models.vgg16(pretrained=True)\n",
    "vgg16.to(device)\n",
    "\n",
    "\n",
    "def compute_L1_norm(model, device=device):\n",
    "    l1_reg = torch.tensor(0., device=device)\n",
    "    for param in model.parameters():\n",
    "        l1_reg += torch.norm(param, p=1)\n",
    "    return l1_reg\n",
    "\n",
    "# Define L1-penalized loss function\n",
    "class CustomLoss(nn.Module):\n",
    "    def __init__(self, l1_penalty):\n",
    "        super(CustomLoss, self).__init__()\n",
    "        self.l1_penalty = l1_penalty\n",
    "        self.criterion = nn.CrossEntropyLoss()\n",
    "\n",
    "    def forward(self, outputs, labels, model):\n",
    "        ce_loss = self.criterion(outputs, labels)  # Cross Entropy loss\n",
    "\n",
    "        # Calculate L1 penalty\n",
    "        l1_reg = compute_L1_norm(model, outputs.device)\n",
    "\n",
    "        # Combine Cross Entropy loss and L1 penalty\n",
    "        loss = ce_loss + self.l1_penalty * l1_reg\n",
    "        return loss\n",
    "\n",
    "# Define loss function and optimizer\n",
    "l1_penalty_value = 0.1  # Set L1 penalty value, change to 0.001 in other cycle\n",
    "criterion = CustomLoss(l1_penalty=l1_penalty_value)\n",
    "\n",
    "def get_optimizer(model, optimizer = 'sgd', lr = 0.003):\n",
    "    if optimizer == 'sgd':\n",
    "        return optim.SGD(model.parameters(), lr=lr)\n",
    "    elif optimizer == 'adam':\n",
    "        return optim.Adam(model.parameters(), lr=lr)\n",
    "    else:\n",
    "        raise NotImplementedError(\"Currently only adam and sgd are implemented\")\n",
    "\n",
    "optimizer = get_optimizer(vgg16, 'adam', 0.0003)\n",
    "\n",
    "# Training loop\n",
    "num_epochs = 25\n",
    "\n",
    "for epoch in range(num_epochs):\n",
    "    running_loss = 0.0\n",
    "    for i, data in enumerate(train_loader, 0):\n",
    "        inputs, labels = data[0].to(device), data[1].to(device)\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "\n",
    "        # Forward pass\n",
    "        outputs = vgg16(inputs)\n",
    "        loss = criterion(outputs, labels, vgg16)\n",
    "\n",
    "        # Backward pass and optimize\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        running_loss += loss.item()\n",
    "\n",
    "    print(f'Epoch [{epoch + 1}/{num_epochs}], Loss: {running_loss / len(train_loader):.4f}')\n",
    "\n",
    "print(f'Finished Training, L1 norm: {compute_L1_norm(vgg16)}')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9f1cb2c6",
   "metadata": {},
   "source": [
    "## Section 5 - NGD and the Edge of Stability - Code for Figure 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "26c9300c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7efd616b8d30>]"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "x,y = generate_dummy_data(x_shape = (20,500), x_range = (-1,1), y_range = (-1,1))\n",
    "\n",
    "model = LinearReg(x.shape[1])\n",
    "\n",
    "'''\n",
    "Script to compute training curves with the Huber Loss function for learning rates = 0.1, 1 and 10 respectively\n",
    "'''\n",
    "lrs = [0.1,1,10]\n",
    "losses = []\n",
    "for lr in lrs:\n",
    "    l, w = runner(model = model, \n",
    "                   x=x, \n",
    "                   y=y, \n",
    "                   lr=lr,\n",
    "                   l2_penalty = 0, \n",
    "                   l1_penalty = 0,\n",
    "                   max_iter = 50, \n",
    "                   criterion = nn.HuberLoss())\n",
    "    losses.append(l)\n",
    "\n",
    "fig, (ax1, ax2,ax3) = plt.subplots(3)\n",
    "ax1.plot(losses[0])\n",
    "ax2.plot(losses[1])\n",
    "ax3.plot(losses[2])\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "conda_python3",
   "language": "python",
   "name": "conda_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.6.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
