{"cells":[{"cell_type":"markdown","id":"idlhUjsSJxql","metadata":{"id":"idlhUjsSJxql"},"source":["# Mount with Google Drive"]},{"cell_type":"code","execution_count":null,"id":"9xiwqzGPuvGx","metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"9xiwqzGPuvGx","outputId":"5fa90a08-da57-4f3f-cfde-783fbc506b66"},"outputs":[{"name":"stdout","output_type":"stream","text":["\n","\n"]}],"source":["%reset"]},{"cell_type":"code","execution_count":null,"id":"skJrn1mzghZU","metadata":{"id":"skJrn1mzghZU","colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"status":"ok","timestamp":1734362671637,"user_tz":300,"elapsed":681,"user":{"displayName":"Jianlei Huang","userId":"10108074868984712667"}},"outputId":"daa9b992-9a48-460e-d7f2-77d2466e6461"},"outputs":[{"output_type":"stream","name":"stdout","text":["Drive already mounted at /content/drive; to attempt to forcibly remount, call drive.mount(\"/content/drive\", force_remount=True).\n"]}],"source":["from google.colab import drive\n","drive.mount('/content/drive')"]},{"cell_type":"markdown","source":["# Main part"],"metadata":{"id":"boXlHiPKIv9e"},"id":"boXlHiPKIv9e"},{"cell_type":"code","execution_count":null,"id":"34ac382d-c496-43de-97e2-a77b399039db","metadata":{"id":"34ac382d-c496-43de-97e2-a77b399039db","colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"status":"ok","timestamp":1734362671638,"user_tz":300,"elapsed":3,"user":{"displayName":"Jianlei Huang","userId":"10108074868984712667"}},"outputId":"7a78a28c-2b4f-4525-ca80-9015bdf025f0"},"outputs":[{"output_type":"stream","name":"stdout","text":["Using cuda device\n"]}],"source":["## Imports\n","import torch\n","import torch.nn as nn\n","import torch.nn.functional as F\n","import torch.autograd as autograd\n","import matplotlib.pyplot as plt\n","import seaborn as sn\n","import numpy as np\n","import pandas as pd\n","import math\n","import scipy\n","import time as t\n","\n","# Get cpu or gpu device for training.\n","device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n","print(f\"Using {device} device\")\n","# torch.set_default_dtype(torch.float32)\n","torch.set_default_dtype(torch.float64)\n","#torch.manual_seed(13)\n","\n","\n","\n","# Sampling parameters etc\n","#n_axis = 51\n","n_axis = 41\n","#n_time = 131\n","n_time = 81\n","\n","axis = torch.linspace(0,4,n_axis, device=device)\n","time = torch.linspace(0,4,n_time, device=device)\n","Ps = torch.cartesian_prod(axis,axis,time)\n","# Number of points\n","lP = Ps.shape[0]\n","\n","\n","# Initial dataset\n","data_axis = torch.linspace(0,8, 5, device=device)\n","data_time = torch.linspace(0,0,1, device=device)\n","data_Ps = torch.cartesian_prod(data_axis,data_axis,time)\n","#data_Ps = torch.cartesian_prod(axis,axis,time)\n","\n","mask = data_Ps[:,2] == 0.\n","X = data_Ps[mask]\n","dtX = data_Ps[mask]\n","\n","#mask1 = abs(data_Ps[:,0]) == 2.\n","#dyX = data_Ps[mask1]\n","\n","#mask2 = abs(data_Ps[:,1]) == 2.\n","#dxX = data_Ps[mask2]\n","\n","#Y = torch.exp(-(X[:,0]-2)**2*5)+torch.exp(-(X[:,1]+X[:,0]-4)**2*5)\n","Y = torch.exp(-((X[:,0]-2)**2+(X[:,1]-2)**2)*10)\n","#Y = torch.exp(-(4*(X[:,0]-2.5)**2+(X[:,1]-3)**2)*10)\n","#Y = torch.cos((X[:,0]-1)*5)+torch.cos((X[:,1]-1)*5)\n","Y = Y.view(-1,1)\n","\n","#dtY = torch.where( ((dtX[:,0]-1).abs() < 1e-1) & (dtX[:,1].abs() < 1), 0., 0. )\n","#dtY = -2*5*(X[:,0]-2)*torch.exp(-(X[:,0]-2)**2*5)-2*5*(X[:,1]+X[:,0]-4)*torch.exp(-(X[:,1]+X[:,0]-4)**2*5)\n","#dtY = -5*(torch.sin((X[:,0]-1)*5)+torch.sin((X[:,1]-1)*5))\n","#dtY = dtY.view(-1,1)\n","\n","#dxY = torch.where( ((dxX[:,0]-1).abs() < 1e-1) & (dxX[:,1].abs() < 1), 0., 0. )\n","#dxY = dxY.view(-1,1)\n","\n","#dyY = torch.where( ((dyX[:,0]-1).abs() < 1e-1) & (dyX[:,1].abs() < 1), 0., 0. )\n","#dyY = dyY.view(-1,1)\n","\n","#dtX = dtX.to(torch.complex128)\n","#dxX = dxX.to(torch.complex128)\n","#dyX = dyX.to(torch.complex128)\n","#dtY = dtY.to(torch.complex128)\n","#dxY = dxY.to(torch.complex128)\n","#dyY = dyY.to(torch.complex128)\n","X = X.to(torch.complex128)\n","Y = Y.to(torch.complex128)\n","#Y = torch.cat((Y,dtY),0)"]},{"cell_type":"code","source":["Y"],"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"id":"-cvHnHvViSex","executionInfo":{"status":"ok","timestamp":1734362671638,"user_tz":300,"elapsed":3,"user":{"displayName":"Jianlei Huang","userId":"10108074868984712667"}},"outputId":"b37b061a-582a-44f1-b371-672983bfb71e"},"id":"-cvHnHvViSex","execution_count":null,"outputs":[{"output_type":"execute_result","data":{"text/plain":["tensor([[ 1.8049e-35+0.j],\n","        [ 4.2484e-18+0.j],\n","        [ 1.8049e-35+0.j],\n","        [ 1.3839e-87+0.j],\n","        [1.9152e-174+0.j],\n","        [ 4.2484e-18+0.j],\n","        [ 1.0000e+00+0.j],\n","        [ 4.2484e-18+0.j],\n","        [ 3.2575e-70+0.j],\n","        [4.5080e-157+0.j],\n","        [ 1.8049e-35+0.j],\n","        [ 4.2484e-18+0.j],\n","        [ 1.8049e-35+0.j],\n","        [ 1.3839e-87+0.j],\n","        [1.9152e-174+0.j],\n","        [ 1.3839e-87+0.j],\n","        [ 3.2575e-70+0.j],\n","        [ 1.3839e-87+0.j],\n","        [1.0611e-139+0.j],\n","        [1.4685e-226+0.j],\n","        [1.9152e-174+0.j],\n","        [4.5080e-157+0.j],\n","        [1.9152e-174+0.j],\n","        [1.4685e-226+0.j],\n","        [2.0322e-313+0.j]], device='cuda:0')"]},"metadata":{},"execution_count":13}]},{"cell_type":"code","execution_count":null,"id":"21503eb5-201e-43de-b8e8-78dd2617680b","metadata":{"id":"21503eb5-201e-43de-b8e8-78dd2617680b"},"outputs":[],"source":["def getVarietyPoints(base):\n","    x1,y1 = base.unbind(1)\n","    #x2,y2 = base2.unbind(1)\n","    t1 = torch.sqrt(x1.square() + y1.square())\n","    #t2 = torch.sqrt(x2.square() + y2.square())\n","\n","    return torch.stack([ torch.stack([x1,y1,t1],1), torch.stack([-x1,y1,t1],1), torch.stack([x1,y1,-t1],1),  torch.stack([-x1,y1,-t1],1),torch.stack([x1,-y1,t1],1), torch.stack([-x1,-y1,t1],1), torch.stack([x1,-y1,-t1],1),  torch.stack([-x1,-y1,-t1],1) ])\n","\n","def Phi(base, X):\n","    pts = getVarietyPoints(base)\n","    # return (pts.inner(X) * 1.j).exp().mean(0)\n","    return (pts.inner(X)).exp().mean(0)\n","\n","def dtPhi(base, X):\n","    pts = getVarietyPoints(base)\n","    return ((pts.inner(X)).exp().mul(pts[:,:,2].unsqueeze(2).repeat(1, 1, pts.inner(X).shape[2])).mean(0))\n","\n","def dxPhi(base, X):\n","    pts = getVarietyPoints(base)\n","    return ((pts.inner(X)).exp().mul(pts[:,:,0].unsqueeze(2).repeat(1, 1, pts.inner(X).shape[2])).mean(0))\n","\n","def dyPhi(base, X):\n","    pts = getVarietyPoints(base)\n","    return ((pts.inner(X)).exp().mul(pts[:,:,1].unsqueeze(2).repeat(1, 1, pts.inner(X).shape[2])).mean(0))\n","\n","def randomMask(n_pts=2000):\n","    mask = torch.zeros(X.shape[0]).bool()\n","    mask[torch.randperm(X.shape[0])[:n_pts]] = True\n","    return mask\n","\n","\n","def train(N):\n","    for epoch in range(N):\n","        PhiX = Phi(MC_base1 * 1.j, X)\n","        #dtPhiX = dtPhi(MC_base1 * 1.j, MC_base2 * 1.j, dtX)\n","        #dxPhiX = dxPhi(MC_base1 * 1.j, MC_base2 * 1.j, dxX)\n","        #dyPhiX = dyPhi(MC_base1 * 1.j, MC_base2 * 1.j, dyX)\n","        #PhiX = torch.cat((PhiX,dtPhiX),1)\n","        A = torch.diag_embed((eps - S_diag).exp()) + PhiX @ PhiX.H\n","        LA = torch.linalg.cholesky(A)\n","        alpha = torch.linalg.solve_triangular(LA, PhiX @ Y.to(torch.complex128), upper=False)\n","        print(PhiX.shape)\n","        print(Y.shape)\n","        print(A.shape)\n","\n","\n","        nlml = 1/(2*eps.exp()) * (Y.norm().square() - alpha.norm().square())\n","        nlml += (PhiX.shape[1] - PhiX.shape[0])/2 * eps\n","        nlml += LA.diag().real.log().sum()\n","        nlml += 0.5*S_diag.sum()\n","\n","        opt.zero_grad()\n","        nlml.backward()\n","        opt.step()\n","\n","        with torch.no_grad():\n","            train_pred = PhiX.H @ torch.linalg.solve_triangular(LA.H, alpha, upper=True)\n","            err = (train_pred.real - Y).square().mean().sqrt()\n","            print(26*\"~\" + f'\\nepoch {epoch}\\n\\\n","nlml {nlml}\\n\\\n","err {err}\\n\\\n","eps {eps.exp()}\\n\\\n","base1 std {MC_base1.std(0)}\\n\\\n","min,max {train_pred.real.min().detach(),train_pred.real.max().detach()}')"]},{"cell_type":"code","execution_count":null,"id":"qTn_iWiBboK6","metadata":{"id":"qTn_iWiBboK6"},"outputs":[],"source":["n_MC = 1000\n","# MC_axis = torch.linspace(-1,1, n_MC, device=device) * 30\n","MC_base1 = (torch.randn((n_MC, 2), device=device)).requires_grad_()\n","#MC_base2 = (torch.randn((n_MC, 2), device=device)).requires_grad_()\n","# MC_base = torch.cartesian_prod(MC_axis,MC_axis).requires_grad_()\n","S_diag = torch.full((n_MC,), -np.log(n_MC), requires_grad=True, device=device)\n","# S_diag = torch.full((n_MC**2,), -np.log(n_MC**2), requires_grad=False, device=device)\n","eps = torch.tensor(np.log(1e-6), requires_grad=False, device=device)"]},{"cell_type":"code","execution_count":null,"id":"9b3b7aba-9a69-4e32-a452-04047f352c2c","metadata":{"id":"9b3b7aba-9a69-4e32-a452-04047f352c2c"},"outputs":[],"source":["time_start = t.time()\n","opt = torch.optim.Adam([\n","    {'params': [MC_base1], 'lr': 1e-1},\n","    {'params': eps, 'lr': 1e-2}])\n","#train(100000)\n","#train(10000)\n","opt = torch.optim.Adam([\n","    {'params': [MC_base1], 'lr': 1e-2},\n","    {'params': eps, 'lr': 1e-2}])\n","#train(100000)\n","#train(10000)\n","opt = torch.optim.Adam([\n","    {'params': [MC_base1], 'lr': 1e-2},\n","    {'params': [S_diag, eps], 'lr': 1e-2}])\n","#train(1000)\n","opt = torch.optim.Adam([\n","    {'params': [MC_base1], 'lr': 1e-3},\n","    {'params': [S_diag, eps], 'lr': 1e-3}])\n","#train(300)\n","torch.save({\n","            'MC_base1': MC_base1.cpu(),\n","            'S_diag': S_diag.cpu(),\n","            'eps': eps.cpu(),\n","    }, \"state.pt\")\n","time_end = t.time()"]},{"cell_type":"code","source":["time_end - time_start"],"metadata":{"id":"9_EodRBD6JmE","colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"status":"ok","timestamp":1734362671866,"user_tz":300,"elapsed":2,"user":{"displayName":"Jianlei Huang","userId":"10108074868984712667"}},"outputId":"95874006-c3c5-4b38-de1b-4d9846ef175d"},"id":"9_EodRBD6JmE","execution_count":null,"outputs":[{"output_type":"execute_result","data":{"text/plain":["0.002145051956176758"]},"metadata":{},"execution_count":17}]},{"cell_type":"code","execution_count":null,"id":"afb3c4a4-300f-474d-be4b-54306ca163c4","metadata":{"id":"afb3c4a4-300f-474d-be4b-54306ca163c4","colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"status":"ok","timestamp":1734362680214,"user_tz":300,"elapsed":8350,"user":{"displayName":"Jianlei Huang","userId":"10108074868984712667"}},"outputId":"0befe8a6-2a5b-4e2f-b8ff-e68d656e3076"},"outputs":[{"output_type":"stream","name":"stderr","text":["<ipython-input-18-67f7dd72ae5a>:9: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n","  st = torch.load(\"state.pt\")\n"]}],"source":["#torch.save({\n","#            'MC_base1': MC_base1.cpu(),\n","#            'MC_base2': MC_base2.cpu(),\n","#            'S_diag': S_diag.cpu(),\n","#            'eps': eps.cpu(),\n","#    }, \"state.pt\")\n","\n","\n","st = torch.load(\"state.pt\")\n","MC_base1 = st['MC_base1']\n","S_diag = st['S_diag']\n","eps = st['eps']\n","\n","# Prediction\n","#Phi_ = Phi(MC_base * 1.j, Ps.to(torch.complex128)).to(device)\n","#Phi_ = Phi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#dtPhi_ = dtPhi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#dxPhi_ = dxPhi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#dyPhi_ = dyPhi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#Phi_ = torch.cat((Phi_,dtPhi_,dxPhi_,dyPhi_),1)\n","#Phi_ = torch.cat((Phi_,dtPhi_),1)\n","#Y = torch.cat((Y,dtY),0)\n","PhiX = Phi(MC_base1 * 1.j, X.to(\"cpu\"))\n","#dtPhiX = dtPhi(MC_base1 * 1.j, dtX.to(\"cpu\"))\n","#dxPhiX = dxPhi(MC_base * 1.j, dxX.to(\"cpu\"))\n","#dyPhiX = dyPhi(MC_base * 1.j, dyX.to(\"cpu\"))\n","#PhiX = torch.cat((PhiX,dxPhiX,dyPhiX),1)\n","#PhiX = torch.cat((PhiX,dtPhiX),1)\n","A = torch.diag_embed((eps - S_diag).exp()) + PhiX @ PhiX.H\n","LA = torch.linalg.cholesky(A)\n","alpha = torch.linalg.solve_triangular(LA, PhiX @ Y.to(\"cpu\").to(torch.complex128), upper=False)\n","Phi_ = Phi(MC_base1 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","predwave = Phi_.H @ torch.linalg.solve_triangular(LA.H, alpha.to(\"cpu\"), upper=True)\n","predwave = predwave.real\n","predwave.detach().cpu().numpy().tofile(\"predwave4.dat\")\n","del Phi_, predwave\n","#dtPhi_ = dtPhi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#preddt = dtPhi_.H @ torch.linalg.solve_triangular(LA.H, alpha.to(\"cpu\"), upper=True)\n","#preddt = preddt.real\n","#preddt.detach().cpu().numpy().tofile(\"preddt.dat\")\n","#del dtPhi_, preddt\n","#dxPhi_ = dxPhi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#preddx = dxPhi_.H @ torch.linalg.solve_triangular(LA.H, alpha.to(\"cpu\"), upper=True)\n","#preddx = preddx.real\n","#preddx.detach().cpu().numpy().tofile(\"preddx.dat\")\n","#del dxPhi_, preddx\n","#dyPhi_ = dyPhi(MC_base1 * 1.j, MC_base2 * 1.j, Ps.to(torch.complex128).to(\"cpu\"))\n","#preddy = dyPhi_.H @ torch.linalg.solve_triangular(LA.H, alpha.to(\"cpu\"), upper=True)\n","#preddy = preddy.real\n","#preddy.detach().cpu().numpy().tofile(\"preddy.dat\")\n","#del dyPhi_, preddy\n","\n","axis.cpu().numpy().tofile(\"axis.dat\")\n","time.cpu().numpy().tofile(\"time.dat\")\n"]},{"cell_type":"code","execution_count":null,"id":"-5qUr2RGgTf_","metadata":{"id":"-5qUr2RGgTf_"},"outputs":[],"source":["import locale\n","def getpreferredencoding(do_setlocale = True):\n","    return \"UTF-8\"\n","locale.getpreferredencoding = getpreferredencoding\n","!cp predwave4.dat \"/content/drive/MyDrive/Colab Notebooks\"\n","#!cp preddt.dat \"/content/drive/MyDrive/Colab Notebooks\"\n","#!cp preddx.dat \"/content/drive/MyDrive/Colab Notebooks\"\n","#!cp preddy.dat \"/content/drive/MyDrive/Colab Notebooks\"\n","!cp axis.dat \"/content/drive/MyDrive/Colab Notebooks\"\n","!cp time.dat \"/content/drive/MyDrive/Colab Notebooks\"\n","!cp state.pt \"/content/drive/MyDrive/Colab Notebooks\""]},{"cell_type":"code","execution_count":null,"id":"_nQ83U-3DLSk","metadata":{"id":"_nQ83U-3DLSk","colab":{"base_uri":"https://localhost:8080/","height":452},"executionInfo":{"status":"ok","timestamp":1734362682178,"user_tz":300,"elapsed":1579,"user":{"displayName":"Jianlei Huang","userId":"10108074868984712667"}},"outputId":"719da9dc-aacf-4f83-882c-18a3fdce4511"},"outputs":[{"output_type":"display_data","data":{"text/plain":["<Figure size 640x480 with 1 Axes>"],"image/png":"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r95yyy16+umnNWXKFF1++eU66aSTtG/fPr3yyiv6/e9/r3379hV9zaeffrp+//vf67bbbtPIkSPV2NioKVOm+H3LAJTL8qwdADHJTd8tNeX28OHD3pIlS7zGxkavT58+XkNDg7do0SLvb3/7W+d9XnnlFe/LX/6yN3r0aK+6utobPny498///M/exo0buz2Xekzf9TzP++53v+t95CMf8aqqqrpN5e05fTfnz3/+syfJk+Q9++yzedu8e/dub+HChV5DQ4PXp08f79hjj/U+8YlPePfcc0/J92XLli3etGnTvP79+3uSmMoLxCzjeT3qqwAAADFhjAgAALCGIAIAAKwhiAAAAGsIIgAAwBqCCAAAsIYgAgAArHF6QbOOjg7t2rVLgwYNYhlmAAASwvM8HThwQCNHjjxqEcKenA4iu3btUkNDg+1mAACAEHbs2KFRo0YVvY/TQWTQoEGSsi+ktrbWcmsAAIAfLS0tamho6LyOF+N0EMl1x9TW1hJEAABIGD/DKhisCgAArCGIAAAAawgiAADAGoIIAACwhiACAACsIYgAAABrCCIAAMAagggAALCGIAIAAKwhiAAAAGsIIgAAwBqCCAAAsIYgAgAArCGIAAAAawgiAADAGoIIAACwhiACAACsIYgAAABrYgsit9xyizKZjK6++uq4DgkAABwXSxB56aWXdPfdd+uUU06J43AAACAhjAeRgwcP6qKLLtJPf/pTHXPMMaYPBwAAEsR4EFm4cKE+85nPaObMmSXv29bWppaWlm43AACQXr1NPvmDDz6oV155RS+99JKv+y9btkxLliwx2SQAAOAQYxWRHTt26N/+7d90//33q1+/fr4es2jRIjU3N3feduzYYap5AADAARnP8zwTT/zII49o9uzZ6tWrV+f32tvblclkVFVVpba2tm4/y6elpUV1dXVqbm5WbW2tiWYCAICIBbl+G+ua+cQnPqE//vGP3b538cUXa8KECbr++utLhhAAAJB+xoLIoEGDdPLJJ3f73sCBAzV06NCjvg8AACoTK6sCAABrjM6a6emZZ56J83AAAMBxVEQAAIA1BBEAAGANQQQAAFhDEAEAANYQRAAAgDUEEQAAYE2s03cBJJDnSdvWS3s2S8MnSo3TpEzGdqsApARBBEBx29ZL982WOtqlql7S3NXSmOm2WwUgJeiaAVDcns3ZECJl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Fb1PxvMTV7q44YYb9P3vf7/ofTZv3qwnn3xSDz30kNatW6devXrpnXfeUWNjo1599VWddtppeR/XsyLS0dGhffv2aejQocpkMkGaGbmWlhY1NDRox44dqq2ttdoW03it6VQpr7VSXqfEa02rNLxWz/N04MABjRw5UlVVxUeBBK6IXHvttZo/f37R+4wZM0Z/+MMf9Nxzz6m6urrbzyZNmqSLLrpIK1euPOpx1dXVR91/8ODBQZtoVG1tbWJ/MYLitaZTpbzWSnmdEq81rZL+WktVQnICB5H6+nrV19eXvN9PfvITfe973+v8eteuXZo1a5ZWrVqlKVOmBD0sAABIIWNjREaPHt3t65qaGknS2LFjNWrUKFOHBQAACcLKqj5VV1frxhtvPKrrKI14relUKa+1Ul6nxGtNq0p6rVKIwaoAAABRoSICAACsIYgAAABrCCIAAMAagggAALCGIAIAAKwhiJShra1Np512mjKZjDZt2mS7OUZ89rOf1ejRo9WvXz8dd9xx+spXvqJdu3bZblbk3nnnHV166aVqbGxU//79NXbsWN144406dOiQ7aYZcfPNN2vq1KkaMGCAc6sXl+uOO+7QCSecoH79+mnKlCl68cUXbTcpcuvXr9d5552nkSNHKpPJ6JFHHrHdJCOWLVumM844Q4MGDdLw4cN1/vnna+vWrbabZcSdd96pU045pXM11bPOOkuPPfaY7WbFgiBShm9+85saOXKk7WYYNWPGDD300EPaunWrHn74Yb311lu64IILbDcrclu2bFFHR4fuvvtuvf766/rRj36ku+66S4sXL7bdNCMOHTqkL37xi7ryyittNyVSq1at0jXXXKMbb7xRr7zyik499VTNmjVLe/bssd20SLW2turUU0/VHXfcYbspRq1bt04LFy7U888/r9/97nc6fPiwPvnJT6q1tdV20yI3atQo3XLLLXr55Ze1ceNGffzjH9fnPvc5vf7667abZp6HUB599FFvwoQJ3uuvv+5J8l599VXbTYrF2rVrvUwm4x06dMh2U4z7wQ9+4DU2NtpuhlH33nuvV1dXZ7sZkZk8ebK3cOHCzq/b29u9kSNHesuWLbPYKrMkeWvWrLHdjFjs2bPHk+StW7fOdlNiccwxx3g/+9nPbDfDOCoiIezevVuXX365fvGLX2jAgAG2mxObffv26f7779fUqVPVp08f280xrrm5WUOGDLHdDPh06NAhvfzyy5o5c2bn96qqqjRz5kw999xzFluGqDQ3N0tS6v8u29vb9eCDD6q1tVVnnXWW7eYYRxAJyPM8zZ8/X1dccYUmTZpkuzmxuP766zVw4EANHTpU27dv19q1a203ybg333xTt99+u772ta/Zbgp82rt3r9rb2zVixIhu3x8xYoTee+89S61CVDo6OnT11VfrYx/7mE4++WTbzTHij3/8o2pqalRdXa0rrrhCa9as0UknnWS7WcYRRP7uhhtuUCaTKXrbsmWLbr/9dh04cECLFi2y3eTQ/L7WnOuuu06vvvqqnnzySfXq1Utf/epX5SVkZ4Cgr1WSdu7cqU996lP64he/qMsvv9xSy4ML81qBpFi4cKFee+01Pfjgg7abYsz48eO1adMmvfDCC7ryyis1b948/elPf7LdLOPYa+bvmpqa9Ne//rXofcaMGaMvfelL+s1vfqNMJtP5/fb2dvXq1UsXXXSRVq5cabqpZfP7Wvv27XvU99999101NDRow4YNiSgZBn2tu3bt0vTp03XmmWdqxYoVqqpKTlYP8/91xYoVuvrqq7V//37DrTPv0KFDGjBggH71q1/p/PPP7/z+vHnztH///tRW8jKZjNasWdPtNafNVVddpbVr12r9+vVqbGy03ZzYzJw5U2PHjtXdd99tuylG9bbdAFfU19ervr6+5P1+8pOf6Hvf+17n17t27dKsWbO0atUqTZkyxWQTI+P3tebT0dEhKTt1OQmCvNadO3dqxowZOv3003XvvfcmKoRI5f1/TYO+ffvq9NNP11NPPdV5Ue7o6NBTTz2lq666ym7jEIrnefr617+uNWvW6JlnnqmoECJlf3+Tcq4tB0EkoNGjR3f7uqamRpI0duxYjRo1ykaTjHnhhRf00ksv6eyzz9Yxxxyjt956S9/+9rc1duzYRFRDgti5c6emT5+u448/Xrfeequampo6f3bsscdabJkZ27dv1759+7R9+3a1t7d3roNz4okndv5OJ9E111yjefPmadKkSZo8ebJ+/OMfq7W1VRdffLHtpkXq4MGDevPNNzu/3rZtmzZt2qQhQ4YcdY5KsoULF+qBBx7Q2rVrNWjQoM6xPnV1derfv7/l1kVr0aJFOvfcczV69GgdOHBADzzwgJ555hk98cQTtptmntU5Oymwbdu21E7f/Z//+R9vxowZ3pAhQ7zq6mrvhBNO8K644grv3Xfftd20yN17772epLy3NJo3b17e1/r000/bblrZbr/9dm/06NFe3759vcmTJ3vPP/+87SZF7umnn877/2/evHm2mxapQn+T9957r+2mRe6SSy7xjj/+eK9v375efX2994lPfMJ78sknbTcrFowRAQAA1iSrExwAAKQKQQQAAFhDEAEAANYQRAAAgDUEEQAAYA1BBAAAWEMQAQAA1hBEAACANQQRAABgDUEEAABYQxABAADW/P818Fhk4/FqwgAAAABJRU5ErkJggg==\n"},"metadata":{}}],"source":["plt.ion()\n","f, ax = plt.subplots()\n","sn.kdeplot(x = MC_base1.detach().numpy()[:,0], y = MC_base1.detach().numpy()[:,1], fill=True)\n","sn.scatterplot(x = MC_base1.detach().numpy()[:,0], y = MC_base1.detach().numpy()[:,1], s=10)\n","sn.kdeplot(x = MC_base1.detach().numpy()[:,0], y = MC_base1.detach().numpy()[:,1], bw_adjust=0.5, fill=True)\n","plt.title(\"Positive t\")\n","plt.show()"]}],"metadata":{"accelerator":"GPU","colab":{"gpuType":"A100","machine_shape":"hm","provenance":[],"toc_visible":true},"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.11.7"}},"nbformat":4,"nbformat_minor":5}