{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "8335a06b-0cef-42fb-ba19-6c735ffcbec2",
   "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"
     ]
    }
   ],
   "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",
    "        \n",
    "        # self.v.weight.data = torch.randn(dv, d)*math.exp(0.5)\n",
    "        self.v.weight.data = torch.randn(dv, d)*16\n",
    "        self.k.weight.data = torch.randn(dv, d)*16\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",
    "    noise =  numpy.random.normal(0, strength, size=d)    \n",
    "    norm = np.linalg.norm(noise)\n",
    "    return  noise * (36 / norm)\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",
    "size = 20\n",
    "matrix = np.zeros((size, 100))\n",
    "for n in range(size):\n",
    "    print(n)\n",
    "    test_losses = get_test_loss(100, 36)\n",
    "    matrix[n, :] = test_losses\n",
    " \n",
    "np.savetxt('KV_16', matrix)\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "39b32dce-2974-4a6d-abec-5d13a9a2c596",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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cQJCQz1XqIiIiIfUOdxMnDixXgeePXs2V1xxBeHh4S7bzZkzh2nTpvH6668zdOhQXnzxRUaPHs2OHTuIj4+v9X379u3j7rvvbvZXjM5JOocAvwAOFhykfb8MoF3tV246d4bERMjMNAPO+ed7sFIRERHPaLAxN3/60584fPjwads9//zz3HrrrUyaNImePXvy+uuvExYWxptvvlnre6xWK+PHj+fRRx+lY8eOLo9fWlpKfn6+09aUhAeFM6C1OclNeWtzYcytW6G4uIbGGncjIiLNQIOFG8MwTtumrKyM9evXO81u7Ofnx6hRo1jl4o6exx57jPj4eG6++ebTfsbMmTOJiopybMnJyXU7AR9i75raXpRGQgJYreZdUzWyX61ZutQDlYmIiHieV++WysnJwWq1kpCQ4LQ/ISGBrKysGt+zYsUK/vOf/zBr1qw6fcb06dPJy8tzbAcOHDjruhsb+zpTK+uziOYPP0BpacMXJyIi4mE+dSt4QUEBf/jDH5g1axaxsbF1ek9wcDCRkZFOW1Njv3KzJXsLfQbnAS7CTbdukJBg3jO+Zo2HKhQREfEcr4ab2NhY/P39q43NOXz4MK1bt67Wfvfu3ezbt4/LL7+cgIAAAgICePfdd5k/fz4BAQHNdvmHxBaJdGzZEZthI6zrD4CLcGOxwHnnmY817kZERJogr4aboKAgBg0axJIlSxz7bDYbS5YsITU1tVr77t27s2XLFtLT0x3bFVdcwa9//WvS09Ob5HiaurJfvcmNNAcV79wJubm1NNagYhERacLqfCt4fbVr147AwMDTtps2bRoTJ05k8ODBnHPOObz44osUFRUxadIkACZMmEBSUhIzZ84kJCSE3r17O70/OjoaoNr+5mZ48nDe2/weG4+m0b497NsH69fDBRfU0NgeblauhLIyCAryYKUiIiINq95Xbjp27MjRo0er7c/NzXW6LXvr1q11upIybtw4nn32WR566CH69+9Peno6CxcudAwyzsjIIDMzs75lNjv2QcWrf1nNoCGnWUSzZ0+IjTXvF1+3zkMVioiIeEa9r9zs27cPq9VabX9paSkHDx48oyKmTJnClClTanxt6WluWX777bfP6DObmp5xPYkOiSa3JJekQZtg7qDaw42fnznuZt48s2tq2DCP1ioiItKQ6hxu5s+f73i8aNEioqKiHM+tVitLliyh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Kio4duyYo40nTZkyhS+++ILvvvuOtm3bOva3bt2asrIycnNzndpXPZeaztX+mqcEBQXRuXNnBg0axMyZM+nXrx8vvfSST53D+vXryc7OZuDAgQQEBBAQEMCyZcv4xz/+QUBAAAkJCT5zLpVFR0fTtWtXdu3a5VPfR2JiIj179nTa16NHD0cXm6/9W9+/fz/ffPMNt9xyi2OfL30f99xzD/fffz+/+93v6NOnD3/4wx/461//ysyZM51q8YXvo1OnTixbtozCwkIOHDjAmjVrKC8vp2PHjj51HpW5q+6G/O9N4eYsdOjQgdatW7NkyRLHvvz8fFavXk1qaioAqamp5ObmOqX0b7/9FpvNxtChQz1Wq2EYTJkyhU8//ZRvv/222qXAQYMGERgY6HQuO3bsICMjw+lctmzZ4vQf7OLFi4mMjKz2h8GTbDYbpaWlPnUOF1xwAVu2bCE9Pd2xDR48mPHjxzse+8q5VFZYWMju3btJTEz0qe9j+PDh1aZG2LlzJ+3atQN86986wFtvvUV8fDyXXnqpY58vfR/FxcX4+Tn/efL398dmswG+930AhIeHk5iYyPHjx1m0aBFXXnmlT54HuO/3n5qayvLlyykvL3e0Wbx4Md26dTurLilAt4KfTkFBgbFx40Zj48aNBmA8//zzxsaNG439+/cbhmHeDhcdHW18/vnnxubNm40rr7yyxtvhBgwYYKxevdpYsWKF0aVLF4/fjvjnP//ZiIqKMpYuXep0q2hxcbGjzW233WakpKQY3377rbFu3TojNTXVSE1Ndbxuv030oosuMtLT042FCxcacXFxHr1N9P777zeWLVtm7N2719i8ebNx//33GxaLxfj666995hxqU/luKcPwjXO56667jKVLlxp79+410tLSjFGjRhmxsbFGdna2z5yDYZi34wcEBBgzZswwfv75Z+ODDz4wwsLCjPfff9/Rxlf+rVutViMlJcW47777qr3mK9/HxIkTjaSkJMet4PPmzTNiY2ONe++919HGV76PhQsXGgsWLDD27NljfP3110a/fv2MoUOHGmVlZY36PDzxty83N9dISEgw/vCHPxhbt241PvzwQyMsLEy3gnvCd999ZwDVtokTJxqGYd4S93//939GQkKCERwcbFxwwQXGjh07nI5x9OhR4/rrrzciIiKMyMhIY9KkSUZBQYFHz6OmcwCMt956y9HmxIkTxl/+8hejZcuWRlhYmHHVVVcZmZmZTsfZt2+fMWbMGCM0NNSIjY017rrrLqO8vNxj53HTTTcZ7dq1M4KCgoy4uDjjggsucAQbXzmH2lQNN75wLuPGjTMSExONoKAgIykpyRg3bpzT3DC+cA52//vf/4zevXsbwcHBRvfu3Y1///vfTq/7yr/1RYsWGUC12gzDd76P/Px8Y+rUqUZKSooREhJidOzY0XjggQecbhn2le9jzpw5RseOHY2goCCjdevWxuTJk43c3NxGfx6e+tu3adMmY8SIEUZwcLCRlJRkPPnkk26p32IYlaZ8FBEREfFxGnMjIiIiTYrCjYiIiDQpCjciIiLSpCjciIiISJOicCMiIiJNisKNiIiINCkKNyIiItKkKNyIiIhIk6JwIyK1at++PS+++GKd2y9duhSLxVJtccbm4pFHHqF///7eLkOk2VO4EWkCLBaLy+2RRx45o+OuXbuWP/7xj3VuP2zYMDIzM4mKijqjz6urqiHq7bffJjo6ukE/syqLxcJnn33mtO/uu+92WkzQG+obSEWaogBvFyAiZy8zM9PxeM6cOTz00ENOq1xHREQ4HhuGgdVqJSDg9P/84+Li6lVHUFAQrVu3rtd7GhOr1YrFYqm2InVdRUREOP2uRcQ7dOVGpAlo3bq1Y4uKisJisTie//TTT7Ro0YIFCxYwaNAggoODWbFiBbt37+bKK68kISGBiIgIhgwZwjfffON03KpXASwWC2+88QZXXXUVYWFhdOnShfnz5zter+2KyqJFi+jRowcRERFcfPHFTmGsoqKCO+64g+joaFq1asV9993HxIkTGTt2bJ3OfenSpUyaNIm8vLxqV6pKS0u5++67SUpKIjw8nKFDh7J06VLHe+31zZ8/n549exIcHExGRgZr167lwgsvJDY2lqioKEaOHMmGDRucfi8AV111FRaLxfG8areUzWbjscceo23btgQHB9O/f38WLlzoeH3fvn1YLBbmzZvHr3/9a8LCwujXrx+rVq1ytNm/fz+XX345LVu2JDw8nF69evHVV1/V+Ls4//zz2b9/P3/9618dvwuR5kjhRqSZuP/++3nyySfZvn07ffv2pbCwkEsuuYQlS5awceNGLr74Yi6//HIyMjJcHufRRx/luuuuY/PmzVxyySWMHz+eY8eO1dq+uLiYZ599lvfee4/ly5eTkZHB3Xff7Xj9qaee4oMPPuCtt94iLS2N/Pz8at09rgwbNowXX3yRyMhIMjMzyczMdBx/ypQprFq1ig8//JDNmzdz7bXXcvHFF/Pzzz871ffUU0/xxhtvsG3bNuLj4ykoKGDixImsWLGCH374gS5dunDJJZdQUFAAmN11AG+99RaZmZmO51W99NJLPPfcczz77LNs3ryZ0aNHc8UVVzh9PsADDzzA3XffTXp6Ol27duX666+noqICgMmTJ1NaWsry5cvZsmULTz31VK1Xh+bNm0fbtm157LHHHL8LkWbJLWuLi0ij8dZbbxlRUVGO5999950BGJ999tlp39urVy/j5Zdfdjxv166d8cILLzieA8aDDz7oeF5YWGgAxoIFC5w+6/jx445aAGPXrl2O97z66qtGQkKC43lCQoLxzDPPOJ5XVFQYKSkpxpVXXllrnTV9TuVzNgzD2L9/v+Hv728cPHjQaf8FF1xgTJ8+3am+9PT02n8phmFYrVajRYsWxv/+9z+n38Wnn37q1O7hhx82+vXr53jepk0bY8aMGU5thgwZYvzlL38xDMMw9u7dawDGG2+84Xh927ZtBmBs377dMAzD6NOnj/HII4+4rK+yqt+ZSHOkKzcizcTgwYOdnhcWFnL33XfTo0cPoqOjiYiIYPv27ae9ctO3b1/H4/DwcCIjI8nOzq61fVhYGJ06dXI8T0xMdLTPy8vj8OHDnHPOOY7X/f39GTRoUL3OrSZbtmzBarXStWtXx1iYiIgIli1bxu7dux3tgoKCnM4J4PDhw9x666106dKFqKgoIiMjKSwsPO3vprL8/HwOHTrE8OHDnfYPHz6c7du3O+2r/PmJiYkAjt/RHXfcwd///neGDx/Oww8/zObNm+tcg0hzpQHFIs1EeHi40/O7776bxYsX8+yzz9K5c2dCQ0O55pprKCsrc3mcwMBAp+cWiwWbzVav9oZh1LP6+issLMTf35/169fj7+/v9Frlbp3Q0NBqY1MmTpzI0aNHeemll2jXrh3BwcGkpqae9ndzpir/juy12H+nt9xyC6NHj+bLL7/k66+/ZubMmTz33HPcfvvtDVKLSFOgKzcizVRaWho33ngjV111FX369KF169bs27fPozVERUWRkJDgNGbFarU6Dd6ti6CgIKxWq9O+AQMGYLVayc7OpnPnzk7b6e7oSktL44477uCSSy6hV69eBAcHk5OT49QmMDCw2mdWFhkZSZs2bUhLS6t27J49e9br/JKTk7ntttuYN28ed911F7Nmzaq1bU2/C5HmRuFGpJnq0qUL8+bNIz09nU2bNnHDDTe4vALTUG6//XZmzpzJ559/zo4dO5g6dSrHjx+v150+7du3p7CwkCVLlpCTk0NxcTFdu3Zl/PjxTJgwgXnz5rF3717WrFnDzJkz+fLLL10er0uXLrz33nts376d1atXM378eEJDQ6t95pIlS8jKyuL48eM1Hueee+7hqaeeYs6cOezYsYP777+f9PR0pk6dWudzu/POO1m0aBF79+5lw4YNfPfdd/To0cPl72L58uUcPHiwWiATaS4UbkSaqeeff56WLVsybNgwLr/8ckaPHs3AgQM9Xsd9993H9ddfz4QJE0hNTSUiIoLRo0cTEhJS52MMGzaM2267jXHjxhEXF8fTTz8NmHczTZgwgbvuuotu3boxduxY1q5dS0pKisvj/ec//+H48eMMHDiQP/zhD9xxxx3Ex8c7tXnuuedYvHgxycnJDBgwoMbj3HHHHUybNo277rqLPn36sHDhQubPn0+XLl3qfG5Wq5XJkyfTo0cPLr74Yrp27co///nPWts/9thj7Nu3j06dOtV7niKRpsJieKLzW0Skjmw2Gz169OC6667j8ccf93Y5IuKDNKBYRLxq//79fP3114wcOZLS0lJeeeUV9u7dyw033ODt0kTER6lbSkS8ys/Pj7fffpshQ4YwfPhwtmzZwjfffONyXImIiCvqlhIREZEmRVduREREpElRuBEREZEmReFGREREmhSFGxEREWlSFG5ERESkSVG4ERERkSZF4UZERESaFIUbERERaVL+HxSh2iDH2w7qAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# 加载npy文件中的矩阵\n",
    "matrix1 = np.loadtxt('KV_8')\n",
    "matrix2 = np.loadtxt('KV_16')\n",
    "matrix3 = np.loadtxt('KV_4')\n",
    "# matrix4 = np.loadtxt('8')  # 假设文件名为 '矩阵4'\n",
    "# matrix5 = np.loadtxt('0.1')  # 假设文件名为 '矩阵5'\n",
    "# matrix6 = np.loadtxt('0.01')\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",
    "# mean_matrix4 = np.mean(matrix4, axis=0)\n",
    "# mean_matrix5 = np.mean(matrix5, axis=0)\n",
    "# mean_matrix6 = np.mean(matrix6, 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",
    "# sampled_matrix4 = mean_matrix4[sample_indices]\n",
    "# sampled_matrix5 = mean_matrix5[sample_indices]\n",
    "# sampled_matrix6 = mean_matrix6[sample_indices]\n",
    "\n",
    "\n",
    "# 生成x轴数据\n",
    "x = np.arange(num_samples)\n",
    "\n",
    "# 绘制折线图\n",
    "plt.plot(x, sampled_matrix1, label='σ=1', color='blue')\n",
    "plt.plot(x, sampled_matrix2, label='σ=2', color='red')\n",
    "plt.plot(x, sampled_matrix3, label='σ=4', color='green')\n",
    "# plt.plot(x, sampled_matrix4, label='σ=8', color='purple')  # 添加矩阵4的折线\n",
    "# plt.plot(x, sampled_matrix5, label='0.1', color='orange')  # 添加矩阵5的折线\n",
    "# plt.plot(x, sampled_matrix6, label='0.01', color='brown') \n",
    "\n",
    "\n",
    "# 添加图例\n",
    "plt.legend()\n",
    "\n",
    "# 添加标题和坐标轴标签\n",
    "\n",
    "\n",
    "plt.xlabel('Training Iterations t')\n",
    "plt.ylabel('Test_Loss')\n",
    "plt.xticks(x,[100,200,300,400,500,600,700,800,900,1000])\n",
    "# 显示折线图\n",
    "\n",
    "plt.savefig('V.png', dpi=300, bbox_inches='tight')\n",
    "plt.show()\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c54cda39-1e30-4d3c-8e3c-a677261f6d7a",
   "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
}
