{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "ce51c260-a9b1-43d6-9bbe-f4478df7451d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_75/3250297529.py:87: UserWarning: Creating a tensor from a list of numpy.ndarrays is extremely slow. Please consider converting the list to a single numpy.ndarray with numpy.array() before converting to a tensor. (Triggered internally at /opt/pytorch/pytorch/torch/csrc/utils/tensor_new.cpp:230.)\n",
      "  D = torch.tensor(D, dtype=torch.float32).cuda()\n"
     ]
    },
    {
     "ename": "RuntimeError",
     "evalue": "No CUDA GPUs are available",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mRuntimeError\u001b[0m                              Traceback (most recent call last)",
      "Cell \u001b[0;32mIn [1], line 142\u001b[0m\n\u001b[1;32m    140\u001b[0m     \u001b[38;5;28mprint\u001b[39m(n)\n\u001b[1;32m    141\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m mu \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(size):\n\u001b[0;32m--> 142\u001b[0m         matrix[n, mu] \u001b[38;5;241m=\u001b[39m \u001b[43mget_test_loss\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mint\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43m(\u001b[49m\u001b[43mn\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m+\u001b[39;49m\u001b[43m \u001b[49m\u001b[38;5;241;43m10\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m/\u001b[39;49m\u001b[43m \u001b[49m\u001b[38;5;241;43m10\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43m \u001b[49m\u001b[43mn_scale\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m36\u001b[39;49m\u001b[43m \u001b[49m\u001b[43m,\u001b[49m\u001b[43m(\u001b[49m\u001b[43mmu\u001b[49m\u001b[38;5;241;43m+\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m)\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    143\u001b[0m np\u001b[38;5;241m.\u001b[39msavetxt(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mmatrix.npy\u001b[39m\u001b[38;5;124m'\u001b[39m, matrix)\n",
      "Cell \u001b[0;32mIn [1], line 87\u001b[0m, in \u001b[0;36mget_test_loss\u001b[0;34m(n, mu, p)\u001b[0m\n\u001b[1;32m     84\u001b[0m     D_Y\u001b[38;5;241m.\u001b[39mappend([\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1.\u001b[39m]) \u001b[38;5;28;01mif\u001b[39;00m np\u001b[38;5;241m.\u001b[39mrandom\u001b[38;5;241m.\u001b[39mrand() \u001b[38;5;241m>\u001b[39m p \u001b[38;5;28;01melse\u001b[39;00m D_Y\u001b[38;5;241m.\u001b[39mappend([\u001b[38;5;241m1.\u001b[39m])\n\u001b[1;32m     85\u001b[0m     D_mu\u001b[38;5;241m.\u001b[39mappend(mu2\u001b[38;5;241m.\u001b[39mreshape(\u001b[38;5;241m1\u001b[39m, d))\n\u001b[0;32m---> 87\u001b[0m D \u001b[38;5;241m=\u001b[39m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtensor\u001b[49m\u001b[43m(\u001b[49m\u001b[43mD\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdtype\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfloat32\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcuda\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     88\u001b[0m D_Y \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mtensor(D_Y)\u001b[38;5;241m.\u001b[39mcuda()\n\u001b[1;32m     90\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[38;5;28mint\u001b[39m(n_ \u001b[38;5;241m/\u001b[39m \u001b[38;5;241m2\u001b[39m)):\n",
      "File \u001b[0;32m/usr/local/lib/python3.8/dist-packages/torch/cuda/__init__.py:227\u001b[0m, in \u001b[0;36m_lazy_init\u001b[0;34m()\u001b[0m\n\u001b[1;32m    223\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mAssertionError\u001b[39;00m(\n\u001b[1;32m    224\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mlibcudart functions unavailable. It looks like you have a broken build?\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[1;32m    225\u001b[0m \u001b[38;5;66;03m# This function throws if there's a driver initialization error, no GPUs\u001b[39;00m\n\u001b[1;32m    226\u001b[0m \u001b[38;5;66;03m# are found or any other error occurs\u001b[39;00m\n\u001b[0;32m--> 227\u001b[0m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_C\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_cuda_init\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    228\u001b[0m \u001b[38;5;66;03m# Some of the queued calls may reentrantly call _lazy_init();\u001b[39;00m\n\u001b[1;32m    229\u001b[0m \u001b[38;5;66;03m# we need to just return without initializing in that case.\u001b[39;00m\n\u001b[1;32m    230\u001b[0m \u001b[38;5;66;03m# However, we must not let any *other* threads in!\u001b[39;00m\n\u001b[1;32m    231\u001b[0m _tls\u001b[38;5;241m.\u001b[39mis_initializing \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n",
      "\u001b[0;31mRuntimeError\u001b[0m: No CUDA GPUs are available"
     ]
    }
   ],
   "source": [
    "import math\n",
    "import numpy\n",
    "import numpy as np\n",
    "import torch\n",
    "from torch import nn, optim\n",
    "\n",
    "M = 2\n",
    "d = 1024\n",
    "n_ = 100\n",
    "dh = 512\n",
    "dv = 512\n",
    "cp = 4\n",
    "class TF(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        self.q = nn.Linear(d, dh, bias=False)\n",
    "        self.k = nn.Linear(d, dh, bias=False)\n",
    "        self.v = nn.Linear(d, dv, bias=False)\n",
    "        self.fc = nn.Linear(dv, 1, bias=False)\n",
    "        self.fc.requires_grad_(False)\n",
    "        self.q.weight.data /= 16\n",
    "        self.k.weight.data /= 16\n",
    "        self.v.weight.data /= 16\n",
    "\n",
    "\n",
    "    def forward(self, x):\n",
    "        q = self.q(x)\n",
    "        k = self.k(x)\n",
    "        v = self.v(x)\n",
    "        qk = torch.matmul(q, k.transpose(1, 2))\n",
    "        attn = qk.softmax(dim=2)\n",
    "        attn = torch.sum(attn, dim=1).unsqueeze(1)\n",
    "        attn /= 16\n",
    "        z = torch.matmul(attn, v).squeeze(1)\n",
    "        return self.fc(z)\n",
    "\n",
    "def make_mu1(mu):\n",
    "    mu1 = numpy.zeros(d)\n",
    "    mu1[0] = mu\n",
    "    return mu1\n",
    "\n",
    "def make_mu2(mu):\n",
    "    mu2 = numpy.zeros(d)\n",
    "    mu2[1] = mu\n",
    "    return mu2\n",
    "\n",
    "def make_noise(signal,strength):\n",
    "      \n",
    "    noise =  numpy.random.normal(0, strength, size=d)\n",
    "    \n",
    "    norm = np.linalg.norm(noise)\n",
    "    \n",
    "    return  noise * (36 / norm)\n",
    "\n",
    "def get_test_loss(n, mu,p):\n",
    "    D = []\n",
    "    D_Y = []\n",
    "    D_ = []\n",
    "    D_Y_ = []\n",
    "\n",
    "    mu1 = make_mu1(mu)\n",
    "    mu2 = make_mu2(mu)\n",
    "\n",
    "    D_mu = []\n",
    "    D_mu_ = []\n",
    "\n",
    "    for i in range(int(n / 2)):\n",
    "        X = mu1.copy().reshape(1, d)\n",
    "        # X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        \n",
    "        # for j in range(M - 2):\n",
    "        #     X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        \n",
    "        D.append(X)\n",
    "        D_Y.append([1.]) if np.random.rand() >p else D_Y.append([-1.])\n",
    "        D_mu.append(mu1.reshape(1, d))\n",
    "        \n",
    "        X = mu2.copy().reshape(1, d)\n",
    "        # X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        # for j in range(M - 2):\n",
    "        #     X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        D.append(X)\n",
    "        #标签反转\n",
    "        D_Y.append([-1.]) if np.random.rand() > p else D_Y.append([1.])\n",
    "        D_mu.append(mu2.reshape(1, d))\n",
    "\n",
    "    D = torch.tensor(D, dtype=torch.float32).cuda()\n",
    "    D_Y = torch.tensor(D_Y).cuda()\n",
    "\n",
    "    for i in range(int(n_ / 2)):\n",
    "        X = mu1.copy().reshape(1, d)\n",
    "        # X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        \n",
    "        # for j in range(M - 2):\n",
    "        #     X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        \n",
    "        D_.append(X)\n",
    "        D_Y_.append([1.])if np.random.rand() > p else  D_Y_.append([-1.])\n",
    "        D_mu_.append(mu1.reshape(1, d))\n",
    "        X = mu2.copy().reshape(1, d)\n",
    "        # X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        \n",
    "        # for j in range(M - 2):\n",
    "        #     X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        \n",
    "        D_.append(X)\n",
    "        D_Y_.append([-1.])if np.random.rand() > p else  D_Y_.append([1.])\n",
    "        D_mu_.append(mu2.reshape(1, d))\n",
    "\n",
    "    D_ = torch.tensor(D_, dtype=torch.float32).cuda()\n",
    "    D_Y_ = torch.tensor(D_Y_).cuda()\n",
    "\n",
    "    model = TF().cuda()\n",
    "\n",
    "    optimizer = optim.SGD(model.parameters(), lr=0.1, momentum=0)\n",
    "    loss_fn = nn.SoftMarginLoss().cuda()\n",
    "    EPOCHS = 1000\n",
    "\n",
    "    for epoch in range(1, EPOCHS + 1):\n",
    "        model.train()\n",
    "        optimizer.zero_grad()\n",
    "        output = model(D)\n",
    "\n",
    "        training_loss = loss_fn(output, D_Y)\n",
    "        training_loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        model.eval()\n",
    "        output = model(D_)\n",
    "        test_loss = loss_fn(output, D_Y_)\n",
    "\n",
    "        if training_loss < 0.01:\n",
    "            return test_loss\n",
    "\n",
    "size = 100\n",
    "n_scale = 2\n",
    "mu_scale = 1\n",
    "matrix = np.zeros((size, size))\n",
    "for n in range(size):\n",
    "    print(n)\n",
    "    for mu in range(size):\n",
    "        matrix[n, mu] = get_test_loss(int((n + 10) / 10) * n_scale, 36 ,(mu+1)*0.005)\n",
    "np.savetxt('matrix.npy', matrix)\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4d69847f-4d25-4c26-bf41-d8edc032feb1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "# 禁用LaTeX渲染\n",
    "matplotlib.rcParams['text.usetex'] = False\n",
    "\n",
    "size = 100\n",
    "matrix = np.loadtxt('matrix.npy')\n",
    "\n",
    "x = []\n",
    "y = []\n",
    "# 生成拟合的曲线\n",
    "for i in range(size):\n",
    "    x.append(i + 1)\n",
    "    y.append(204800 / (i + 1)**2 - 20)\n",
    "plt.plot(x, y, color='red', linewidth=5, label=r'$N \\cdot \\mathrm{SNR}^2 = 1000$')\n",
    "\n",
    "\n",
    "#将矩阵中对应位置上的值转换为颜色\n",
    "# cmap 参数指定颜色映射（colormap），用于将矩阵中的数值映射为颜色。\n",
    "# 'viridis' 是一种常用的颜色映射，从紫色到黄色渐变\n",
    "# 'nearest' 表示使用最近邻插值，即每个像素直接使用矩阵中对应位置的值\n",
    "# origin 参数指定矩阵的起始位置'lower' 表示矩阵的第一行对应图像的下方，最后一行对应图像的上方。\n",
    "plt.imshow(matrix, cmap='viridis', interpolation='nearest', origin='lower')\n",
    "#添加一个颜色条，由最大值和最小值决定，出现白色的点说明出现了异常值\n",
    "plt.colorbar()\n",
    "\n",
    "\n",
    "plt.xlabel('Signal-to-Noise Ratio SNR', size=14)\n",
    "plt.ylabel('Sample size N', size=14)\n",
    "\n",
    "x_label = np.array((0.16, 5, 10, 15))\n",
    "y_label = np.arange(2, 22, 2)\n",
    "x_ticks = np.array((0, 31, 63, 95))\n",
    "y_ticks = np.arange(0, size, 10)\n",
    "\n",
    "plt.xticks(x_ticks, x_label)\n",
    "plt.yticks(y_ticks, y_label)\n",
    "plt.legend(loc='best')\n",
    "plt.savefig('heatmap.png', dpi=1200, bbox_inches='tight')\n",
    "#随着映射范围的变化而发生改变\n",
    "#M=2,标签反转0.1\n",
    "# matrix[matrix < 0.5] = 0\n",
    "# matrix[matrix >= 0.5] = 1\n",
    "#M=2,标签反转0.01\n",
    "# matrix[matrix < 0.25] = 0\n",
    "# matrix[matrix >= 0.25] = 1\n",
    "#M=2,标签反转0.001\n",
    "# matrix[matrix < 0.1] = 0\n",
    "# matrix[matrix >= 0.1] = 1\n",
    "# plt.cla()\n",
    "# plt.clf()\n",
    "\n",
    "# plt.figure(1)\n",
    "# plt.plot(x, y, color='red', linewidth=5, label=r'$N \\cdot \\mathrm{SNR}^2 = 1000$')\n",
    "# plt.imshow(matrix, cmap='viridis', interpolation='nearest', origin='lower')\n",
    "# plt.colorbar()\n",
    "# plt.xlabel('Signal-to-Noise Ratio SNR', size=14)\n",
    "# plt.ylabel('Sample size N', size=14)\n",
    "# plt.xticks(x_ticks, x_label)\n",
    "# plt.yticks(y_ticks, y_label)\n",
    "# plt.legend(loc='best')\n",
    "# plt.savefig('heatmap_cutoff.png', dpi=1200, bbox_inches='tight')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "f9e6ce3c-8110-4975-8650-2970cecb3fbd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n",
      "20\n",
      "21\n",
      "22\n",
      "23\n",
      "24\n",
      "25\n",
      "26\n",
      "27\n",
      "28\n",
      "29\n",
      "30\n",
      "31\n",
      "32\n",
      "33\n",
      "34\n",
      "35\n",
      "36\n",
      "37\n",
      "38\n",
      "39\n",
      "40\n",
      "41\n",
      "42\n",
      "43\n",
      "44\n",
      "45\n",
      "46\n",
      "47\n",
      "48\n",
      "49\n",
      "50\n",
      "51\n",
      "52\n",
      "53\n",
      "54\n",
      "55\n",
      "56\n",
      "57\n",
      "58\n",
      "59\n",
      "60\n",
      "61\n",
      "62\n",
      "63\n",
      "64\n",
      "65\n",
      "66\n",
      "67\n",
      "68\n",
      "69\n",
      "70\n",
      "71\n",
      "72\n",
      "73\n",
      "74\n",
      "75\n",
      "76\n",
      "77\n",
      "78\n",
      "79\n",
      "80\n",
      "81\n",
      "82\n",
      "83\n",
      "84\n",
      "85\n",
      "86\n",
      "87\n",
      "88\n",
      "89\n",
      "90\n",
      "91\n",
      "92\n",
      "93\n",
      "94\n",
      "95\n",
      "96\n",
      "97\n",
      "98\n",
      "99\n"
     ]
    }
   ],
   "source": [
    "import math\n",
    "import numpy\n",
    "import numpy as np\n",
    "import torch\n",
    "from torch import nn, optim\n",
    "\n",
    "M = 2\n",
    "d = 1024\n",
    "n_ = 100\n",
    "dh = 512\n",
    "dv = 512\n",
    "cp = 4\n",
    "class TF(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        self.q = nn.Linear(d, dh, bias=False)\n",
    "        self.k = nn.Linear(d, dh, bias=False)\n",
    "        self.v = nn.Linear(d, dv, bias=False)\n",
    "        self.fc = nn.Linear(dv, 1, bias=False)\n",
    "        self.fc.requires_grad_(False)\n",
    "        self.q.weight.data /= 16\n",
    "        self.k.weight.data /= 16\n",
    "        self.v.weight.data /= 16\n",
    "\n",
    "\n",
    "    def forward(self, x):\n",
    "        q = self.q(x)\n",
    "        k = self.k(x)\n",
    "        v = self.v(x)\n",
    "        qk = torch.matmul(q, k.transpose(1, 2))\n",
    "        attn = qk.softmax(dim=2)\n",
    "        attn = torch.sum(attn, dim=1).unsqueeze(1)\n",
    "        attn /= 16\n",
    "        z = torch.matmul(attn, v).squeeze(1)\n",
    "        return self.fc(z)\n",
    "\n",
    "def make_mu1(mu):\n",
    "    mu1 = numpy.zeros(d)\n",
    "    mu1[0] = mu\n",
    "    return mu1\n",
    "\n",
    "def make_mu2(mu):\n",
    "    mu2 = numpy.zeros(d)\n",
    "    mu2[1] = mu\n",
    "    return mu2\n",
    "\n",
    "def make_noise(strength):\n",
    "    return numpy.random.normal(0, strength, size=d)\n",
    "\n",
    "def get_test_loss(n, mu,M):\n",
    "    D = []\n",
    "    D_Y = []\n",
    "    D_ = []\n",
    "    D_Y_ = []\n",
    "\n",
    "    mu1 = make_mu1(mu)\n",
    "    mu2 = make_mu2(mu)\n",
    "\n",
    "    D_mu = []\n",
    "    D_mu_ = []\n",
    "\n",
    "    for i in range(int(n / 2)):\n",
    "        X = mu1.copy().reshape(1, d)\n",
    "        X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        \n",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        \n",
    "        D.append(X)\n",
    "        D_Y.append([1.]) \n",
    "        # D_Y.append([1.]) if np.random.rand() >p else D_Y.append([-1.])\n",
    "        D_mu.append(mu1.reshape(1, d))\n",
    "        \n",
    "        X = mu2.copy().reshape(1, d)\n",
    "        X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        D.append(X)\n",
    "        #标签反转\n",
    "        D_Y.append([-1.])\n",
    "        # D_Y.append([-1.]) if np.random.rand() > p else D_Y.append([1.])\n",
    "        D_mu.append(mu2.reshape(1, d))\n",
    "\n",
    "    D = torch.tensor(D, dtype=torch.float32).cuda()\n",
    "    D_Y = torch.tensor(D_Y).cuda()\n",
    "\n",
    "    for i in range(int(n_ / 2)):\n",
    "        X = mu1.copy().reshape(1, d)\n",
    "        X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        \n",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        \n",
    "        D_.append(X)\n",
    "        D_Y_.append([1.])\n",
    "        # D_Y_.append([1.])if np.random.rand() > p else  D_Y_.append([-1.])\n",
    "        D_mu_.append(mu1.reshape(1, d))\n",
    "        X = mu2.copy().reshape(1, d)\n",
    "        X = numpy.concatenate((X, (make_noise(cp)).reshape(1, d)), 0)\n",
    "        \n",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        \n",
    "        D_.append(X)\n",
    "        D_Y_.append([-1.])\n",
    "        # D_Y_.append([-1.])if np.random.rand() > p else  D_Y_.append([1.])\n",
    "        D_mu_.append(mu2.reshape(1, d))\n",
    "\n",
    "    D_ = torch.tensor(D_, dtype=torch.float32).cuda()\n",
    "    D_Y_ = torch.tensor(D_Y_).cuda()\n",
    "\n",
    "    model = TF().cuda()\n",
    "\n",
    "    optimizer = optim.SGD(model.parameters(), lr=0.1, momentum=0)\n",
    "    loss_fn = nn.SoftMarginLoss().cuda()\n",
    "    EPOCHS = 1000\n",
    "\n",
    "    for epoch in range(1, EPOCHS + 1):\n",
    "        model.train()\n",
    "        optimizer.zero_grad()\n",
    "        output = model(D)\n",
    "\n",
    "        training_loss = loss_fn(output, D_Y)\n",
    "        training_loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        model.eval()\n",
    "        output = model(D_)\n",
    "        test_loss = loss_fn(output, D_Y_)\n",
    "\n",
    "        if training_loss < 0.01:\n",
    "            return test_loss\n",
    "\n",
    "size = 100\n",
    "n_scale = 2\n",
    "mu_scale = 1\n",
    "matrix = np.zeros((size, size))\n",
    "for n in range(size):\n",
    "    print(n)\n",
    "    for mu in range(2,4):\n",
    "        matrix[n, mu] = get_test_loss(int((n + 10) / 10) * n_scale, 16 , mu)\n",
    "np.savetxt('matrix.npy', matrix)\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "67d012ef-2ce5-4d14-9931-af70109ba27c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "\n",
    "# matplotlib.rcParams['text.usetex'] = False\n",
    "# 读取npy文件\n",
    "matrix = np.loadtxt('matrix.npy')\n",
    "\n",
    "# 计算矩阵每列的平均值\n",
    "column_means = np.mean(matrix, axis=0)\n",
    "\n",
    "# 绘制折线图\n",
    "plt.plot(column_means, marker='o')  # 使用圆圈标记每个数据点\n",
    "plt.title('Column Means of Matrix')  # 图表标题\n",
    "plt.xlabel('Column Index')  # x轴标签\n",
    "plt.ylabel('Mean Value')  # y轴标签\n",
    "plt.grid(True)  # 显示网格\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "34af4dd7-c47a-434d-bdd3-68194e59914a",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
