{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "e7ae3742-88fe-4e60-bb2b-ddb5a1db23b8",
   "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",
      "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",
      "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"
     ]
    }
   ],
   "source": [
    "import math\n",
    "import numpy\n",
    "import numpy as np\n",
    "import torch\n",
    "from torch import nn, optim\n",
    "\n",
    "M = 16\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 /= M\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",
    "\n",
    "def Get_test_loss(n, mu):\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",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        D.append(X)\n",
    "        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",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        D.append(X)\n",
    "        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",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        D_.append(X)\n",
    "        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",
    "        for j in range(M - 2):\n",
    "            X = numpy.concatenate((X, (make_noise(0.2)).reshape(1, d)), 0)\n",
    "        D_.append(X)\n",
    "        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",
    "    test_losses = []\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",
    "        # return test_loss\n",
    "        if epoch%10 == 0:\n",
    "            test_losses.append(test_loss.item())\n",
    "\n",
    "    return test_losses  \n",
    "        \n",
    "        \n",
    "size = 20\n",
    "n_scale = 2\n",
    "mu_scale = 1\n",
    "# matrix = np.zeros((size, 100))\n",
    "# for n in range(size):\n",
    "#     print(n)\n",
    "#     for mu in range(100):\n",
    "#         matrix[n, mu] = get_test_loss(100, (mu + 1) * mu_scale)\n",
    "# np.savetxt('n100.npy', matrix)\n",
    "\n",
    "matrix1 = np.zeros((20, 100))\n",
    "for n in range(20):\n",
    "    print(n)\n",
    "    test_losses = Get_test_loss(100, 16)\n",
    "    matrix1[n, :] = test_losses\n",
    "np.savetxt('16.npy', matrix1)\n",
    "                                      \n",
    "matrix2 = np.zeros((20, 100))\n",
    "for n in range(20):\n",
    "    print(n)\n",
    "    test_losses = Get_test_loss(100, 36)\n",
    "    matrix2[n, :] = test_losses\n",
    "np.savetxt('36.npy', matrix2)\n",
    "\n",
    "matrix3 = np.zeros((20, 100))\n",
    "for n in range(20):\n",
    "    print(n)\n",
    "    test_losses = Get_test_loss(100, 64)\n",
    "    matrix3[n, :] = test_losses\n",
    "np.savetxt('64.npy', matrix3)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "4a276e87-c21b-4bdd-8b3b-9fe052a70319",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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voxr166hR48e+mlarRYAYGNjg/Pnz6N+/foGy6jVaqjVahNUT0RERFWRrEdu7OzsEBwcjPj4eP00rVaL+Ph4hIaGFmnfqFEjnDp1ComJifqhZ8+eeOmll5CYmMhTTkRERCTvkRsAiIqKwtChQ9GmTRu0bdsWixYtQlZWFoYPHw4AGDJkCHx9fRETEwN7e3s0a9bMYHl3d3cAKDKdiIiIqifZw01kZCTu3LmDqVOnIjk5GUFBQdi5c6e+k/GNGzegVFpU1yAiIiKSkUIIIeQuwpzS09Ph5uaGtLQ0uLq6yl0OERERlUF5fr95SISIiIisCsMNERERWRWGGyIiIrIqDDdERERkVRhuiIiIyKow3BAREZFVYbghIiIiq8JwQ0RERFaF4YaIiIisCsMNERERWRWGGyIiIrIqDDdERERkVRhuiIiIyKow3BAREZFVYbghIiIiq8JwQ0RERFaF4YaIiIisCsMNERERWZUKhZs1a9bg559/1r9///334e7ujnbt2uH69etGK46IiIiovCoUbmbPng0HBwcAQEJCApYsWYJ58+bBw8MD77zzjlELJCIiIioPm4oslJSUhAYNGgAANm/ejNdeew1vvvkm2rdvjxdffNGY9RERERGVS4WO3Dg7O+PevXsAgN27d6NTp04AAHt7ezx8+NB41RERERGVU4WO3HTq1AkjR45Eq1atcOHCBXTt2hUAcObMGQQEBBizPiIiIqJyqdCRmyVLliA0NBR37tzBjz/+iFq1agEAjh07hv79+xu1QCIiIqLyUAghhNxFmFN6ejrc3NyQlpYGV1dXucshIiKiMijP73eFjtzs3LkTv/32m/79kiVLEBQUhAEDBuDBgwcVWSURERGRUVQo3EycOBHp6ekAgFOnTuHdd99F165dcfXqVURFRRm1QCIiIqLyqFCH4qtXr6JJkyYAgB9//BHdu3fH7Nmzcfz4cX3nYiIiIiI5VOjIjZ2dHbKzswEAe/fuRefOnQEANWvW1B/RISIiIpJDhY7cPP/884iKikL79u1x5MgRxMXFAQAuXLiAOnXqGLVAIiIiovKo0JGbxYsXw8bGBhs2bMCyZcvg6+sLANixYwdeeeUVoxZIREREVB68FJyIiIiqvPL8flfotBQAaDQabN68GefOnQMANG3aFD179oRKparoKomIiIgqrULh5tKlS+jatStu3ryJ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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# 加载npy文件中的矩阵\n",
    "matrix1 = np.loadtxt('16.npy')\n",
    "matrix2 = np.loadtxt('36.npy')\n",
    "matrix3 = np.loadtxt('64.npy')\n",
    "\n",
    "\n",
    "# 计算每个矩阵列方向的平均值\n",
    "mean_matrix1 = np.mean(matrix1, axis=0)\n",
    "mean_matrix2 = np.mean(matrix2, axis=0)\n",
    "mean_matrix3 = np.mean(matrix3, axis=0)\n",
    "\n",
    "\n",
    "# 假设要均匀抽出10条数据\n",
    "num_samples = 10\n",
    "sample_indices = np.linspace(0, len(mean_matrix1) - 1, num_samples, dtype=int)\n",
    "\n",
    "# 抽取数据\n",
    "sampled_matrix1 = mean_matrix1[sample_indices]\n",
    "sampled_matrix2 = mean_matrix2[sample_indices]\n",
    "sampled_matrix3 = mean_matrix3[sample_indices]\n",
    "\n",
    "\n",
    "# 生成x轴数据\n",
    "x = np.arange(num_samples)\n",
    "\n",
    "# 绘制折线图\n",
    "plt.plot(x, sampled_matrix1, label='μ=16', color='blue')\n",
    "plt.plot(x, sampled_matrix2, label='μ=36', color='red')\n",
    "plt.plot(x, sampled_matrix3, label='μ=64', color='green')\n",
    "\n",
    "\n",
    "# 添加图例\n",
    "plt.legend()\n",
    "\n",
    "# 添加标题和坐标轴标签\n",
    "\n",
    "plt.title('Column Mean Values of Matrices')\n",
    "plt.xlabel('μ')\n",
    "plt.ylabel('loss')\n",
    "plt.xticks(x,[100,200,300,400,500,600,700,800,900,1000])\n",
    "# 显示折线图\n",
    "plt.show()\n",
    "\n",
    "plt.savefig('不同信号强度对损失的影响.png', dpi=1200, bbox_inches='tight')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5b09138d-6497-40fe-97f8-6a1db2ff95d8",
   "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
}
