{
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "89hdk3d7N-hb"
      },
      "source": [
        "##### Copyright 2020 Google LLC."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 1,
      "metadata": {
        "id": "IbX4a6NKOBTr"
      },
      "outputs": [],
      "source": [
        "#@title Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "# https://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License."
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "XYNno0xtOFJ-"
      },
      "source": [
        "<a href=\"https://colab.research.google.com/github/google-research/simclr/blob/master/colabs/distillation_self_training.ipynb\" target=\"_parent\"><img src=\"https://colab.research.google.com/assets/colab-badge.svg\" alt=\"Open In Colab\"/></a>\n"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "I9oIl-rCOypT"
      },
      "source": [
        "## SimCLR: A Simple Framework for Contrastive Learning of Visual Representations\n",
        "\n",
        "This colab demonstrates how to load pretrained/finetuned SimCLR models from hub modules for distillation / self-training (without needing actual labels).\n",
        "\n",
        "The checkpoints are accessible in the following Google Cloud Storage folders.\n",
        "\n",
        "* Pretrained SimCLRv2 models with a linear classifier: [gs://simclr-checkpoints/simclrv2/pretrained](https://console.cloud.google.com/storage/browser/simclr-checkpoints/simclrv2/pretrained)\n",
        "* Fine-tuned SimCLRv2 models on 1% of labels: [gs://simclr-checkpoints/simclrv2/finetuned_1pct](https://console.cloud.google.com/storage/browser/simclr-checkpoints/simclrv2/finetuned_1pct)\n",
        "* Fine-tuned SimCLRv2 models on 10% of labels: [gs://simclr-checkpoints/simclrv2/finetuned_10pct](https://console.cloud.google.com/storage/browser/simclr-checkpoints/simclrv2/finetuned_10pct)\n",
        "* Fine-tuned SimCLRv2 models on 100% of labels: [gs://simclr-checkpoints/simclrv2/finetuned_100pct](https://console.cloud.google.com/storage/browser/simclr-checkpoints/simclrv2/finetuned_100pct)\n",
        "* Supervised models with the same architectures: [gs://simclr-checkpoints/simclrv2/pretrained](https://console.cloud.google.com/storage/browser/simclr-checkpoints/simclrv2/pretrained)\n",
        "\n",
        "Use the corresponding checkpoint / hub-module paths for accessing the model. For example, to use a pre-trained model (with a linear classifier) with ResNet-152 (2x+SK), set the path to `gs://simclr-checkpoints/simclrv2/pretrained/r152_2x_sk1`."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 2,
      "metadata": {
        "id": "Ih5NlvdDEOI1"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "2024-05-21 23:09:17.947156: I external/local_tsl/tsl/cuda/cudart_stub.cc:31] Could not find cuda drivers on your machine, GPU will not be used.\n",
            "2024-05-21 23:09:18.133509: E external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:9261] Unable to register cuDNN factory: Attempting to register factory for plugin cuDNN when one has already been registered\n",
            "2024-05-21 23:09:18.133580: E external/local_xla/xla/stream_executor/cuda/cuda_fft.cc:607] Unable to register cuFFT factory: Attempting to register factory for plugin cuFFT when one has already been registered\n",
            "2024-05-21 23:09:18.135520: E external/local_xla/xla/stream_executor/cuda/cuda_blas.cc:1515] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered\n",
            "2024-05-21 23:09:18.149625: I external/local_tsl/tsl/cuda/cudart_stub.cc:31] Could not find cuda drivers on your machine, GPU will not be used.\n",
            "2024-05-21 23:09:18.151374: I tensorflow/core/platform/cpu_feature_guard.cc:182] This TensorFlow binary is optimized to use available CPU instructions in performance-critical operations.\n",
            "To enable the following instructions: AVX2 FMA, in other operations, rebuild TensorFlow with the appropriate compiler flags.\n",
            "2024-05-21 23:09:25.667083: W tensorflow/compiler/tf2tensorrt/utils/py_utils.cc:38] TF-TRT Warning: Could not find TensorRT\n",
            "/home/stat/anaconda3/envs/flower/lib/python3.10/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
            "  from .autonotebook import tqdm as notebook_tqdm\n"
          ]
        }
      ],
      "source": [
        "import re\n",
        "import numpy as np\n",
        "\n",
        "# import tensorflow.compat.v1 as tf\n",
        "# tf.disable_eager_execution()\n",
        "\n",
        "# @title Imports.\n",
        "import tensorflow.compat.v2 as tf\n",
        "tf.enable_v2_behavior()\n",
        "\n",
        "\n",
        "import tensorflow_hub as hub\n",
        "import tensorflow_datasets as tfds\n",
        "\n",
        "import matplotlib\n",
        "import matplotlib.pyplot as plt"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 3,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Be57htrgawXJ",
        "outputId": "4eb5eb10-f3cd-47c0-8adc-11e6c685d2fe"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "[]\n",
            "4.9.4\n",
            "2.15.0\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "2024-05-21 23:09:38.542736: W tensorflow/core/common_runtime/gpu/gpu_device.cc:2256] Cannot dlopen some GPU libraries. Please make sure the missing libraries mentioned above are installed properly if you would like to use GPU. Follow the guide at https://www.tensorflow.org/install/gpu for how to download and setup the required libraries for your platform.\n",
            "Skipping registering GPU devices...\n"
          ]
        }
      ],
      "source": [
        "print(tf.config.list_physical_devices('GPU'))\n",
        "import tensorflow_datasets as tfds\n",
        "\n",
        "# Print the version of TensorFlow Datasets\n",
        "print(tfds.__version__)\n",
        "\n",
        "print(tf.__version__)  # Ensure TensorFlow is 2.11"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 4,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "mnyvq6g-P2rW",
        "outputId": "2d6c958b-3a79-4513-ee1b-5c95a54e2e79"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "--2024-05-21 23:09:39--  https://storage.googleapis.com/bit_models/ilsvrc2012_wordnet_lemmas.txt\n",
            "Resolving storage.googleapis.com (storage.googleapis.com)... 173.194.219.207, 64.233.177.207, 108.177.122.207, ...\n",
            "Connecting to storage.googleapis.com (storage.googleapis.com)|173.194.219.207|:443... connected.\n",
            "HTTP request sent, awaiting response... 200 OK\n",
            "Length: 21675 (21K) [text/plain]\n",
            "Saving to: ‘ilsvrc2012_wordnet_lemmas.txt.6’\n",
            "\n",
            "ilsvrc2012_wordnet_ 100%[===================>]  21.17K  --.-KB/s    in 0.004s  \n",
            "\n",
            "2024-05-21 23:09:40 (5.08 MB/s) - ‘ilsvrc2012_wordnet_lemmas.txt.6’ saved [21675/21675]\n",
            "\n"
          ]
        }
      ],
      "source": [
        "#@title Load class id to label text mapping from big_transfer (hidden)\n",
        "# Code snippet credit: https://github.com/google-research/big_transfer\n",
        "\n",
        "!wget https://storage.googleapis.com/bit_models/ilsvrc2012_wordnet_lemmas.txt\n",
        "\n",
        "imagenet_int_to_str = {}\n",
        "\n",
        "with open('ilsvrc2012_wordnet_lemmas.txt', 'r') as f:\n",
        "  for i in range(1000):\n",
        "    row = f.readline()\n",
        "    row = row.rstrip()\n",
        "    imagenet_int_to_str.update({i: row})\n",
        "\n",
        "tf_flowers_labels = ['dandelion', 'daisy', 'tulips', 'sunflowers', 'roses']"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 5,
      "metadata": {
        "id": "BxhfMmVdHoZM"
      },
      "outputs": [],
      "source": [
        "#@title Preprocessing functions from data_util.py in SimCLR repository (hidden).\n",
        "\n",
        "FLAGS_color_jitter_strength = 0.3\n",
        "CROP_PROPORTION = 0.875  # Standard for ImageNet.\n",
        "\n",
        "\n",
        "def random_apply(func, p, x):\n",
        "  \"\"\"Randomly apply function func to x with probability p.\"\"\"\n",
        "  return tf.cond(\n",
        "      tf.less(tf.random.uniform([], minval=0, maxval=1, dtype=tf.float32),\n",
        "              tf.cast(p, tf.float32)),\n",
        "      lambda: func(x),\n",
        "      lambda: x)\n",
        "\n",
        "\n",
        "def random_brightness(image, max_delta, impl='simclrv2'):\n",
        "  \"\"\"A multiplicative vs additive change of brightness.\"\"\"\n",
        "  if impl == 'simclrv2':\n",
        "    factor = tf.random.uniform(\n",
        "        [], tf.maximum(1.0 - max_delta, 0), 1.0 + max_delta)\n",
        "    image = image * factor\n",
        "  elif impl == 'simclrv1':\n",
        "    image = random_brightness(image, max_delta=max_delta)\n",
        "  else:\n",
        "    raise ValueError('Unknown impl {} for random brightness.'.format(impl))\n",
        "  return image\n",
        "\n",
        "\n",
        "def to_grayscale(image, keep_channels=True):\n",
        "  image = tf.image.rgb_to_grayscale(image)\n",
        "  if keep_channels:\n",
        "    image = tf.tile(image, [1, 1, 3])\n",
        "  return image\n",
        "\n",
        "\n",
        "def color_jitter(image,\n",
        "                 strength,\n",
        "                 random_order=True):\n",
        "  \"\"\"Distorts the color of the image.\n",
        "  Args:\n",
        "    image: The input image tensor.\n",
        "    strength: the floating number for the strength of the color augmentation.\n",
        "    random_order: A bool, specifying whether to randomize the jittering order.\n",
        "  Returns:\n",
        "    The distorted image tensor.\n",
        "  \"\"\"\n",
        "  brightness = 0.8 * strength\n",
        "  contrast = 0.8 * strength\n",
        "  saturation = 0.8 * strength\n",
        "  hue = 0.2 * strength\n",
        "  if random_order:\n",
        "    return color_jitter_rand(image, brightness, contrast, saturation, hue)\n",
        "  else:\n",
        "    return color_jitter_nonrand(image, brightness, contrast, saturation, hue)\n",
        "\n",
        "\n",
        "def color_jitter_nonrand(image, brightness=0, contrast=0, saturation=0, hue=0):\n",
        "  \"\"\"Distorts the color of the image (jittering order is fixed).\n",
        "  Args:\n",
        "    image: The input image tensor.\n",
        "    brightness: A float, specifying the brightness for color jitter.\n",
        "    contrast: A float, specifying the contrast for color jitter.\n",
        "    saturation: A float, specifying the saturation for color jitter.\n",
        "    hue: A float, specifying the hue for color jitter.\n",
        "  Returns:\n",
        "    The distorted image tensor.\n",
        "  \"\"\"\n",
        "  with tf.name_scope('distort_color'):\n",
        "    def apply_transform(i, x, brightness, contrast, saturation, hue):\n",
        "      \"\"\"Apply the i-th transformation.\"\"\"\n",
        "      if brightness != 0 and i == 0:\n",
        "        x = random_brightness(x, max_delta=brightness)\n",
        "      elif contrast != 0 and i == 1:\n",
        "        x = tf.image.random_contrast(\n",
        "            x, lower=1-contrast, upper=1+contrast)\n",
        "      elif saturation != 0 and i == 2:\n",
        "        x = tf.image.random_saturation(\n",
        "            x, lower=1-saturation, upper=1+saturation)\n",
        "      elif hue != 0:\n",
        "        x = tf.image.random_hue(x, max_delta=hue)\n",
        "      return x\n",
        "\n",
        "    for i in range(4):\n",
        "      image = apply_transform(i, image, brightness, contrast, saturation, hue)\n",
        "      image = tf.clip_by_value(image, 0., 1.)\n",
        "    return image\n",
        "\n",
        "\n",
        "def color_jitter_rand(image, brightness=0, contrast=0, saturation=0, hue=0):\n",
        "  \"\"\"Distorts the color of the image (jittering order is random).\n",
        "  Args:\n",
        "    image: The input image tensor.\n",
        "    brightness: A float, specifying the brightness for color jitter.\n",
        "    contrast: A float, specifying the contrast for color jitter.\n",
        "    saturation: A float, specifying the saturation for color jitter.\n",
        "    hue: A float, specifying the hue for color jitter.\n",
        "  Returns:\n",
        "    The distorted image tensor.\n",
        "  \"\"\"\n",
        "  with tf.name_scope('distort_color'):\n",
        "    def apply_transform(i, x):\n",
        "      \"\"\"Apply the i-th transformation.\"\"\"\n",
        "      def brightness_foo():\n",
        "        if brightness == 0:\n",
        "          return x\n",
        "        else:\n",
        "          return random_brightness(x, max_delta=brightness)\n",
        "      def contrast_foo():\n",
        "        if contrast == 0:\n",
        "          return x\n",
        "        else:\n",
        "          return tf.image.random_contrast(x, lower=1-contrast, upper=1+contrast)\n",
        "      def saturation_foo():\n",
        "        if saturation == 0:\n",
        "          return x\n",
        "        else:\n",
        "          return tf.image.random_saturation(\n",
        "              x, lower=1-saturation, upper=1+saturation)\n",
        "      def hue_foo():\n",
        "        if hue == 0:\n",
        "          return x\n",
        "        else:\n",
        "          return tf.image.random_hue(x, max_delta=hue)\n",
        "      x = tf.cond(tf.less(i, 2),\n",
        "                  lambda: tf.cond(tf.less(i, 1), brightness_foo, contrast_foo),\n",
        "                  lambda: tf.cond(tf.less(i, 3), saturation_foo, hue_foo))\n",
        "      return x\n",
        "\n",
        "    perm = tf.random_shuffle(tf.range(4))\n",
        "    for i in range(4):\n",
        "      image = apply_transform(perm[i], image)\n",
        "      image = tf.clip_by_value(image, 0., 1.)\n",
        "    return image\n",
        "\n",
        "\n",
        "def _compute_crop_shape(\n",
        "    image_height, image_width, aspect_ratio, crop_proportion):\n",
        "  \"\"\"Compute aspect ratio-preserving shape for central crop.\n",
        "  The resulting shape retains `crop_proportion` along one side and a proportion\n",
        "  less than or equal to `crop_proportion` along the other side.\n",
        "  Args:\n",
        "    image_height: Height of image to be cropped.\n",
        "    image_width: Width of image to be cropped.\n",
        "    aspect_ratio: Desired aspect ratio (width / height) of output.\n",
        "    crop_proportion: Proportion of image to retain along the less-cropped side.\n",
        "  Returns:\n",
        "    crop_height: Height of image after cropping.\n",
        "    crop_width: Width of image after cropping.\n",
        "  \"\"\"\n",
        "  image_width_float = tf.cast(image_width, tf.float32)\n",
        "  image_height_float = tf.cast(image_height, tf.float32)\n",
        "\n",
        "  def _requested_aspect_ratio_wider_than_image():\n",
        "    crop_height = tf.cast(tf.rint(\n",
        "        crop_proportion / aspect_ratio * image_width_float), tf.int32)\n",
        "    crop_width = tf.cast(tf.rint(\n",
        "        crop_proportion * image_width_float), tf.int32)\n",
        "    return crop_height, crop_width\n",
        "\n",
        "  def _image_wider_than_requested_aspect_ratio():\n",
        "    crop_height = tf.cast(\n",
        "        tf.rint(crop_proportion * image_height_float), tf.int32)\n",
        "    crop_width = tf.cast(tf.rint(\n",
        "        crop_proportion * aspect_ratio *\n",
        "        image_height_float), tf.int32)\n",
        "    return crop_height, crop_width\n",
        "\n",
        "  return tf.cond(\n",
        "      aspect_ratio > image_width_float / image_height_float,\n",
        "      _requested_aspect_ratio_wider_than_image,\n",
        "      _image_wider_than_requested_aspect_ratio)\n",
        "\n",
        "\n",
        "def center_crop(image, height, width, crop_proportion):\n",
        "  \"\"\"Crops to center of image and rescales to desired size.\n",
        "  Args:\n",
        "    image: Image Tensor to crop.\n",
        "    height: Height of image to be cropped.\n",
        "    width: Width of image to be cropped.\n",
        "    crop_proportion: Proportion of image to retain along the less-cropped side.\n",
        "  Returns:\n",
        "    A `height` x `width` x channels Tensor holding a central crop of `image`.\n",
        "  \"\"\"\n",
        "  shape = tf.shape(image)\n",
        "  image_height = shape[0]\n",
        "  image_width = shape[1]\n",
        "  crop_height, crop_width = _compute_crop_shape(\n",
        "      image_height, image_width, height / width, crop_proportion)\n",
        "  offset_height = ((image_height - crop_height) + 1) // 2\n",
        "  offset_width = ((image_width - crop_width) + 1) // 2\n",
        "  image = tf.image.crop_to_bounding_box(\n",
        "      image, offset_height, offset_width, crop_height, crop_width)\n",
        "\n",
        "  image = tf.image.resize_bicubic([image], [height, width])[0]\n",
        "\n",
        "  return image\n",
        "\n",
        "\n",
        "def distorted_bounding_box_crop(image,\n",
        "                                bbox,\n",
        "                                min_object_covered=0.1,\n",
        "                                aspect_ratio_range=(0.75, 1.33),\n",
        "                                area_range=(0.05, 1.0),\n",
        "                                max_attempts=100,\n",
        "                                scope=None):\n",
        "  \"\"\"Generates cropped_image using one of the bboxes randomly distorted.\n",
        "  See `tf.image.sample_distorted_bounding_box` for more documentation.\n",
        "  Args:\n",
        "    image: `Tensor` of image data.\n",
        "    bbox: `Tensor` of bounding boxes arranged `[1, num_boxes, coords]`\n",
        "        where each coordinate is [0, 1) and the coordinates are arranged\n",
        "        as `[ymin, xmin, ymax, xmax]`. If num_boxes is 0 then use the whole\n",
        "        image.\n",
        "    min_object_covered: An optional `float`. Defaults to `0.1`. The cropped\n",
        "        area of the image must contain at least this fraction of any bounding\n",
        "        box supplied.\n",
        "    aspect_ratio_range: An optional list of `float`s. The cropped area of the\n",
        "        image must have an aspect ratio = width / height within this range.\n",
        "    area_range: An optional list of `float`s. The cropped area of the image\n",
        "        must contain a fraction of the supplied image within in this range.\n",
        "    max_attempts: An optional `int`. Number of attempts at generating a cropped\n",
        "        region of the image of the specified constraints. After `max_attempts`\n",
        "        failures, return the entire image.\n",
        "    scope: Optional `str` for name scope.\n",
        "  Returns:\n",
        "    (cropped image `Tensor`, distorted bbox `Tensor`).\n",
        "  \"\"\"\n",
        "  # with tf.name_scope(scope, 'distorted_bounding_box_crop', [image, bbox]):\n",
        "  if scope is None:\n",
        "    scope = 'distorted_bounding_box_crop'\n",
        "  else:\n",
        "    scope = f\"{scope}_distorted_bounding_box_crop\"\n",
        "\n",
        "  with tf.name_scope(scope):\n",
        "    shape = tf.shape(image)\n",
        "    sample_distorted_bounding_box = tf.image.sample_distorted_bounding_box(\n",
        "        shape,\n",
        "        bounding_boxes=bbox,\n",
        "        min_object_covered=min_object_covered,\n",
        "        aspect_ratio_range=aspect_ratio_range,\n",
        "        area_range=area_range,\n",
        "        max_attempts=max_attempts,\n",
        "        use_image_if_no_bounding_boxes=True)\n",
        "    bbox_begin, bbox_size, _ = sample_distorted_bounding_box\n",
        "\n",
        "    # Crop the image to the specified bounding box.\n",
        "    offset_y, offset_x, _ = tf.unstack(bbox_begin)\n",
        "    target_height, target_width, _ = tf.unstack(bbox_size)\n",
        "    image = tf.image.crop_to_bounding_box(\n",
        "        image, offset_y, offset_x, target_height, target_width)\n",
        "\n",
        "    return image\n",
        "\n",
        "\n",
        "def crop_and_resize(image, height, width):\n",
        "  \"\"\"Make a random crop and resize it to height `height` and width `width`.\n",
        "  Args:\n",
        "    image: Tensor representing the image.\n",
        "    height: Desired image height.\n",
        "    width: Desired image width.\n",
        "  Returns:\n",
        "    A `height` x `width` x channels Tensor holding a random crop of `image`.\n",
        "  \"\"\"\n",
        "  bbox = tf.constant([0.0, 0.0, 1.0, 1.0], dtype=tf.float32, shape=[1, 1, 4])\n",
        "  aspect_ratio = width / height\n",
        "  image = distorted_bounding_box_crop(\n",
        "      image,\n",
        "      bbox,\n",
        "      min_object_covered=0.1,\n",
        "      aspect_ratio_range=(3. / 4 * aspect_ratio, 4. / 3. * aspect_ratio),\n",
        "      area_range=(0.08, 1.0),\n",
        "      max_attempts=100,\n",
        "      scope=None)\n",
        "  resized_image = tf.image.resize(image, [height, width], method='bicubic')\n",
        "  return resized_image # for tf v1: tf.image.resize_bicubic([image], [height, width])[0]\n",
        "\n",
        "\n",
        "def gaussian_blur(image, kernel_size, sigma, padding='SAME'):\n",
        "  \"\"\"Blurs the given image with separable convolution.\n",
        "  Args:\n",
        "    image: Tensor of shape [height, width, channels] and dtype float to blur.\n",
        "    kernel_size: Integer Tensor for the size of the blur kernel. This is should\n",
        "      be an odd number. If it is an even number, the actual kernel size will be\n",
        "      size + 1.\n",
        "    sigma: Sigma value for gaussian operator.\n",
        "    padding: Padding to use for the convolution. Typically 'SAME' or 'VALID'.\n",
        "  Returns:\n",
        "    A Tensor representing the blurred image.\n",
        "  \"\"\"\n",
        "  radius = tf.to_int32(kernel_size / 2)\n",
        "  kernel_size = radius * 2 + 1\n",
        "  x = tf.to_float(tf.range(-radius, radius + 1))\n",
        "  blur_filter = tf.exp(\n",
        "      -tf.pow(x, 2.0) / (2.0 * tf.pow(tf.to_float(sigma), 2.0)))\n",
        "  blur_filter /= tf.reduce_sum(blur_filter)\n",
        "  # One vertical and one horizontal filter.\n",
        "  blur_v = tf.reshape(blur_filter, [kernel_size, 1, 1, 1])\n",
        "  blur_h = tf.reshape(blur_filter, [1, kernel_size, 1, 1])\n",
        "  num_channels = tf.shape(image)[-1]\n",
        "  blur_h = tf.tile(blur_h, [1, 1, num_channels, 1])\n",
        "  blur_v = tf.tile(blur_v, [1, 1, num_channels, 1])\n",
        "  expand_batch_dim = image.shape.ndims == 3\n",
        "  if expand_batch_dim:\n",
        "    # Tensorflow requires batched input to convolutions, which we can fake with\n",
        "    # an extra dimension.\n",
        "    image = tf.expand_dims(image, axis=0)\n",
        "  blurred = tf.nn.depthwise_conv2d(\n",
        "      image, blur_h, strides=[1, 1, 1, 1], padding=padding)\n",
        "  blurred = tf.nn.depthwise_conv2d(\n",
        "      blurred, blur_v, strides=[1, 1, 1, 1], padding=padding)\n",
        "  if expand_batch_dim:\n",
        "    blurred = tf.squeeze(blurred, axis=0)\n",
        "  return blurred\n",
        "\n",
        "\n",
        "def random_crop_with_resize(image, height, width, p=1.0):\n",
        "  \"\"\"Randomly crop and resize an image.\n",
        "  Args:\n",
        "    image: `Tensor` representing an image of arbitrary size.\n",
        "    height: Height of output image.\n",
        "    width: Width of output image.\n",
        "    p: Probability of applying this transformation.\n",
        "  Returns:\n",
        "    A preprocessed image `Tensor`.\n",
        "  \"\"\"\n",
        "  def _transform(image):  # pylint: disable=missing-docstring\n",
        "    image = crop_and_resize(image, height, width)\n",
        "    return image\n",
        "  return random_apply(_transform, p=p, x=image)\n",
        "\n",
        "\n",
        "def random_color_jitter(image, p=1.0):\n",
        "  def _transform(image):\n",
        "    color_jitter_t = functools.partial(\n",
        "        color_jitter, strength=FLAGS_color_jitter_strength)\n",
        "    image = random_apply(color_jitter_t, p=0.8, x=image)\n",
        "    return random_apply(to_grayscale, p=0.2, x=image)\n",
        "  return random_apply(_transform, p=p, x=image)\n",
        "\n",
        "\n",
        "def random_blur(image, height, width, p=1.0):\n",
        "  \"\"\"Randomly blur an image.\n",
        "  Args:\n",
        "    image: `Tensor` representing an image of arbitrary size.\n",
        "    height: Height of output image.\n",
        "    width: Width of output image.\n",
        "    p: probability of applying this transformation.\n",
        "  Returns:\n",
        "    A preprocessed image `Tensor`.\n",
        "  \"\"\"\n",
        "  del width\n",
        "  def _transform(image):\n",
        "    sigma = tf.random.uniform([], 0.1, 2.0, dtype=tf.float32)\n",
        "    return gaussian_blur(\n",
        "        image, kernel_size=height//10, sigma=sigma, padding='SAME')\n",
        "  return random_apply(_transform, p=p, x=image)\n",
        "\n",
        "\n",
        "def batch_random_blur(images_list, height, width, blur_probability=0.5):\n",
        "  \"\"\"Apply efficient batch data transformations.\n",
        "  Args:\n",
        "    images_list: a list of image tensors.\n",
        "    height: the height of image.\n",
        "    width: the width of image.\n",
        "    blur_probability: the probaility to apply the blur operator.\n",
        "  Returns:\n",
        "    Preprocessed feature list.\n",
        "  \"\"\"\n",
        "  def generate_selector(p, bsz):\n",
        "    shape = [bsz, 1, 1, 1]\n",
        "    selector = tf.cast(\n",
        "        tf.less(tf.random.uniform(shape, 0, 1, dtype=tf.float32), p),\n",
        "        tf.float32)\n",
        "    return selector\n",
        "\n",
        "  new_images_list = []\n",
        "  for images in images_list:\n",
        "    images_new = random_blur(images, height, width, p=1.)\n",
        "    selector = generate_selector(blur_probability, tf.shape(images)[0])\n",
        "    images = images_new * selector + images * (1 - selector)\n",
        "    images = tf.clip_by_value(images, 0., 1.)\n",
        "    new_images_list.append(images)\n",
        "\n",
        "  return new_images_list\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 7,
      "metadata": {
        "id": "_KphYz_UsIGv"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "from scipy.linalg import eigh\n",
        "\n",
        "\n",
        "\n",
        "cls_loss_object = tf.keras.losses.CategoricalCrossentropy(\n",
        "    from_logits=True,\n",
        "    reduction=tf.keras.losses.Reduction.NONE,\n",
        "    label_smoothing=0.0 # Ensure this is a float.\n",
        ")\n",
        "\n",
        "def get_cls_loss(labels, outputs):\n",
        "  return tf.reduce_mean(cls_loss_object(labels, outputs))\n",
        "\n",
        "\n",
        "def repmat(A, N, M):\n",
        "    # A is the input matrix/tensor\n",
        "    # N is the number of times A is repeated along the rows\n",
        "    # M is the number of times A is repeated along the columns\n",
        "    expanded_A = tf.expand_dims(A, 0)  # Add a new dimension for tiling\n",
        "    tiled_A = tf.tile(expanded_A, [N, M, 1])  # Tile in the new dimension\n",
        "    final_shape = [N * tf.shape(A)[0], M * tf.shape(A)[1]]\n",
        "    return tf.reshape(tiled_A, final_shape)\n",
        "\n",
        "\n",
        "def get_nfda_loss(z1, z2, penal_para):   # [batch_size, dim]\n",
        "    # print(\"the shape of z1\")\n",
        "    # print(z1.shape)\n",
        "    batch_size = tf.shape(z1)[0]\n",
        "    n=2*batch_size\n",
        "    # print(\"batch size\")\n",
        "    # print(batch_size)\n",
        "    dim = tf.shape(z1)[1]\n",
        "\n",
        "    z1 = tf.math.l2_normalize(z1, -1)\n",
        "    z2 = tf.math.l2_normalize(z2, -1)\n",
        "\n",
        "\n",
        "    z=tf.concat([z1, z2], axis=0) # dimension: (2*batch_size,dim)\n",
        "\n",
        "    Sb = tf.zeros((dim, dim), dtype=tf.float32)  # Between-class scatter matrix\n",
        "    Sw = tf.zeros((dim, dim), dtype=tf.float32)  # Within-class scatter matrix\n",
        "\n",
        "    zmean=tf.zeros((batch_size, z.shape[1]), dtype=z.dtype)\n",
        "\n",
        "    for i in tf.range(batch_size):\n",
        "        zc = tf.gather(z, indices=[i, i + batch_size], axis=0) #Kz[:, (i,i+batch_size)]\n",
        "\n",
        "        # 2. Sum the squared elements row-wise\n",
        "        col_sums_squared = tf.reduce_sum(tf.square(zc), axis=1)\n",
        "\n",
        "        # 4. Transpose to get a row matrix\n",
        "        zc2 = tf.reshape(col_sums_squared, (1, -1))  # Explicitly reshape to a row matrix if needed.\n",
        "\n",
        "        mean_zc = tf.reduce_mean(zc, axis=0)\n",
        "\n",
        "        updates = tf.reshape(mean_zc, [1, -1])\n",
        "\n",
        "\n",
        "        #indices = tf.constant([[int(i)]])\n",
        "        indices = tf.reshape(i, [1, 1])  #tf.constant([[i]], dtype=tf.int32)\n",
        "\n",
        "        zmean = tf.tensor_scatter_nd_update(zmean, indices, updates)\n",
        "\n",
        "\n",
        "        rep_zc2 = repmat(zc2, 2, 1)\n",
        "\n",
        "        rep_zc2_T = repmat(tf.transpose(zc2), 1, 2)\n",
        "\n",
        "        #distance2 = rep_zc2 + rep_zc2_T - 2 * tf.matmul(zc,tf.transpose(zc))\n",
        "\n",
        "\n",
        "        A = tf.ones((2, 2)) #tf.exp(-distance2) #get_affinity_matrix(distance2, 1, 2)\n",
        "\n",
        "\n",
        "        zc1 = tf.reduce_sum(zc, axis=0, keepdims=True)\n",
        "\n",
        "\n",
        "        colSums_A = tf.reshape(tf.reduce_sum(A, axis=0), [-1, 1])  # Sum columns and ensure it's a row vector\n",
        "        replicated_colSums_A = tf.tile(colSums_A, [1, tf.shape(zc)[1]])  # Replicate across rows to match Kc's row count\n",
        "\n",
        "        # Perform the matrix operations as described\n",
        "        # Note: In TensorFlow, matrix multiplication is done using tf.matmul or the '@' operator\n",
        "        Z = tf.matmul(tf.transpose(zc), replicated_colSums_A * zc) - tf.matmul(tf.matmul(tf.transpose(zc), A), zc)\n",
        "\n",
        "\n",
        "        Sb += (Z / tf.cast(n, tf.float32)) + tf.transpose(zc) @ zc * (1 - 2.0 / tf.cast(n, tf.float32)) + tf.transpose(zc1) @ zc1 / tf.cast(n, tf.float32)\n",
        "        Sw += Z / 2.0\n",
        "\n",
        "    z1 = tf.reduce_sum(z, axis=0, keepdims=True)\n",
        "\n",
        "    Sb = Sb -  tf.transpose(z1) @ z1 / tf.cast(n, tf.float32) - Sw\n",
        "\n",
        "    Sb = (Sb + tf.transpose(Sb)) / 2.0  # Final between-class scatter matrix\n",
        "    Sw = (Sw + tf.transpose(Sw)) / 2.0  # Final within-class scatter matrix\n",
        "\n",
        "\n",
        "    eye_mat = penal_para * tf.eye(tf.shape(Sw)[0], dtype=Sb.dtype)\n",
        "\n",
        "    B = Sw + eye_mat  # Make sure this is positive definite\n",
        "\n",
        "\n",
        "    temp = tf.linalg.pinv(B) @ Sb\n",
        "\n",
        "\n",
        "    temp=(temp + tf.transpose(temp)) / 2.0\n",
        "\n",
        "    evals, evecs = tf.linalg.eig(temp)\n",
        "\n",
        "    evals_real = tf.math.real(evals)\n",
        "    evecs_real = tf.math.real(evecs)\n",
        "\n",
        "\n",
        "    sorted_indices = tf.argsort(evals_real)\n",
        "    evals_sorted = tf.gather(evals_real, sorted_indices)\n",
        "    evecs_sorted = tf.gather(evecs_real, axis=1, indices=sorted_indices)\n",
        "\n",
        "    n_components = 128 - 1\n",
        "    evals = evals_sorted[-n_components:]\n",
        "\n",
        "    loss = -tf.reduce_mean(evals)\n",
        "\n",
        "    coef = tf.matmul(tf.matmul(zmean, evecs_sorted), tf.transpose(evecs_sorted))  # CxD\n",
        "    intercept = -0.5 * tf.linalg.diag_part(tf.matmul(zmean, tf.transpose(coef)))  # C\n",
        "\n",
        "\n",
        "    logit = tf.matmul(z, coef, transpose_b=True) + intercept\n",
        "\n",
        "    ypred=tf.argmax(logit, axis=1)\n",
        "\n",
        "    labels1 = tf.range(batch_size)\n",
        "    labels2 = tf.range(batch_size)\n",
        "\n",
        "    labels= tf.concat([labels1, labels2], axis=0)\n",
        "\n",
        "    correct_predictions = tf.equal(ypred, tf.cast(labels, tf.int64))\n",
        "    accuracy = tf.reduce_mean(tf.cast(correct_predictions, tf.float32))\n",
        "\n",
        "\n",
        "    return tf.reduce_mean(loss), accuracy #tf.reduce_mean(loss), sim\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 8,
      "metadata": {
        "id": "n7JEyeKp2J1o"
      },
      "outputs": [],
      "source": [
        "#@title Model.\n",
        "def dense_bn_relu(units):\n",
        "  return tf.keras.Sequential([\n",
        "      tf.keras.layers.Dense(\n",
        "          units, use_bias=False,\n",
        "          kernel_regularizer=tf.keras.regularizers.l2(1e-4)),\n",
        "      tf.keras.layers.BatchNormalization(center=True, scale=True),\n",
        "      tf.keras.layers.ReLU()\n",
        "  ])\n",
        "\n",
        "def conv2d_bn_relu(filters, kernel_size, strides):\n",
        "  return tf.keras.Sequential([\n",
        "      tf.keras.layers.Conv2D(\n",
        "          filters, kernel_size, strides, use_bias=False,\n",
        "          kernel_regularizer=tf.keras.regularizers.l2(1e-4)),\n",
        "      tf.keras.layers.BatchNormalization(center=True, scale=True),\n",
        "      tf.keras.layers.ReLU()\n",
        "  ])\n",
        "\n",
        "class ConvN(tf.keras.Model):\n",
        "\n",
        "  def __init__(self, width_multiplier):\n",
        "    super(ConvN, self).__init__()\n",
        "    self.num_classes = 5\n",
        "    self.latent_dim = 256 * width_multiplier\n",
        "    self.proj_dim = self.latent_dim / 2\n",
        "\n",
        "    self.enc = tf.keras.Sequential([\n",
        "        conv2d_bn_relu(32 * width_multiplier, 3, 2),\n",
        "        conv2d_bn_relu(64 * width_multiplier, 3, 2),\n",
        "        conv2d_bn_relu(64 * width_multiplier, 3, 2),\n",
        "        tf.keras.layers.Flatten(),\n",
        "        dense_bn_relu(self.latent_dim)\n",
        "    ])\n",
        "\n",
        "    self.proj = tf.keras.Sequential([\n",
        "        dense_bn_relu(self.latent_dim * 2),\n",
        "        tf.keras.layers.Dense(\n",
        "            self.proj_dim, use_bias=False, activation=None,\n",
        "            kernel_regularizer=tf.keras.regularizers.l2(1e-4)),\n",
        "    ])\n",
        "\n",
        "    self.classifier = tf.keras.layers.Dense(self.num_classes)\n",
        "\n",
        "  def call(self, inputs, training):\n",
        "    y = self.enc(inputs, training)\n",
        "    z = self.proj(y, training)\n",
        "    pred = self.classifier(tf.stop_gradient(y))\n",
        "    return y, z, pred"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "mA0YTB11pB4L"
      },
      "source": [
        "# Preprocessing for train and test"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 9,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "dnBLNC1EpAsn",
        "outputId": "530c5dfe-4365-48c6-b92e-b1c0d6f93bf9"
      },
      "outputs": [],
      "source": [
        "\n",
        "num_mnist_digits_used = 256 \n",
        "\n",
        "mnist = tf.keras.datasets.mnist\n",
        "(mnist_images, mnist_labels), _ = mnist.load_data()\n",
        "mnist_images = mnist_images[:num_mnist_digits_used]\n",
        "\n",
        "def map_train_fn(image, label):\n",
        "  # Sample one mnist image.\n",
        "  # print(\"train label shape\")\n",
        "  # print(label.shape)\n",
        "\n",
        "  i = tf.random.uniform([], maxval=num_mnist_digits_used, dtype=tf.int32)\n",
        "  digit = tf.squeeze(tf.slice(mnist_images, [i, 0, 0], [1, 28, 28]))\n",
        "  digit_label = tf.squeeze(tf.slice(mnist_labels, [i], [1]))\n",
        "  digit = tf.image.grayscale_to_rgb(tf.expand_dims(digit, -1))\n",
        "  digit = tf.image.convert_image_dtype(digit, dtype=tf.float32)\n",
        "  image = tf.image.resize(image, [224, 224]) / 255.\n",
        "  \n",
        "\n",
        "  resized_digit_size = 72  # New size for the digit\n",
        "  digit = tf.image.resize(digit, [resized_digit_size, resized_digit_size])\n",
        "\n",
        "  size_big = 224\n",
        "  size_small = resized_digit_size\n",
        "  images = []\n",
        "\n",
        "  images.append(image)\n",
        "  \n",
        "  image = tf.reduce_max(tf.stack(images, 0), 0)\n",
        "\n",
        "  image = tf.image.convert_image_dtype(image, dtype=tf.float32)\n",
        "\n",
        "  original_image = image\n",
        "\n",
        "  image_a = random_crop_with_resize(image, 224, 224)\n",
        "\n",
        "  image_a = tf.image.random_flip_left_right(image)\n",
        "\n",
        "  image_a = tf.reshape(image, [224, 224, 3])\n",
        "  image_a = tf.clip_by_value(image, 0., 1.)\n",
        "\n",
        "  image_b = random_crop_with_resize(image, 224, 224)\n",
        "\n",
        "  image_b = tf.image.random_flip_left_right(image)\n",
        "\n",
        "  image_b = tf.reshape(image, [224, 224, 3])\n",
        "  image_b = tf.clip_by_value(image, 0., 1.)\n",
        "\n",
        "  image = tf.stack([image_a, image_b], axis=0)  # [2, h, w, c]\n",
        "\n",
        "  # print(\"train mapped image shape\")\n",
        "  # print(image.shape)\n",
        "\n",
        "  label = tf.cast(label, tf.int32)\n",
        "  digit_label = tf.cast(digit_label, tf.int32)\n",
        "\n",
        "\n",
        "  return image, label, digit_label, original_image\n",
        "\n",
        "\n",
        "def map_test_fn(image, label):\n",
        "  # Sample one mnist image.\n",
        "  i = tf.random.uniform([], maxval=num_mnist_digits_used, dtype=tf.int32)\n",
        "  digit = tf.squeeze(tf.slice(mnist_images, [i, 0, 0], [1, 28, 28]))\n",
        "  digit_label = tf.squeeze(tf.slice(mnist_labels, [i], [1]))\n",
        "  digit = tf.image.grayscale_to_rgb(tf.expand_dims(digit, -1))\n",
        "  digit = tf.image.convert_image_dtype(digit, dtype=tf.float32)\n",
        "  image = tf.image.resize(image, [224, 224]) / 255.\n",
        "\n",
        "\n",
        "  resized_digit_size = 72  # New size for the digit\n",
        "  digit = tf.image.resize(digit, [resized_digit_size, resized_digit_size])\n",
        "\n",
        "  size_big = 224\n",
        "  size_small = resized_digit_size\n",
        "  images = []\n",
        "\n",
        "  images.append(image)\n",
        "  \n",
        "\n",
        "  image = tf.reduce_max(tf.stack(images, 0), 0)\n",
        "\n",
        "\n",
        "  label = tf.cast(label, tf.int32)\n",
        "  digit_label = tf.cast(digit_label, tf.int32)\n",
        "\n",
        "\n",
        "  return image, label, digit_label\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 16,
      "metadata": {
        "id": "3dX8179j2PBo"
      },
      "outputs": [],
      "source": [
        "import tensorflow_datasets as tfds\n",
        "import matplotlib.pyplot as plt\n",
        "import seaborn as sns\n",
        "import math\n",
        "import pandas as pd\n",
        "\n",
        "#@title Define train_and_eval() for contrastive learning.\n",
        "def train_and_eval(\n",
        "    batch_size=128,\n",
        "    width_multiplier=1,\n",
        "    extra_channel_bits=10,\n",
        "    nt_xent_temp=0.1,\n",
        "    learning_rate=0.001,\n",
        "    epochs=10,\n",
        "    log_summary_every_n_steps=1000,\n",
        "    eval_every_n_steps=1,\n",
        "    print_output=False):\n",
        "  strategy = tf.distribute.MirroredStrategy()\n",
        "  strategy = tf.distribute.get_strategy()\n",
        "\n",
        "  # # Load dataset.\n",
        "  # builder = tfds.builder('mnist')\n",
        "  # builder.download_and_prepare()\n",
        "\n",
        "  # print(\"batch size\")\n",
        "  # print(batch_size)\n",
        "\n",
        "  # Use `tf_flowers` in this example instead of `imagenet2012` because the latter\n",
        "  # needs to be downloaded manually.\n",
        "  # (See https://www.tensorflow.org/datasets/catalog/imagenet2012)\n",
        "\n",
        "  builder = tfds.builder('tf_flowers')\n",
        "  builder.download_and_prepare()\n",
        "\n",
        "\n",
        "  train_dataset = builder.as_dataset(split='train', as_supervised=True)\n",
        "  total_samples = builder.info.splits['train'].num_examples\n",
        "  shuffled_dataset = train_dataset.shuffle(buffer_size=total_samples, reshuffle_each_iteration=False)\n",
        "\n",
        "\n",
        "  # Define the split ratio for training and testing\n",
        "  train_split_ratio = 0.8\n",
        "  num_train_samples = int(train_split_ratio * total_samples)\n",
        "\n",
        "  # Create training and testing datasets\n",
        "  train_dataset = shuffled_dataset.take(num_train_samples)\n",
        "  test_dataset = shuffled_dataset.skip(num_train_samples)\n",
        "\n",
        "\n",
        "  train_dataset = train_dataset.repeat().map(map_train_fn)\n",
        "  # train_dataset = train_dataset.repeat().map(_preprocess_train)\n",
        "  train_dataset = train_dataset.batch(batch_size)\n",
        "\n",
        "  train_iter = iter(train_dataset)\n",
        "\n",
        "\n",
        "  test_dataset = test_dataset.map(map_test_fn)\n",
        "  test_dataset = test_dataset.batch(batch_size, drop_remainder=False)\n",
        "  test_iter = iter(test_dataset)\n",
        "\n",
        "  total_steps = int(2936 * epochs / batch_size)\n",
        "  steps_per_epoch_test = math.ceil(734 / batch_size) # total: 3670\n",
        "\n",
        "\n",
        "  # Model and optimizer.\n",
        "  model = ConvN(width_multiplier)\n",
        "  global_step = tf.Variable(\n",
        "      1, trainable=False, name=\"global_step\", dtype=tf.int64)\n",
        "\n",
        "  lr_schedule = tf.keras.optimizers.schedules.PolynomialDecay(\n",
        "      initial_learning_rate=learning_rate,\n",
        "      decay_steps=total_steps,\n",
        "      end_learning_rate=0.)\n",
        "  optimizer = tf.keras.optimizers.Adam(lr_schedule)\n",
        "\n",
        "  # Define metrics.\n",
        "  loss_metrics = [\n",
        "      \"train_classification_loss\",\n",
        "      \"train_contrastive_loss\",\n",
        "      \"train_total_loss\",\n",
        "  ]\n",
        "  acc_metrics = [\"train_accuracy\", \"eval_accuracy\",\n",
        "                 \"train_contrastive_accuracy\"]\n",
        "  metric_list = {s: tf.keras.metrics.Mean(name=s) for s in loss_metrics}\n",
        "  metric_list.update({\n",
        "      s: tf.keras.metrics.SparseCategoricalAccuracy(name=s)\n",
        "      for s in acc_metrics\n",
        "  })\n",
        "\n",
        "  #metric_list[\"rank\"] = tf.keras.metrics.Mean(name=\"rank\")\n",
        "  metric_list[\"rank\"] = tf.keras.metrics.Mean(name=\"rank\")\n",
        "\n",
        "  # Step functions.\n",
        "  @tf.function\n",
        "  def train_step(iterator):\n",
        "    def step_fn(inputs):\n",
        "\n",
        "      images, labels, mnist_labels, original_image = inputs\n",
        "      labels_one_hot = tf.one_hot(labels, depth=5)\n",
        "      images_a, images_b = tf.unstack(images, num=2, axis=1)\n",
        "\n",
        "      with tf.GradientTape() as tape:\n",
        "        _, za, pred_a = model(images_a, training=True)\n",
        "        _, zb, pred_b = model(images_b, training=True)\n",
        "        _, z, pred_b = model(original_image, training=True)\n",
        "\n",
        "        contrastive_loss, _ = (\n",
        "            get_nfda_loss(za, zb, nt_xent_temp))\n",
        "\n",
        "        classifier_loss = get_cls_loss(labels_one_hot, pred_a)\n",
        "\n",
        "        wd_loss = sum(model.losses)\n",
        "\n",
        "        loss = classifier_loss + wd_loss + contrastive_loss\n",
        "\n",
        "        z_transposed = tf.transpose(z)\n",
        "        c = tf.linalg.matmul(z_transposed, z_transposed, transpose_b=True)\n",
        "\n",
        "        rank = tf.linalg.matrix_rank(c)\n",
        "\n",
        "        batch_size = tf.shape(images)[0]\n",
        "        metric_list[\"train_contrastive_loss\"].update_state(contrastive_loss)\n",
        "        metric_list[\"train_classification_loss\"].update_state(classifier_loss)\n",
        "        metric_list[\"train_accuracy\"].update_state(labels, pred_a)\n",
        "        metric_list[\"train_total_loss\"].update_state(loss)\n",
        "        metric_list[\"rank\"].update_state(rank)\n",
        "\n",
        "\n",
        "      gradients = tape.gradient(loss, model.trainable_variables)\n",
        "      optimizer.apply_gradients(zip(gradients, model.trainable_variables))\n",
        "\n",
        "\n",
        "\n",
        "    strategy.run(step_fn, args=(next(iterator),))\n",
        "    global_step.assign_add(1)\n",
        "\n",
        "\n",
        "\n",
        "  @tf.function\n",
        "  def eval_step(iterator):\n",
        "    def step_fn(inputs):\n",
        "      images, labels, mnist_labels = inputs\n",
        "      _, _, predictions = model(images, training=False)\n",
        "      metric_list[\"eval_accuracy\"].update_state(labels, predictions)\n",
        "    strategy.run(step_fn, args=(next(iterator),))\n",
        "\n",
        "  # Train and eval loop.\n",
        "  steps = []\n",
        "  eval_accuracies = []\n",
        "  ranks=[]\n",
        "  loss_total=[]\n",
        "  nfda_loss=[]\n",
        "  while global_step.numpy() <= total_steps:\n",
        "    train_step(train_iter)\n",
        "    step = global_step.numpy()\n",
        "\n",
        "    if step % log_summary_every_n_steps == 0:\n",
        "      log_msg = \"Steps: {}\".format(step)\n",
        "      for m in loss_metrics + acc_metrics:\n",
        "        if m.startswith(\"train\"):\n",
        "          log_msg += \", {}: {}\".format(m, metric_list[m].result())\n",
        "          metric_list[m].reset_states()\n",
        "      if print_output:\n",
        "        print(log_msg)\n",
        "\n",
        "    if (step % eval_every_n_steps == 0) or (step == total_steps):\n",
        "    #if (step == total_steps):\n",
        "      eval_iter = iter(test_dataset)\n",
        "\n",
        "      for m in loss_metrics + acc_metrics:\n",
        "        if m.startswith(\"eval\"):\n",
        "          metric_list[m].reset_states()\n",
        "\n",
        "      for _ in range(steps_per_epoch_test):\n",
        "        eval_step(eval_iter)\n",
        "      if print_output:\n",
        "        print(\"Steps: {}, Test accuracy: {}\".format(\n",
        "            step, metric_list[\"eval_accuracy\"].result()))\n",
        "      steps.append(step)\n",
        "      eval_accuracies.append(metric_list[\"eval_accuracy\"].result())\n",
        "    ranks.append(metric_list[\"rank\"].result())\n",
        "    loss_total.append(metric_list[\"train_total_loss\"].result())\n",
        "    #nfda_loss.append(metric_list[\"train_contrastive_loss\"].result())\n",
        "  return steps, eval_accuracies, ranks, loss_total"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 17,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "M5bUbFNx8afv",
        "outputId": "06da6845-b406-4b29-8f43-21124f674d98"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 5, Test accuracy: 0.25885558128356934\n",
            "Steps: 10, Test accuracy: 0.20163488388061523\n",
            "Steps: 15, Test accuracy: 0.25613078474998474\n",
            "Steps: 20, Test accuracy: 0.29427793622016907\n",
            "Steps: 25, Test accuracy: 0.2711171805858612\n",
            "Steps: 30, Test accuracy: 0.30926430225372314\n",
            "Steps: 35, Test accuracy: 0.3215258717536926\n",
            "Steps: 40, Test accuracy: 0.31880107522010803\n",
            "Steps: 45, Test accuracy: 0.31198909878730774\n",
            "Steps: 50, Test accuracy: 0.31062671542167664\n",
            "Steps: 55, Test accuracy: 0.3923705816268921\n",
            "Steps: 60, Test accuracy: 0.3760218024253845\n",
            "Steps: 65, Test accuracy: 0.3405994474887848\n",
            "Steps: 70, Test accuracy: 0.3256130814552307\n",
            "Steps: 75, Test accuracy: 0.3732970058917999\n",
            "Steps: 80, Test accuracy: 0.4550408720970154\n",
            "Steps: 85, Test accuracy: 0.42779290676116943\n",
            "Steps: 90, Test accuracy: 0.37057220935821533\n",
            "Steps: 95, Test accuracy: 0.37193459272384644\n",
            "Steps: 100, Test accuracy: 0.4468664824962616\n",
            "Steps: 105, Test accuracy: 0.4591280519962311\n",
            "Steps: 110, Test accuracy: 0.4618528485298157\n",
            "Steps: 115, Test accuracy: 0.43051770329475403\n",
            "Steps: 120, Test accuracy: 0.4482288956642151\n",
            "Steps: 125, Test accuracy: 0.47138965129852295\n",
            "Steps: 130, Test accuracy: 0.47002723813056946\n",
            "Steps: 135, Test accuracy: 0.47956404089927673\n",
            "Steps: 140, Test accuracy: 0.4959128201007843\n",
            "Steps: 145, Test accuracy: 0.4986376166343689\n",
            "Steps: 150, Test accuracy: 0.5204359889030457\n",
            "Steps: 155, Test accuracy: 0.4945504069328308\n",
            "Steps: 160, Test accuracy: 0.46321526169776917\n",
            "Steps: 165, Test accuracy: 0.47275203466415405\n",
            "Steps: 170, Test accuracy: 0.4891008138656616\n",
            "Steps: 175, Test accuracy: 0.5190735459327698\n",
            "Steps: 180, Test accuracy: 0.5136239528656006\n",
            "Steps: 185, Test accuracy: 0.5122615694999695\n",
            "Steps: 190, Test accuracy: 0.5027247667312622\n",
            "Steps: 195, Test accuracy: 0.5108991861343384\n",
            "Steps: 200, Test accuracy: 0.5177111625671387\n",
            "Steps: 205, Test accuracy: 0.5190735459327698\n",
            "Steps: 210, Test accuracy: 0.5136239528656006\n",
            "Steps: 215, Test accuracy: 0.5190735459327698\n",
            "Steps: 220, Test accuracy: 0.5163487792015076\n",
            "Steps: 225, Test accuracy: 0.5\n",
            "Steps: 230, Test accuracy: 0.5108991861343384\n",
            "Steps: 235, Test accuracy: 0.5163487792015076\n",
            "Steps: 240, Test accuracy: 0.5054495930671692\n",
            "Steps: 245, Test accuracy: 0.5054495930671692\n",
            "Steps: 250, Test accuracy: 0.5108991861343384\n",
            "Steps: 255, Test accuracy: 0.5204359889030457\n",
            "Steps: 260, Test accuracy: 0.5326975584030151\n",
            "Steps: 265, Test accuracy: 0.5217983722686768\n",
            "Steps: 270, Test accuracy: 0.5122615694999695\n",
            "Steps: 275, Test accuracy: 0.5081743597984314\n",
            "Steps: 280, Test accuracy: 0.5149863958358765\n",
            "Steps: 285, Test accuracy: 0.5149863958358765\n",
            "Steps: 290, Test accuracy: 0.5204359889030457\n",
            "Steps: 295, Test accuracy: 0.5204359889030457\n",
            "Steps: 300, Test accuracy: 0.5177111625671387\n",
            "Steps: 305, Test accuracy: 0.5190735459327698\n",
            "Steps: 310, Test accuracy: 0.5190735459327698\n",
            "Steps: 315, Test accuracy: 0.5217983722686768\n",
            "Steps: 320, Test accuracy: 0.5326975584030151\n",
            "Steps: 325, Test accuracy: 0.531335175037384\n",
            "Steps: 330, Test accuracy: 0.5395095348358154\n",
            "Steps: 335, Test accuracy: 0.5422343611717224\n",
            "Steps: 340, Test accuracy: 0.5422343611717224\n",
            "Steps: 344, Test accuracy: 0.5395095348358154\n",
            "Steps: 345, Test accuracy: 0.5422343611717224\n",
            "temp=0.01, margin=0.1, eval_accuracy=0.5422343611717224\n",
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 5, Test accuracy: 0.17983651161193848\n",
            "Steps: 10, Test accuracy: 0.2207084447145462\n",
            "Steps: 15, Test accuracy: 0.29972752928733826\n",
            "Steps: 20, Test accuracy: 0.30653950572013855\n",
            "Steps: 25, Test accuracy: 0.29427793622016907\n",
            "Steps: 30, Test accuracy: 0.24931880831718445\n",
            "Steps: 35, Test accuracy: 0.24795641005039215\n",
            "Steps: 40, Test accuracy: 0.2602179944515228\n",
            "Steps: 45, Test accuracy: 0.2670299708843231\n",
            "Steps: 50, Test accuracy: 0.24931880831718445\n",
            "Steps: 55, Test accuracy: 0.24659401178359985\n",
            "Steps: 60, Test accuracy: 0.2683923840522766\n",
            "Steps: 65, Test accuracy: 0.23297002911567688\n",
            "Steps: 70, Test accuracy: 0.20844686031341553\n",
            "Steps: 75, Test accuracy: 0.19891008734703064\n",
            "Steps: 80, Test accuracy: 0.23705722391605377\n",
            "Steps: 85, Test accuracy: 0.29427793622016907\n",
            "Steps: 90, Test accuracy: 0.3337874710559845\n",
            "Steps: 95, Test accuracy: 0.30790191888809204\n",
            "Steps: 100, Test accuracy: 0.30108991265296936\n",
            "Steps: 105, Test accuracy: 0.29564031958580017\n",
            "Steps: 110, Test accuracy: 0.3365122675895691\n",
            "Steps: 115, Test accuracy: 0.41416892409324646\n",
            "Steps: 120, Test accuracy: 0.41144412755966187\n",
            "Steps: 125, Test accuracy: 0.42098093032836914\n",
            "Steps: 130, Test accuracy: 0.401907354593277\n",
            "Steps: 135, Test accuracy: 0.42234331369400024\n",
            "Steps: 140, Test accuracy: 0.42234331369400024\n",
            "Steps: 145, Test accuracy: 0.42506811022758484\n",
            "Steps: 150, Test accuracy: 0.41825613379478455\n",
            "Steps: 155, Test accuracy: 0.41416892409324646\n",
            "Steps: 160, Test accuracy: 0.4427793025970459\n",
            "Steps: 165, Test accuracy: 0.4550408720970154\n",
            "Steps: 170, Test accuracy: 0.47275203466415405\n",
            "Steps: 175, Test accuracy: 0.48092642426490784\n",
            "Steps: 180, Test accuracy: 0.46049046516418457\n",
            "Steps: 185, Test accuracy: 0.48092642426490784\n",
            "Steps: 190, Test accuracy: 0.4564032554626465\n",
            "Steps: 195, Test accuracy: 0.46457764506340027\n",
            "Steps: 200, Test accuracy: 0.47138965129852295\n",
            "Steps: 205, Test accuracy: 0.47275203466415405\n",
            "Steps: 210, Test accuracy: 0.4577656686306\n",
            "Steps: 215, Test accuracy: 0.4591280519962311\n",
            "Steps: 220, Test accuracy: 0.46457764506340027\n",
            "Steps: 225, Test accuracy: 0.47820162773132324\n",
            "Steps: 230, Test accuracy: 0.48092642426490784\n",
            "Steps: 235, Test accuracy: 0.46594005823135376\n",
            "Steps: 240, Test accuracy: 0.47547683119773865\n",
            "Steps: 245, Test accuracy: 0.46594005823135376\n",
            "Steps: 250, Test accuracy: 0.4455040991306305\n",
            "Steps: 255, Test accuracy: 0.48365122079849243\n",
            "Steps: 260, Test accuracy: 0.47275203466415405\n",
            "Steps: 265, Test accuracy: 0.47683924436569214\n",
            "Steps: 270, Test accuracy: 0.48228883743286133\n",
            "Steps: 275, Test accuracy: 0.486376017332077\n",
            "Steps: 280, Test accuracy: 0.47138965129852295\n",
            "Steps: 285, Test accuracy: 0.47683924436569214\n",
            "Steps: 290, Test accuracy: 0.47547683119773865\n",
            "Steps: 295, Test accuracy: 0.47411444783210754\n",
            "Steps: 300, Test accuracy: 0.47547683119773865\n",
            "Steps: 305, Test accuracy: 0.47411444783210754\n",
            "Steps: 310, Test accuracy: 0.4891008138656616\n",
            "Steps: 315, Test accuracy: 0.486376017332077\n",
            "Steps: 320, Test accuracy: 0.4877384305000305\n",
            "Steps: 325, Test accuracy: 0.4850136339664459\n",
            "Steps: 330, Test accuracy: 0.48228883743286133\n",
            "Steps: 335, Test accuracy: 0.48365122079849243\n",
            "Steps: 340, Test accuracy: 0.48092642426490784\n",
            "Steps: 344, Test accuracy: 0.47956404089927673\n",
            "Steps: 345, Test accuracy: 0.47956404089927673\n",
            "temp=0.01, margin=0.1, eval_accuracy=0.47956404089927673\n",
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 5, Test accuracy: 0.24931880831718445\n",
            "Steps: 10, Test accuracy: 0.2193460464477539\n",
            "Steps: 15, Test accuracy: 0.23841962218284607\n",
            "Steps: 20, Test accuracy: 0.2888283431529999\n",
            "Steps: 25, Test accuracy: 0.3419618606567383\n",
            "Steps: 30, Test accuracy: 0.35967302322387695\n",
            "Steps: 35, Test accuracy: 0.31198909878730774\n",
            "Steps: 40, Test accuracy: 0.332425057888031\n",
            "Steps: 45, Test accuracy: 0.3801089823246002\n",
            "Steps: 50, Test accuracy: 0.3787465989589691\n",
            "Steps: 55, Test accuracy: 0.40871936082839966\n",
            "Steps: 60, Test accuracy: 0.3433242440223694\n",
            "Steps: 65, Test accuracy: 0.3787465989589691\n",
            "Steps: 70, Test accuracy: 0.41008174419403076\n",
            "Steps: 75, Test accuracy: 0.41689372062683105\n",
            "Steps: 80, Test accuracy: 0.4482288956642151\n",
            "Steps: 85, Test accuracy: 0.4591280519962311\n",
            "Steps: 90, Test accuracy: 0.48228883743286133\n",
            "Steps: 95, Test accuracy: 0.4536784887313843\n",
            "Steps: 100, Test accuracy: 0.4591280519962311\n",
            "Steps: 105, Test accuracy: 0.4931880235671997\n",
            "Steps: 110, Test accuracy: 0.47275203466415405\n",
            "Steps: 115, Test accuracy: 0.47956404089927673\n",
            "Steps: 120, Test accuracy: 0.47683924436569214\n",
            "Steps: 125, Test accuracy: 0.5095368027687073\n",
            "Steps: 130, Test accuracy: 0.5081743597984314\n",
            "Steps: 135, Test accuracy: 0.5231607556343079\n",
            "Steps: 140, Test accuracy: 0.5136239528656006\n",
            "Steps: 145, Test accuracy: 0.5095368027687073\n",
            "Steps: 150, Test accuracy: 0.5095368027687073\n",
            "Steps: 155, Test accuracy: 0.5040872097015381\n",
            "Steps: 160, Test accuracy: 0.5190735459327698\n",
            "Steps: 165, Test accuracy: 0.527247965335846\n",
            "Steps: 170, Test accuracy: 0.531335175037384\n",
            "Steps: 175, Test accuracy: 0.5326975584030151\n",
            "Steps: 180, Test accuracy: 0.5122615694999695\n",
            "Steps: 185, Test accuracy: 0.524523138999939\n",
            "Steps: 190, Test accuracy: 0.527247965335846\n",
            "Steps: 195, Test accuracy: 0.528610348701477\n",
            "Steps: 200, Test accuracy: 0.5190735459327698\n",
            "Steps: 205, Test accuracy: 0.5122615694999695\n",
            "Steps: 210, Test accuracy: 0.5149863958358765\n",
            "Steps: 215, Test accuracy: 0.524523138999939\n",
            "Steps: 220, Test accuracy: 0.528610348701477\n",
            "Steps: 225, Test accuracy: 0.5231607556343079\n",
            "Steps: 230, Test accuracy: 0.531335175037384\n",
            "Steps: 235, Test accuracy: 0.5258855819702148\n",
            "Steps: 240, Test accuracy: 0.5190735459327698\n",
            "Steps: 245, Test accuracy: 0.5367847681045532\n",
            "Steps: 250, Test accuracy: 0.524523138999939\n",
            "Steps: 255, Test accuracy: 0.528610348701477\n",
            "Steps: 260, Test accuracy: 0.5299727320671082\n",
            "Steps: 265, Test accuracy: 0.5204359889030457\n",
            "Steps: 270, Test accuracy: 0.5367847681045532\n",
            "Steps: 275, Test accuracy: 0.528610348701477\n",
            "Steps: 280, Test accuracy: 0.524523138999939\n",
            "Steps: 285, Test accuracy: 0.528610348701477\n",
            "Steps: 290, Test accuracy: 0.5299727320671082\n",
            "Steps: 295, Test accuracy: 0.527247965335846\n",
            "Steps: 300, Test accuracy: 0.528610348701477\n",
            "Steps: 305, Test accuracy: 0.5340599417686462\n",
            "Steps: 310, Test accuracy: 0.5367847681045532\n",
            "Steps: 315, Test accuracy: 0.5340599417686462\n",
            "Steps: 320, Test accuracy: 0.5354223251342773\n",
            "Steps: 325, Test accuracy: 0.5299727320671082\n",
            "Steps: 330, Test accuracy: 0.5381471514701843\n",
            "Steps: 335, Test accuracy: 0.5354223251342773\n",
            "Steps: 340, Test accuracy: 0.5354223251342773\n",
            "Steps: 344, Test accuracy: 0.5395095348358154\n",
            "Steps: 345, Test accuracy: 0.5408719182014465\n",
            "temp=0.01, margin=0.1, eval_accuracy=0.5408719182014465\n",
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 5, Test accuracy: 0.20435968041419983\n",
            "Steps: 10, Test accuracy: 0.19618529081344604\n",
            "Steps: 15, Test accuracy: 0.21253405511379242\n",
            "Steps: 20, Test accuracy: 0.20844686031341553\n",
            "Steps: 25, Test accuracy: 0.2792915403842926\n",
            "Steps: 30, Test accuracy: 0.25749319791793823\n",
            "Steps: 35, Test accuracy: 0.265667587518692\n",
            "Steps: 40, Test accuracy: 0.30517712235450745\n",
            "Steps: 45, Test accuracy: 0.31471389532089233\n",
            "Steps: 50, Test accuracy: 0.30381470918655396\n",
            "Steps: 55, Test accuracy: 0.2874659299850464\n",
            "Steps: 60, Test accuracy: 0.30381470918655396\n",
            "Steps: 65, Test accuracy: 0.332425057888031\n",
            "Steps: 70, Test accuracy: 0.3297002613544464\n",
            "Steps: 75, Test accuracy: 0.30517712235450745\n",
            "Steps: 80, Test accuracy: 0.31062671542167664\n",
            "Steps: 85, Test accuracy: 0.3446866571903229\n",
            "Steps: 90, Test accuracy: 0.36376020312309265\n",
            "Steps: 95, Test accuracy: 0.3446866571903229\n",
            "Steps: 100, Test accuracy: 0.31198909878730774\n",
            "Steps: 105, Test accuracy: 0.3228882849216461\n",
            "Steps: 110, Test accuracy: 0.35967302322387695\n",
            "Steps: 115, Test accuracy: 0.3814713954925537\n",
            "Steps: 120, Test accuracy: 0.35831063985824585\n",
            "Steps: 125, Test accuracy: 0.36103543639183044\n",
            "Steps: 130, Test accuracy: 0.3923705816268921\n",
            "Steps: 135, Test accuracy: 0.40326976776123047\n",
            "Steps: 140, Test accuracy: 0.401907354593277\n",
            "Steps: 145, Test accuracy: 0.3814713954925537\n",
            "Steps: 150, Test accuracy: 0.3760218024253845\n",
            "Steps: 155, Test accuracy: 0.42370572686195374\n",
            "Steps: 160, Test accuracy: 0.40735694766044617\n",
            "Steps: 165, Test accuracy: 0.40871936082839966\n",
            "Steps: 170, Test accuracy: 0.40735694766044617\n",
            "Steps: 175, Test accuracy: 0.41144412755966187\n",
            "Steps: 180, Test accuracy: 0.43051770329475403\n",
            "Steps: 185, Test accuracy: 0.41689372062683105\n",
            "Steps: 190, Test accuracy: 0.41008174419403076\n",
            "Steps: 195, Test accuracy: 0.4005449712276459\n",
            "Steps: 200, Test accuracy: 0.41961851716041565\n",
            "Steps: 205, Test accuracy: 0.41961851716041565\n",
            "Steps: 210, Test accuracy: 0.4291553199291229\n",
            "Steps: 215, Test accuracy: 0.43051770329475403\n",
            "Steps: 220, Test accuracy: 0.43051770329475403\n",
            "Steps: 225, Test accuracy: 0.4564032554626465\n",
            "Steps: 230, Test accuracy: 0.4414168894290924\n",
            "Steps: 235, Test accuracy: 0.43051770329475403\n",
            "Steps: 240, Test accuracy: 0.4495912790298462\n",
            "Steps: 245, Test accuracy: 0.4427793025970459\n",
            "Steps: 250, Test accuracy: 0.4455040991306305\n",
            "Steps: 255, Test accuracy: 0.4414168894290924\n",
            "Steps: 260, Test accuracy: 0.4332424998283386\n",
            "Steps: 265, Test accuracy: 0.4536784887313843\n",
            "Steps: 270, Test accuracy: 0.4536784887313843\n",
            "Steps: 275, Test accuracy: 0.4591280519962311\n",
            "Steps: 280, Test accuracy: 0.4564032554626465\n",
            "Steps: 285, Test accuracy: 0.4468664824962616\n",
            "Steps: 290, Test accuracy: 0.4523160755634308\n",
            "Steps: 295, Test accuracy: 0.4618528485298157\n",
            "Steps: 300, Test accuracy: 0.4591280519962311\n",
            "Steps: 305, Test accuracy: 0.46049046516418457\n",
            "Steps: 310, Test accuracy: 0.4550408720970154\n",
            "Steps: 315, Test accuracy: 0.4523160755634308\n",
            "Steps: 320, Test accuracy: 0.4564032554626465\n",
            "Steps: 325, Test accuracy: 0.4523160755634308\n",
            "Steps: 330, Test accuracy: 0.4536784887313843\n",
            "Steps: 335, Test accuracy: 0.4495912790298462\n",
            "Steps: 340, Test accuracy: 0.4468664824962616\n",
            "Steps: 344, Test accuracy: 0.4495912790298462\n",
            "Steps: 345, Test accuracy: 0.4495912790298462\n",
            "temp=0.01, margin=0.1, eval_accuracy=0.4495912790298462\n",
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 5, Test accuracy: 0.173024520277977\n",
            "Steps: 10, Test accuracy: 0.17847411334514618\n",
            "Steps: 15, Test accuracy: 0.19209809601306915\n",
            "Steps: 20, Test accuracy: 0.25885558128356934\n",
            "Steps: 25, Test accuracy: 0.30517712235450745\n",
            "Steps: 30, Test accuracy: 0.3269754648208618\n",
            "Steps: 35, Test accuracy: 0.35149863362312317\n",
            "Steps: 40, Test accuracy: 0.3760218024253845\n",
            "Steps: 45, Test accuracy: 0.3760218024253845\n",
            "Steps: 50, Test accuracy: 0.3787465989589691\n",
            "Steps: 55, Test accuracy: 0.3910081684589386\n",
            "Steps: 60, Test accuracy: 0.40599456429481506\n",
            "Steps: 65, Test accuracy: 0.40599456429481506\n",
            "Steps: 70, Test accuracy: 0.3828337788581848\n",
            "Steps: 75, Test accuracy: 0.3937329649925232\n",
            "Steps: 80, Test accuracy: 0.4359672963619232\n",
            "Steps: 85, Test accuracy: 0.4455040991306305\n",
            "Steps: 90, Test accuracy: 0.4291553199291229\n",
            "Steps: 95, Test accuracy: 0.42779290676116943\n",
            "Steps: 100, Test accuracy: 0.444141685962677\n",
            "Steps: 105, Test accuracy: 0.46730244159698486\n",
            "Steps: 110, Test accuracy: 0.4536784887313843\n",
            "Steps: 115, Test accuracy: 0.4577656686306\n",
            "Steps: 120, Test accuracy: 0.4523160755634308\n",
            "Steps: 125, Test accuracy: 0.47411444783210754\n",
            "Steps: 130, Test accuracy: 0.47002723813056946\n",
            "Steps: 135, Test accuracy: 0.46866485476493835\n",
            "Steps: 140, Test accuracy: 0.47820162773132324\n",
            "Steps: 145, Test accuracy: 0.47138965129852295\n",
            "Steps: 150, Test accuracy: 0.47820162773132324\n",
            "Steps: 155, Test accuracy: 0.4850136339664459\n",
            "Steps: 160, Test accuracy: 0.46866485476493835\n",
            "Steps: 165, Test accuracy: 0.47138965129852295\n",
            "Steps: 170, Test accuracy: 0.47820162773132324\n",
            "Steps: 175, Test accuracy: 0.4918256103992462\n",
            "Steps: 180, Test accuracy: 0.47275203466415405\n",
            "Steps: 185, Test accuracy: 0.47820162773132324\n",
            "Steps: 190, Test accuracy: 0.4850136339664459\n",
            "Steps: 195, Test accuracy: 0.47820162773132324\n",
            "Steps: 200, Test accuracy: 0.4904632270336151\n",
            "Steps: 205, Test accuracy: 0.4850136339664459\n",
            "Steps: 210, Test accuracy: 0.47683924436569214\n",
            "Steps: 215, Test accuracy: 0.47820162773132324\n",
            "Steps: 220, Test accuracy: 0.48365122079849243\n",
            "Steps: 225, Test accuracy: 0.4904632270336151\n",
            "Steps: 230, Test accuracy: 0.4972752034664154\n",
            "Steps: 235, Test accuracy: 0.47547683119773865\n",
            "Steps: 240, Test accuracy: 0.47547683119773865\n",
            "Steps: 245, Test accuracy: 0.4931880235671997\n",
            "Steps: 250, Test accuracy: 0.4986376166343689\n",
            "Steps: 255, Test accuracy: 0.48365122079849243\n",
            "Steps: 260, Test accuracy: 0.47820162773132324\n",
            "Steps: 265, Test accuracy: 0.48228883743286133\n",
            "Steps: 270, Test accuracy: 0.4945504069328308\n",
            "Steps: 275, Test accuracy: 0.4986376166343689\n",
            "Steps: 280, Test accuracy: 0.4891008138656616\n",
            "Steps: 285, Test accuracy: 0.4986376166343689\n",
            "Steps: 290, Test accuracy: 0.4904632270336151\n",
            "Steps: 295, Test accuracy: 0.4850136339664459\n",
            "Steps: 300, Test accuracy: 0.4877384305000305\n",
            "Steps: 305, Test accuracy: 0.4904632270336151\n",
            "Steps: 310, Test accuracy: 0.4945504069328308\n",
            "Steps: 315, Test accuracy: 0.4877384305000305\n",
            "Steps: 320, Test accuracy: 0.4918256103992462\n",
            "Steps: 325, Test accuracy: 0.4904632270336151\n",
            "Steps: 330, Test accuracy: 0.4918256103992462\n",
            "Steps: 335, Test accuracy: 0.4918256103992462\n",
            "Steps: 340, Test accuracy: 0.4945504069328308\n",
            "Steps: 344, Test accuracy: 0.4986376166343689\n",
            "Steps: 345, Test accuracy: 0.5013623833656311\n",
            "temp=0.01, margin=0.1, eval_accuracy=0.5013623833656311\n",
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:tensorflow:Using MirroredStrategy with devices ('/job:localhost/replica:0/task:0/device:CPU:0',)\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 5, Test accuracy: 0.20980925858020782\n",
            "Steps: 10, Test accuracy: 0.2207084447145462\n",
            "Steps: 15, Test accuracy: 0.24523161351680756\n",
            "Steps: 20, Test accuracy: 0.23569482564926147\n",
            "Steps: 25, Test accuracy: 0.2670299708843231\n",
            "Steps: 30, Test accuracy: 0.29972752928733826\n",
            "Steps: 35, Test accuracy: 0.30108991265296936\n",
            "Steps: 40, Test accuracy: 0.3283378779888153\n",
            "Steps: 45, Test accuracy: 0.3297002613544464\n",
            "Steps: 50, Test accuracy: 0.3242506682872772\n",
            "Steps: 55, Test accuracy: 0.29836511611938477\n",
            "Steps: 60, Test accuracy: 0.2888283431529999\n",
            "Steps: 65, Test accuracy: 0.3283378779888153\n",
            "Steps: 70, Test accuracy: 0.3242506682872772\n",
            "Steps: 75, Test accuracy: 0.3297002613544464\n",
            "Steps: 80, Test accuracy: 0.30108991265296936\n",
            "Steps: 85, Test accuracy: 0.3215258717536926\n",
            "Steps: 90, Test accuracy: 0.35694822669029236\n",
            "Steps: 95, Test accuracy: 0.35149863362312317\n",
            "Steps: 100, Test accuracy: 0.3392370641231537\n",
            "Steps: 105, Test accuracy: 0.3405994474887848\n",
            "Steps: 110, Test accuracy: 0.3732970058917999\n",
            "Steps: 115, Test accuracy: 0.3828337788581848\n",
            "Steps: 120, Test accuracy: 0.3855585753917694\n",
            "Steps: 125, Test accuracy: 0.35013625025749207\n",
            "Steps: 130, Test accuracy: 0.3978201746940613\n",
            "Steps: 135, Test accuracy: 0.41416892409324646\n",
            "Steps: 140, Test accuracy: 0.41008174419403076\n",
            "Steps: 145, Test accuracy: 0.4005449712276459\n",
            "Steps: 150, Test accuracy: 0.3828337788581848\n",
            "Steps: 155, Test accuracy: 0.4346049129962921\n",
            "Steps: 160, Test accuracy: 0.3978201746940613\n",
            "Steps: 165, Test accuracy: 0.43051770329475403\n",
            "Steps: 170, Test accuracy: 0.40326976776123047\n",
            "Steps: 175, Test accuracy: 0.40871936082839966\n",
            "Steps: 180, Test accuracy: 0.40871936082839966\n",
            "Steps: 185, Test accuracy: 0.3950953781604767\n",
            "Steps: 190, Test accuracy: 0.3923705816268921\n",
            "Steps: 195, Test accuracy: 0.4005449712276459\n",
            "Steps: 200, Test accuracy: 0.42370572686195374\n",
            "Steps: 205, Test accuracy: 0.401907354593277\n",
            "Steps: 210, Test accuracy: 0.41008174419403076\n",
            "Steps: 215, Test accuracy: 0.40871936082839966\n",
            "Steps: 220, Test accuracy: 0.41825613379478455\n",
            "Steps: 225, Test accuracy: 0.42643052339553833\n",
            "Steps: 230, Test accuracy: 0.41689372062683105\n",
            "Steps: 235, Test accuracy: 0.41416892409324646\n",
            "Steps: 240, Test accuracy: 0.41144412755966187\n",
            "Steps: 245, Test accuracy: 0.42098093032836914\n",
            "Steps: 250, Test accuracy: 0.42643052339553833\n",
            "Steps: 255, Test accuracy: 0.42779290676116943\n",
            "Steps: 260, Test accuracy: 0.4332424998283386\n",
            "Steps: 265, Test accuracy: 0.4386920928955078\n",
            "Steps: 270, Test accuracy: 0.444141685962677\n",
            "Steps: 275, Test accuracy: 0.4427793025970459\n",
            "Steps: 280, Test accuracy: 0.4373297095298767\n",
            "Steps: 285, Test accuracy: 0.4427793025970459\n",
            "Steps: 290, Test accuracy: 0.4523160755634308\n",
            "Steps: 295, Test accuracy: 0.4455040991306305\n",
            "Steps: 300, Test accuracy: 0.46049046516418457\n",
            "Steps: 305, Test accuracy: 0.4509536921977997\n",
            "Steps: 310, Test accuracy: 0.4468664824962616\n",
            "Steps: 315, Test accuracy: 0.4482288956642151\n",
            "Steps: 320, Test accuracy: 0.4482288956642151\n",
            "Steps: 325, Test accuracy: 0.4550408720970154\n",
            "Steps: 330, Test accuracy: 0.4550408720970154\n",
            "Steps: 335, Test accuracy: 0.4523160755634308\n",
            "Steps: 340, Test accuracy: 0.4550408720970154\n",
            "Steps: 344, Test accuracy: 0.4536784887313843\n",
            "Steps: 345, Test accuracy: 0.4536784887313843\n",
            "temp=0.01, margin=0.1, eval_accuracy=0.4536784887313843\n",
            "Temp: 0.01, Margin: 0.1, Eval Accuracy: 0.5422343611717224\n",
            "Temp: 0.01, Margin: 0.1, Eval Accuracy: 0.47956404089927673\n",
            "Temp: 0.01, Margin: 0.1, Eval Accuracy: 0.5408719182014465\n",
            "Temp: 0.01, Margin: 0.1, Eval Accuracy: 0.4495912790298462\n",
            "Temp: 0.01, Margin: 0.1, Eval Accuracy: 0.5013623833656311\n",
            "Temp: 0.01, Margin: 0.1, Eval Accuracy: 0.4536784887313843\n"
          ]
        }
      ],
      "source": [
        "rows = []\n",
        "cols = ['temp', 'margin', 'eval_accuracy']\n",
        "\n",
        "rank_matrix = []\n",
        "loss_matrix = []\n",
        "acc_matrix = []\n",
        "\n",
        "bits=0\n",
        "\n",
        "for temp in [0.01]:\n",
        "  for bits in [0, 5, 10, 15, 20]:\n",
        "    steps, accuracies, rank, loss_total = train_and_eval(\n",
        "        batch_size=128,\n",
        "        width_multiplier=1,\n",
        "        extra_channel_bits=bits,\n",
        "        nt_xent_temp=temp, # [1,5,10]\n",
        "        learning_rate=0.005,\n",
        "        epochs=15,\n",
        "        eval_every_n_steps=5,\n",
        "        print_output=True)\n",
        "    rows.append([temp, bits, accuracies[-1].numpy()])\n",
        "    rank_matrix.append(rank)\n",
        "    loss_matrix.append(loss_total)\n",
        "    acc_matrix.append(accuracies)\n",
        "    print(\"temp={}, bits{}, eval_accuracy={}\".format(\n",
        "        temp, bits, accuracies[-1]))\n",
        "plot_df = pd.DataFrame.from_records(rows, columns=cols)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 19,
      "metadata": {},
      "outputs": [
        {
          "data": {
            "text/plain": [
              "[[0.01, 0.1, 0.54223436],\n",
              " [0.01, 0.1, 0.47956404],\n",
              " [0.01, 0.1, 0.5408719],\n",
              " [0.01, 0.1, 0.44959128],\n",
              " [0.01, 0.1, 0.5013624],\n",
              " [0.01, 0.1, 0.4536785]]"
            ]
          },
          "execution_count": 19,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": []
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 430
        },
        "id": "qHXHOxDWlwWW",
        "outputId": "254d472a-6b0e-4c6e-dec7-67f76b850fba"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x7f040baa69e0>]"
            ]
          },
          "execution_count": 12,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "ranks_lis = [tensor.numpy() for tensor in rank_matrix[0]]\n",
        "rank1=ranks_lis\n",
        "x = range(1, len(rank1) + 1)\n",
        "plt.plot(x, rank1)\n",
        "#plt.savefig('rank_nfda_0.001_0.01.png')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "data": {
            "text/plain": [
              "[<tf.Tensor: shape=(), dtype=float32, numpy=0.1852861>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.24659401>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.2520436>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.253406>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.25613078>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.25613078>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.26430517>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.26702997>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.26158038>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.27247956>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.27520436>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.28746593>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.29564032>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.29972753>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.3119891>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.32561308>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.33106267>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.34059945>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.35558584>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.35422343>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.373297>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.39237058>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.40326977>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.41144413>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4223433>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4386921>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4427793>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.44686648>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4346049>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4482289>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4373297>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4482289>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.45776567>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.45231608>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4455041>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4373297>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4414169>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4373297>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4386921>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42915532>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4277929>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4332425>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42915532>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.41961852>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42098093>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.41689372>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42098093>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.41416892>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.41689372>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42098093>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42098093>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42370573>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42643052>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4250681>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.4223433>,\n",
              " <tf.Tensor: shape=(), dtype=float32, numpy=0.42098093>]"
            ]
          },
          "execution_count": 13,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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ro4fPz6sn/NDP9AHAfTDGfTDGfQDfRDQn34wxujUqVPyer1KSOh9lMmQrKAkJCRo3LhxUf2OtLS0YX3hBdAP9EEA/XAG/XAG/UAfBETaD31VTgIYJAsAAGIOAQUAAMQcAkoIj8ejtWvXyuPxON0UR9EP9EEA/XAG/XAG/UAfBES7H4bkIFkAABDfqKAAAICYQ0ABAAAxh4ACAABiDgEFAADEHALKp5SVleniiy/WBRdcoFmzZul3v/ud002KqnXr1snlcgX9XHrppdb21tZW3XPPPcrKytLIkSN100036cMPP3SwxYNj165dWrRokbxer1wul8rLy4O2G2P00EMPKScnRyNGjND8+fN1+PDhoH0aGhq0dOlSpaWlKSMjQ3fccYeamppsPIuB6asPVqxYcd61UVRUFLTPUO8DSSotLdVf/MVfaNSoURo7dqxuvPFGVVVVBe0Tzu9BbW2tFi5cqJSUFI0dO1b333+/Ojs77TyVfgunDwoLC8+7Hu6+++6gfYZyH0jShg0bNH36dGvRsdmzZ+ull16ytsf7dRDQVz/Yei0YGGOMefbZZ43b7TY//elPzaFDh8zKlStNRkaG+fDDD51uWtSsXbvWXH755eaDDz6wfj766CNr+913321yc3PNq6++avbu3Ws+//nPm6uvvtrBFg+OF1980fzTP/2T2bRpk5FkNm/eHLT9X//1X016eropLy83v//97831119vJkyYYE6fPm3tU1RUZPLz881vf/tb8/rrr5vJkyebJUuW2Hwm/ddXHyxfvtwUFRUFXRsNDQ1B+wz1PjDGmGuvvdY8+eST5uDBg6aystJcd911Ji8vzzQ1NVn79PV70NnZaa644gozf/58c+DAAfPiiy+a0aNHm5KSEidOKWLh9MFf/uVfmpUrVwZdD42Njdb2od4HxhizZcsW8z//8z/mj3/8o6mqqjIPPvigSU5ONgcPHjTGxP91ENBXP9h5LRBQzpo5c6a55557rNddXV3G6/Wa0tJSB1sVXWvXrjX5+fndbjt58qRJTk42//3f/22998477xhJpqKiwqYWRl/oH2e/32+ys7PNv/3bv1nvnTx50ng8HvPzn//cGGPMH/7wByPJ7Nmzx9rnpZdeMi6Xy7z//vu2tX2w9BRQbrjhhh4/E299EHDixAkjybz22mvGmPB+D1588UWTkJBg6uvrrX02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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "plt.plot([round(x) for x in rank1])\n",
        "round_rank = [round(x) for x in rank1]\n",
        "np.array(round_rank)\n",
        "\n",
        "accuracies\n",
        "#np.save('rank_nfda_0.001_0.01_digit_4_num_30.npy', np.array(round_rank))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [],
      "source": [
        "#np.save('rank_nfda_0.001_0.01_digit_9_num_30.npy', np.array(round_rank))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<tf.Tensor: shape=(), dtype=float32, numpy=0.4414169>"
            ]
          },
          "execution_count": 15,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "#np.save('loss_nfda_0.001_0.01_digit_9_num_30.npy', np.array(loss_matrix[0]))\n",
        "\n",
        "accuracies[50]"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x7f030c38bdf0>]"
            ]
          },
          "execution_count": 16,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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7zQpKSlmy5Ytxpie8V041Rgk8ntAmDlNF110kZk1a1b4ud/vN/369TOLFi2ysFfxc/fdd5sLL7yw3X2HDx82KSkp5s9//nP4tW3bthlJpry8PEE9jL+v/hEPBAKmsLDQLF68OPza4cOHjdvtNs8884wxxpiPPvrISDLvv/9+uM2rr75qbDab2bNnT8L6HksnCjNXXXXVCd/THcdh//79RpJ56623jDEd+3fwyiuvGLvdbqqqqsJtHnnkEZOVlWV8Pl9iP0CMfHUcjAn+EfvZz352wvd0x3Ewxpjc3Fzz2GOP9djvgjHHxsCYxH4POM10GpqamrRhwwZNmDAh/JrdbteECRNUXl5uYc/i65NPPlG/fv00aNAgff/731dlZaUkacOGDWpubo4Yj3POOUf9+/fv1uOxc+dOVVVVRXzu7OxsjR49Ovy5y8vLlZOTo69//evhNhMmTJDdbtd7772X8D7H07p169SnTx8NHTpUM2fO1MGDB8P7uuM41NbWSpJ69eolqWP/DsrLy3XBBReooKAg3KasrExer1dbt25NYO9j56vjEPL0008rLy9P559/vubNm6ejR4+G93W3cfD7/Xr22Wd15MgRlZaW9sjvwlfHICRR34MecaPJWDtw4ID8fn/EfwBJKigoUEVFhUW9iq/Ro0dr+fLlGjp0qPbt26eFCxfq0ksv1ZYtW1RVVSWXy6WcnJyI9xQUFKiqqsqaDidA6LO19z0I7auqqlKfPn0i9judTvXq1atbjc2kSZP0ne98RwMHDtSnn36qO++8U1dccYXKy8vlcDi63TgEAgHdeuutuuSSS3T++edLUof+HVRVVbX7fQntSzbtjYMkXX/99RowYID69eunTZs26Y477tD27dv117/+VVL3GYfNmzertLRUjY2NysjI0AsvvKBzzz1XHo+nx3wXTjQGUmK/B4QZdMgVV1wRfjx8+HCNHj1aAwYM0HPPPae0tDQLe4au4Nprrw0/vuCCCzR8+HCdffbZWrdunS6//HILexYfs2bN0pYtW7R+/Xqru2KpE43DzTffHH58wQUXqG/fvrr88sv16aef6uyzz050N+Nm6NCh8ng8qq2t1fPPP68bb7xRb731ltXdSqgTjcG5556b0O8Bp5lOQ15enhwOx3GV6dXV1SosLLSoV4mVk5OjIUOGaMeOHSosLFRTU5MOHz4c0aa7j0fos53se1BYWKj9+/dH7G9paVFNTU23HptBgwYpLy9PO3bskNS9xuGWW27R3//+d61du1ZFRUXh1zvy76CwsLDd70toXzI50Ti0Z/To0ZIU8X3oDuPgcrlUUlKiUaNGadGiRbrwwgu1dOnSHvVdONEYtCee3wPCzGlwuVwaNWqU3nzzzfBrgUBAb775ZsS5wu6svr5en376qfr27atRo0YpJSUlYjy2b9+uysrKbj0eAwcOVGFhYcTn9nq9eu+998Kfu7S0VIcPH9aGDRvCbdasWaNAIBD+h90dffHFFzp48KD69u0rqXuMgzFGt9xyi1544QWtWbNGAwcOjNjfkX8HpaWl2rx5c0SwW716tbKyssJT813dqcahPR6PR5Iivg/JPg7tCQQC8vl8Pea70J7QGLQnrt+D0yhWhglemu12u83y5cvNRx99ZG6++WaTk5MTUZXdncyZM8esW7fO7Ny507z99ttmwoQJJi8vz+zfv98YE7wMsX///mbNmjXmgw8+MKWlpaa0tNTiXkevrq7ObNy40WzcuNFIMkuWLDEbN240n3/+uTEmeGl2Tk6Oeemll8ymTZvMVVdd1e6l2SNGjDDvvfeeWb9+vRk8eHBSXZJszMnHoa6uzvz85z835eXlZufOnea///u/zciRI83gwYNNY2Nj+BjJPg4zZ8402dnZZt26dRGXmh49ejTc5lT/DkKXok6cONF4PB7z2muvmfz8/KS6HPdU47Bjxw5z7733mg8++MDs3LnTvPTSS2bQoEFm7Nix4WN0h3GYO3eueeutt8zOnTvNpk2bzNy5c43NZjNvvPGGMaZnfBdONgaJ/h4QZqLwhz/8wfTv39+4XC5z0UUXmXfffdfqLsXNNddcY/r27WtcLpc588wzzTXXXGN27NgR3t/Q0GB+/OMfm9zcXJOenm7+7d/+zezbt8/CHsfG2rVrjaTjthtvvNEYE7w8+6677jIFBQXG7Xabyy+/3Gzfvj3iGAcPHjTXXXedycjIMFlZWeYHP/iBqaurs+DTnL6TjcPRo0fNxIkTTX5+vklJSTEDBgwwN91003HBPtnHob3PL8k8+eST4TYd+Xewa9cuc8UVV5i0tDSTl5dn5syZY5qbmxP8aU7fqcahsrLSjB071vTq1cu43W5TUlJifvGLX0SsL2JM8o/DtGnTzIABA4zL5TL5+fnm8ssvDwcZY3rGd+FkY5Do74HNGGM6N5cDAADQdVAzAwAAkhphBgAAJDXCDAAASGqEGQAAkNQIMwAAIKkRZgAAQFIjzAAAgKRGmAEAAEmNMAMAAJIaYQYAACQ1wgwAAEhqhBkAAJDU/i/w8xxMZsSd2AAAAABJRU5ErkJggg==",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "plt.plot(np.array(loss_matrix[0]))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [],
      "source": [
        "array = np.load('rank_nfda_0.001_0.01_digit_4_num_30.npy')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "0.45776567\n"
          ]
        }
      ],
      "source": [
        "acc = [tensor.numpy() for tensor in acc_matrix[0]]\n",
        "#x = [i * 5  for i in range(460 // 5)]\n",
        "#len(acc)\n",
        "#plt.plot(x,acc)\n",
        "\n",
        "#np.save('acc_nfda_0.001_0.01_digit_4_num_30.npy', np.array(acc))\n",
        "#plt.plot(x,np.load('acc_nfda_0.001_0.01.npy'))\n",
        "print(max(acc))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "data": {
            "text/plain": [
              "92"
            ]
          },
          "execution_count": 19,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "len([i * 5  for i in range(460 // 5)])"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "[-139.31558, -157.02217, -168.84363, -175.64621, -179.67056, -182.09587, -183.91832, -185.34322, -186.59682, -187.61418, -188.53542, -189.34764, -190.02965, -190.64409, -191.1958, -191.68774, -192.1235, -192.50067, -192.81238, -193.12622, -193.41733, -193.6545, -193.90126, -194.11818, -194.3105, -194.49962, -194.68195, -194.85109, -195.00876, -195.15446, -195.29391, -195.42009, -195.54517, -195.65808, -195.76651, -195.8647, -195.96474, -196.06046, -196.1489, -196.23729, -196.32243, -196.40404, -196.48001, -196.5412, -196.60767, -196.67372, -196.73965, -196.80058, -196.85306, -196.9093, -196.9662, -197.0203, -197.07455, -197.12561, -197.17017, -197.21562, -197.25977, -197.3023, -197.34511, -197.38567, -197.42664, -197.46544, -197.50107, -197.53064, -197.56512, -197.60121, -197.6358, -197.66655, -197.69653, -197.72684, -197.75677, -197.78435, -197.81346, -197.84023, -197.86627, -197.89177, -197.9138, -197.93741, -197.96004, -197.98314, -198.0043, -198.02487, -198.0438, -198.06546, -198.08536, -198.10576, -198.12468, -198.14595, -198.16571, -198.18335, -198.2016, -198.22064, -198.23795, -198.25609, -198.27284, -198.2903, -198.30644, -198.32329, -198.33945, -198.35336, -198.3689, -198.38266, -198.3979, -198.41187, -198.4251, -198.43816, -198.45154, -198.46465, -198.47818, -198.49135, -198.5053, -198.51843, -198.53116, -198.54436, -198.5549, -198.56691, -198.57849, -198.59, -198.60149, -198.61311, -198.62468, -198.63612, -198.64708, -198.65791, -198.66794, -198.6783, -198.68797, -198.69807, -198.70793, -198.71802, -198.72815, -198.73744, -198.74525, -198.75488, -198.7648, -198.77405, -198.78308, -198.79152, -198.80022, -198.80779, -198.81577, -198.82236, -198.83015, -198.83893, -198.84683, -198.85477, -198.86241, -198.87004, -198.87766, -198.8849, -198.89215, -198.89906, -198.90652, -198.91425, -198.92183, -198.92863, -198.93604, -198.94371, -198.95021, -198.95656, -198.96333, -198.96991, -198.97615, -198.98294, -198.98875, -198.99478, -199.00043, -199.00623, -199.01132, -199.01689, -199.02228, -199.02832, -199.03397, -199.03955, -199.04451, -199.0502, -199.05591, -199.06161, -199.0665, -199.07191, -199.07777, -199.0827, -199.08807, -199.09311, -199.09827, -199.10324, -199.10829, -199.113, -199.11778, -199.1227, -199.12721, -199.1311, -199.13536, -199.1381, -199.1428, -199.1469, -199.15126, -199.15518, -199.15942, -199.16386, -199.16806, -199.17178, -199.17572, -199.18011, -199.18314, -199.18701, -199.19069, -199.19426, -199.19778, -199.20183, -199.2052, -199.20917, -199.21284, -199.21626, -199.21907, -199.22203, -199.22379, -199.22726, -199.23045, -199.23372, -199.23668, -199.2398, -199.2434, -199.24666, -199.24976, -199.2528, -199.2561, -199.25879, -199.26111, -199.26338, -199.26544, -199.26787, -199.27089, -199.2732, -199.27637, -199.27924, -199.28178, -199.28357, -199.2859, -199.28731, -199.28963, -199.29156, -199.29361, -199.29527, -199.29721, -199.29984, -199.3022, -199.30438, -199.30717, -199.30978, -199.31184, -199.31349, -199.31516, -199.31561, -199.31693, -199.31868, -199.31981, -199.32156, -199.32379, -199.32552, -199.32716, -199.32939, -199.33063, -199.3325, -199.33365, -199.33507, -199.3362, -199.33696, -199.33826, -199.33916, -199.34032, -199.34277, -199.3448, -199.34622, -199.3479, -199.34978, -199.35098, -199.35187, -199.3532, -199.3536, -199.35481, -199.35608, -199.35686, -199.35773, -199.35944, -199.36087, -199.36243, -199.36444, -199.36623, -199.3681, -199.36975, -199.3714, -199.372, -199.37317, -199.37502, -199.37552, -199.37619, -199.37755, -199.37904, -199.38104, -199.38248, -199.3846, -199.38628, -199.38843, -199.39044, -199.39186, -199.39311, -199.39471, -199.39597, -199.39717, -199.39885, -199.40022, -199.40218, -199.40411, -199.40631, -199.4083, -199.41046, -199.41248, -199.41469, -199.41623, -199.41794, -199.41988, -199.42184, -199.42319, -199.4252, -199.42705, -199.42941, -199.4312, -199.43315, -199.43509, -199.43684, -199.4386, -199.44037, -199.44243, -199.44403, -199.4461, -199.44803, -199.45012, -199.45187, -199.45384, -199.45578, -199.4579, -199.4595, -199.46114]\n"
          ]
        },
        {
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x7f030c2185e0>]"
            ]
          },
          "execution_count": 20,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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xw4VFBSE7QteDZ6Zmam33npLM2fO1IgRI5STk6P58+eH1piRpFGjRulPf/qTfvnLX+qee+7RwIED9fLLL7PGDAAAkBTndWaswjozAAAknvb+/e4aS2gCAIBuizADAAASGmEGAAAkNMIMAABIaIQZAACQ0AgzAAAgocX1rtlWCV59zt2zAQBIHMG/22daRaZbhJmamhpJ4u7ZAAAkoJqaGmVmZp5yf7dYNC8QCGjv3r1KT0+XzWaLyjG9Xq8KCwu1e/fubr0QH+PQjHFgDIIYh2aMA2MQFMk4GGNUU1OjPn36nHR/x9a6xcyM3W4/6ZYK0ZKRkdGtv6RBjEMzxoExCGIcmjEOjEHQ2Y7D6WZkgigABgAACY0wAwAAEhph5iy53W7dd999crvdVnfFUoxDM8aBMQhiHJoxDoxBUDzGoVsUAAMAgK6LmRkAAJDQCDMAACChEWYAAEBCI8wAAICERpg5S48++qjOO+88JScna+TIkfrf//1fq7sUM/fff79sNlvYdsEFF4T219fXa+bMmerZs6fS0tL03e9+V5WVlRb2ODr+53/+R9/+9rfVp08f2Ww2vfzyy2H7jTGaP3++evfurZSUFI0fP16fffZZWJuqqip9//vfV0ZGhrKysjRt2jTV1tbG8VNE7kzjMGXKlJO+HxMnTgxrk+jjsGjRIn3jG99Qenq6evXqpeuuu07bt28Pa9Oefwfl5eWaNGmSUlNT1atXL/3iF79QU1NTPD9KRNozDmPGjDnp+zB9+vSwNok8Do899piGDh0aWgCuuLhYr7/+emh/d/geSGceh7h/Dww67Pnnnzcul8s89dRTZuvWrebWW281WVlZprKy0uquxcR9991nLrroIrNv377QduDAgdD+6dOnm8LCQvP222+bjz76yFx++eVm1KhRFvY4Ol577TUzb94889e//tVIMi+99FLY/l//+tcmMzPTvPzyy+bjjz82//qv/2r69+9v6urqQm0mTpxoLr30UvP++++bf/zjH6aoqMjceOONcf4kkTnTONxyyy1m4sSJYd+PqqqqsDaJPg4lJSXm6aefNlu2bDEej8dcc801pm/fvqa2tjbU5kz/DpqamszFF19sxo8fbzZu3Ghee+01k5OTY+bOnWvFRzor7RmHq666ytx6661h34fq6urQ/kQfh7/97W9m1apV5tNPPzXbt28399xzj0lKSjJbtmwxxnSP74ExZx6HeH8PCDNn4bLLLjMzZ84MPff7/aZPnz5m0aJFFvYqdu677z5z6aWXtrnvyJEjJikpyfz5z38OvbZt2zYjyZSWlsaph7F34h/xQCBg8vPzzeLFi0OvHTlyxLjdbvPcc88ZY4z55JNPjCTz4Ycfhtq8/vrrxmazmT179sSt79F0qjBz7bXXnvI9XXEc9u/fbySZd955xxjTvn8Hr732mrHb7aaioiLU5rHHHjMZGRnG5/PF9wNEyYnjYEzzH7Gf/exnp3xPVxyH7Oxs88QTT3Tb70FQcByMif/3gNNMHdTQ0KANGzZo/PjxodfsdrvGjx+v0tJSC3sWW5999pn69OmjAQMG6Pvf/77Ky8slSRs2bFBjY2PYeFxwwQXq27dvlx6PnTt3qqKiIuxzZ2ZmauTIkaHPXVpaqqysLH39618PtRk/frzsdrs++OCDuPc5ltatW6devXpp8ODBmjFjhg4dOhTa1xXHobq6WpLUo0cPSe37d1BaWqpLLrlEeXl5oTYlJSXyer3aunVrHHsfPSeOQ9Czzz6rnJwcXXzxxZo7d66OHTsW2teVxsHv9+v555/X0aNHVVxc3G2/ByeOQ1A8vwfd4kaT0XTw4EH5/f6w/wCSlJeXp7KyMot6FVsjR47U8uXLNXjwYO3bt08LFizQlVdeqS1btqiiokIul0tZWVlh78nLy1NFRYU1HY6D4Gdr63sQ3FdRUaFevXqF7Xc6nerRo0eXGpuJEyfqO9/5jvr376/PP/9c99xzj66++mqVlpbK4XB0uXEIBAK64447dMUVV+jiiy+WpHb9O6ioqGjz+xLcl2jaGgdJuummm9SvXz/16dNHmzZt0t13363t27frr3/9q6SuMQ6bN29WcXGx6uvrlZaWppdeekkXXnihPB5Pt/oenGocpPh/DwgzOKOrr7469Hjo0KEaOXKk+vXrpxdeeEEpKSkW9gydwQ033BB6fMkll2jo0KE6//zztW7dOo0bN87CnsXGzJkztWXLFq1fv97qrljqVONw2223hR5fcskl6t27t8aNG6fPP/9c559/fry7GRODBw+Wx+NRdXW1XnzxRd1yyy165513rO5W3J1qHC688MK4fw84zdRBOTk5cjgcJ1WnV1ZWKj8/36JexVdWVpYGDRqkHTt2KD8/Xw0NDTpy5EhYm64+HsHPdrrvQX5+vvbv3x+2v6mpSVVVVV16bAYMGKCcnBzt2LFDUtcah9tvv11///vftXbtWhUUFIReb8+/g/z8/Da/L8F9ieRU49CWkSNHSlLY9yHRx8HlcqmoqEgjRozQokWLdOmll2rp0qXd7ntwqnFoS6y/B4SZDnK5XBoxYoTefvvt0GuBQEBvv/122LnCrqy2tlaff/65evfurREjRigpKSlsPLZv367y8vIuPR79+/dXfn5+2Of2er364IMPQp+7uLhYR44c0YYNG0Jt1qxZo0AgEPqH3RV99dVXOnTokHr37i2pa4yDMUa33367XnrpJa1Zs0b9+/cP29+efwfFxcXavHlzWLBbvXq1MjIyQlPznd2ZxqEtHo9HksK+D4k+DicKBALy+Xzd5ntwKsFxaEvMvwcdLhmGef75543b7TbLly83n3zyibnttttMVlZWWFV2VzJ79myzbt06s3PnTvPuu++a8ePHm5ycHLN//35jTPOliH379jVr1qwxH330kSkuLjbFxcUW9zpyNTU1ZuPGjWbjxo1GklmyZInZuHGj+fLLL40xzZdmZ2VlmVdeecVs2rTJXHvttW1emj1s2DDzwQcfmPXr15uBAwcm1CXJxpx+HGpqaszPf/5zU1paanbu3Gn++7//2wwfPtwMHDjQ1NfXh46R6OMwY8YMk5mZadatWxd2qemxY8dCbc707yB4KeqECROMx+Mxb7zxhsnNzU2oS3LPNA47duwwDzzwgPnoo4/Mzp07zSuvvGIGDBhgRo8eHTpGoo/DnDlzzDvvvGN27txpNm3aZObMmWNsNpt56623jDHd43tgzOnHwYrvAWHmLP3hD38wffv2NS6Xy1x22WXm/ffft7pLMXP99deb3r17G5fLZc4991xz/fXXmx07doT219XVmR//+McmOzvbpKammn/7t38z+/bts7DH0bF27Voj6aTtlltuMcY0X5597733mry8PON2u824cePM9u3bw45x6NAhc+ONN5q0tDSTkZFhfvCDH5iamhoLPs3ZO904HDt2zEyYMMHk5uaapKQk069fP3PrrbeeFOwTfRza+vySzNNPPx1q055/B7t27TJXX321SUlJMTk5OWb27NmmsbExzp/m7J1pHMrLy83o0aNNjx49jNvtNkVFReYXv/hF2PoixiT2OEydOtX069fPuFwuk5uba8aNGxcKMsZ0j++BMacfByu+BzZjjOn4fA4AAEDnQM0MAABIaIQZAACQ0AgzAAAgoRFmAABAQiPMAACAhEaYAQAACY0wAwAAEhphBgAAJDTCDAAASGiEGQAAkNAIMwAAIKERZgAAQEL7vxRxHExsa8w6AAAAAElFTkSuQmCC",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "loss_lis = [tensor.numpy() for tensor in loss_total]\n",
        "print(loss_lis)\n",
        "x = range(1, len(loss_lis) + 1)\n",
        "plt.plot(x, loss_lis)\n",
        "#plt.savefig('loss_ff_nfda_0.5.png')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "ename": "NameError",
          "evalue": "name 'nfda_loss' is not defined",
          "output_type": "error",
          "traceback": [
            "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
            "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
            "Cell \u001b[0;32mIn[21], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m nfda_loss_lis \u001b[38;5;241m=\u001b[39m [tensor\u001b[38;5;241m.\u001b[39mnumpy() \u001b[38;5;28;01mfor\u001b[39;00m tensor \u001b[38;5;129;01min\u001b[39;00m \u001b[43mnfda_loss\u001b[49m]\n\u001b[1;32m      2\u001b[0m \u001b[38;5;28mprint\u001b[39m(nfda_loss_lis)\n\u001b[1;32m      3\u001b[0m x \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mrange\u001b[39m(\u001b[38;5;241m1\u001b[39m, \u001b[38;5;28mlen\u001b[39m(nfda_loss_lis) \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m)\n",
            "\u001b[0;31mNameError\u001b[0m: name 'nfda_loss' is not defined"
          ]
        }
      ],
      "source": [
        "nfda_loss_lis = [tensor.numpy() for tensor in nfda_loss]\n",
        "print(nfda_loss_lis)\n",
        "x = range(1, len(nfda_loss_lis) + 1)\n",
        "plt.plot(x, nfda_loss_lis)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [
        {
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x7fad98252f80>]"
            ]
          },
          "execution_count": 38,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "plt.plot([tensor.numpy() for tensor in rank_matrix[4]])\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 143
        },
        "id": "EPXBBte939bI",
        "outputId": "137a2302-2ab6-4068-f405-6d2e4980c5bc"
      },
      "outputs": [
        {
          "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>temp</th>\n",
              "      <th>margin</th>\n",
              "      <th>eval_accuracy</th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th>0</th>\n",
              "      <td>0.1</td>\n",
              "      <td>0.1</td>\n",
              "      <td>0.463215</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>1</th>\n",
              "      <td>0.2</td>\n",
              "      <td>0.1</td>\n",
              "      <td>0.456403</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>2</th>\n",
              "      <td>0.3</td>\n",
              "      <td>0.1</td>\n",
              "      <td>0.470027</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>3</th>\n",
              "      <td>0.4</td>\n",
              "      <td>0.1</td>\n",
              "      <td>0.395095</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>4</th>\n",
              "      <td>0.5</td>\n",
              "      <td>0.1</td>\n",
              "      <td>0.453678</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n",
              "</div>"
            ],
            "text/plain": [
              "   temp  margin  eval_accuracy\n",
              "0   0.1     0.1       0.463215\n",
              "1   0.2     0.1       0.456403\n",
              "2   0.3     0.1       0.470027\n",
              "3   0.4     0.1       0.395095\n",
              "4   0.5     0.1       0.453678"
            ]
          },
          "execution_count": 27,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "plot_df"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "FwsVmmFOez0e",
        "outputId": "11daac9d-f40d-4661-abc2-3f587f4100fe"
      },
      "outputs": [
        {
          "ename": "TypeError",
          "evalue": "train_and_eval() got an unexpected keyword argument 'margin'",
          "output_type": "error",
          "traceback": [
            "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
            "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
            "Cell \u001b[0;32mIn[29], line 13\u001b[0m\n\u001b[1;32m      8\u001b[0m \u001b[38;5;66;03m# for bits in [0]:#[0, 5, 10, 15, 20]:\u001b[39;00m\n\u001b[1;32m      9\u001b[0m \u001b[38;5;66;03m#   for margin in [0.05]: #[0.05,0.1,0.2]:\u001b[39;00m\n\u001b[1;32m     10\u001b[0m \u001b[38;5;66;03m# for temp in [0.05, 0.1, 0.2]:\u001b[39;00m\n\u001b[1;32m     11\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m temp \u001b[38;5;129;01min\u001b[39;00m [\u001b[38;5;241m0.05\u001b[39m, \u001b[38;5;241m0.1\u001b[39m, \u001b[38;5;241m0.2\u001b[39m]:\n\u001b[1;32m     12\u001b[0m   \u001b[38;5;66;03m#for bits in [0, 5, 10, 15, 20]:\u001b[39;00m\n\u001b[0;32m---> 13\u001b[0m   steps, accuracies \u001b[38;5;241m=\u001b[39m \u001b[43mtrain_and_eval\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m     14\u001b[0m \u001b[43m      \u001b[49m\u001b[43mbatch_size\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m128\u001b[39;49m\u001b[43m,\u001b[49m\n\u001b[1;32m     15\u001b[0m \u001b[43m      \u001b[49m\u001b[43mwidth_multiplier\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m,\u001b[49m\n\u001b[1;32m     16\u001b[0m \u001b[43m      \u001b[49m\u001b[43mextra_channel_bits\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mbits\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     17\u001b[0m \u001b[43m      \u001b[49m\u001b[43mnt_xent_temp\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mtemp\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;66;43;03m# [1,5,10]\u001b[39;49;00m\n\u001b[1;32m     18\u001b[0m \u001b[43m      \u001b[49m\u001b[43mmargin\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mmargin\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     19\u001b[0m \u001b[43m      \u001b[49m\u001b[43mlearning_rate\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m0.005\u001b[39;49m\u001b[43m,\u001b[49m\n\u001b[1;32m     20\u001b[0m \u001b[43m      \u001b[49m\u001b[43mepochs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m10\u001b[39;49m\u001b[43m,\u001b[49m\n\u001b[1;32m     21\u001b[0m \u001b[43m      \u001b[49m\u001b[43meval_every_n_steps\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1000\u001b[39;49m\u001b[43m,\u001b[49m\n\u001b[1;32m     22\u001b[0m \u001b[43m      \u001b[49m\u001b[43mprint_output\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m)\u001b[49m\n\u001b[1;32m     23\u001b[0m   rows\u001b[38;5;241m.\u001b[39mappend([temp, margin, accuracies[\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m]\u001b[38;5;241m.\u001b[39mnumpy()])\n\u001b[1;32m     24\u001b[0m   \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mtemp=\u001b[39m\u001b[38;5;132;01m{}\u001b[39;00m\u001b[38;5;124m, margin=\u001b[39m\u001b[38;5;132;01m{}\u001b[39;00m\u001b[38;5;124m, eval_accuracy=\u001b[39m\u001b[38;5;132;01m{}\u001b[39;00m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;241m.\u001b[39mformat(\n\u001b[1;32m     25\u001b[0m       temp, margin, accuracies[\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m]))\n",
            "\u001b[0;31mTypeError\u001b[0m: train_and_eval() got an unexpected keyword argument 'margin'"
          ]
        }
      ],
      "source": [
        "# The actual training. This cell takes a long time to run, especially on a CPU.\n",
        "rows = []\n",
        "cols = ['temp', 'margin', 'eval_accuracy']\n",
        "# bits=0\n",
        "# for temp in [1,5,10]:\n",
        "margin=0.1\n",
        "bits=0\n",
        "# for bits in [0]:#[0, 5, 10, 15, 20]:\n",
        "#   for margin in [0.05]: #[0.05,0.1,0.2]:\n",
        "# for temp in [0.05, 0.1, 0.2]:\n",
        "for temp in [0.05, 0.1, 0.2]:\n",
        "  #for bits in [0, 5, 10, 15, 20]:\n",
        "  steps, accuracies = train_and_eval(\n",
        "      batch_size=128,\n",
        "      width_multiplier=1,\n",
        "      extra_channel_bits=bits,\n",
        "      nt_xent_temp=temp, # [1,5,10]\n",
        "      margin=margin,\n",
        "      learning_rate=0.005,\n",
        "      epochs=10,\n",
        "      eval_every_n_steps=1000,\n",
        "      print_output=False)\n",
        "  rows.append([temp, margin, accuracies[-1].numpy()])\n",
        "  print(\"temp={}, margin={}, eval_accuracy={}\".format(\n",
        "      temp, margin, accuracies[-1]))\n",
        "plot_df = pd.DataFrame.from_records(rows, columns=cols)\n",
        "print(plot_df)"
      ]
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "gpuType": "V100",
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "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.10.14"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}
