{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "bf94d59d",
   "metadata": {
    "id": "bf94d59d"
   },
   "source": [
    "Import Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "09088a1a",
   "metadata": {
    "executionInfo": {
     "elapsed": 13,
     "status": "ok",
     "timestamp": 1688634525819,
     "user": {
      "displayName": "Choco Kiwi",
      "userId": "00673008398261993589"
     },
     "user_tz": -480
    },
    "id": "09088a1a"
   },
   "outputs": [],
   "source": [
    "import jax\n",
    "import flax\n",
    "import optax\n",
    "from jax import lax, random, numpy as jnp\n",
    "from jax import random, grad, vmap, hessian, jacfwd, jit\n",
    "from jax import config\n",
    "from flax import linen as nn\n",
    "from evojax.util import get_params_format_fn\n",
    "from scipy import io\n",
    "import time\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# choose GPU\n",
    "import os\n",
    "os.environ['CUDA_VISIBLE_DEVICES'] = '0'\n",
    "jax.config.update(\"jax_enable_x64\", True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d6562cd7",
   "metadata": {},
   "source": [
    "\n",
    "Nonlinear heat Equation\n",
    "\n",
    "        u_t - ga*u_xx + k1*u*ln(k2*u) + k3*exp(k4*u) = q"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "744088e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "import jax.numpy as jnp\n",
    "from jax import jit\n",
    "\n",
    "pi = jnp.pi\n",
    "\n",
    "# Analytical solution:\n",
    "@jit\n",
    "def eval_u(x, t, k2, k4):\n",
    "    A = jnp.sin(k2 * pi * x) + 1.5   # +1.5确保正值，1.5可调节\n",
    "    B = jnp.exp(-pi * k4 * x**2)\n",
    "    C = jnp.exp(-pi * t**2)\n",
    "    return A * B * C\n",
    "\n",
    "# ∂t = -2πt * u\n",
    "@jit\n",
    "def du_dt(x, t, k2, k4):\n",
    "    u = eval_u(x, t, k2, k4)\n",
    "    return -2 * pi * t * u\n",
    "\n",
    "# ∂²u/∂x²\n",
    "@jit\n",
    "def d2u_dx2(x, t, k2, k4):\n",
    "    alpha = k2 * pi\n",
    "\n",
    "    A = jnp.sin(alpha * x) + 1.5  \n",
    "    A1 = alpha * jnp.cos(alpha * x)\n",
    "    A2 = -alpha**2 * jnp.sin(alpha * x)\n",
    "\n",
    "    B = jnp.exp(-pi * k4 * x**2)\n",
    "    B1 = -2 * pi * k4 * x * B\n",
    "    B2 = (4 * pi**2 * k4**2 * x**2 - 2 * pi * k4) * B\n",
    "\n",
    "    C = jnp.exp(-pi * t**2)\n",
    "\n",
    "    return C * (A2 * B + 2 * A1 * B1 + A * B2)\n",
    "\n",
    "# q(x, t)\n",
    "@jit\n",
    "def eval_q(x, t, gamma, k1, k2, k3, k4):\n",
    "    u = eval_u(x, t, k2, k4)\n",
    "    u_t = du_dt(x, t, k2, k4)\n",
    "    u_xx = d2u_dx2(x, t, k2, k4)\n",
    "\n",
    "    nonlinear1 = k1 * u * jnp.log(k2 * u) \n",
    "    nonlinear2 = k3 * jnp.exp(k4 * u)\n",
    "\n",
    "    return u_t - gamma * u_xx + nonlinear1 + nonlinear2\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "809eb8c6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test task shape: (64, 5)\n",
      "Train task shape: (16, 5)\n"
     ]
    }
   ],
   "source": [
    "# Generate training and test sets\n",
    "seed = 1\n",
    "key, rng = random.split(random.PRNGKey(seed))\n",
    "def sample_param(key, n):\n",
    "    gamma = random.uniform(key, (n, 1), minval=0.5, maxval=5.0); key, _ = random.split(key)\n",
    "    k1 = random.uniform(key, (n, 1), minval=0.5, maxval=5.0); key, _ = random.split(key)\n",
    "    k2 = random.uniform(key, (n, 1), minval=0.5, maxval=2.0); key, _ = random.split(key)\n",
    "    k3 = random.uniform(key, (n, 1), minval=0.5, maxval=5.0); key, _ = random.split(key)\n",
    "    k4 = random.uniform(key, (n, 1), minval=0.5, maxval=2.0)\n",
    "    return jnp.hstack([gamma, k1, k2, k3, k4])\n",
    "\n",
    "key, rng = random.split(rng)\n",
    "test_task = sample_param(key, 64)\n",
    "\n",
    "key, rng = random.split(rng)\n",
    "train_task = sample_param(key, 16)\n",
    "\n",
    "print(\"Test task shape:\", test_task.shape)\n",
    "print(\"Train task shape:\", train_task.shape)\n",
    "n_task = len(train_task)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "id": "755fe01e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "#BC sample = 64\n",
      "#IC sample = 32\n"
     ]
    },
    {
     "data": {
      "image/png": "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OmjZsiWOHz/uur/lSRwiFiotLUXnzp0xYMAAvPfee/JDSM8884wcdJa0a9cuANAtuic92CTFFUh48Txp0bdAIKAqL+Vr4xBG6V60w9J34mDRPzeaO3cufv/9d521RwjB0qVL8e9//1tOW7lyJfLz87Fy5UrD9qTr1f6daAoEArjrrrvw7rvv4t///jf69eunKzNp0iQAwPPPP69K//vvv7FlyxYMHz4cVapUARB6tqFHjx7o3bu3Lr5mR1LfpWu54YYbMG7cOBw6dAi9evUyBaud64+FWPrD8r1++umnuOmmm3DttdfitddeM4wRavX999+jdu3auvhJx44dsWPHDt3NQSAQwMGDB3H11VcztV9hFL+JYcmhn3/+mQAgPp+PNGzYkDRp0oQ0a9aMtGzZkrRv357cd9995M8//ySEhBZuS0lJIWlpaWT+/Plk8+bNZOTIkaRZs2bycxwzZ84kR48eJcXFxaRmzZoEAHn33XdV5xwxYgQBQPr06aNKf++99wgAUrduXXLy5ElCSGgdIum5kqefflpVfsaMGQQAyc/PJ36/X07//vvvCQCSlpZGDh8+TAghZNGiRQQAadCgAVm5ciX55JNPyI033ig/y/LSSy+Rxx9/3OO/rrFmzJhBRFEkTZo0Ub3OO+88Uq1aNQKA7NixQy5//fXXEwCka9eu1PYCgYA8pXrOnDmG5w0EAmT16tWkXbt2pEaNGoaL+UkaMmSIajrvqVOnSJ8+fUh+fj45ceKEXO7ll1+Wp4sOGjTIzp9CpVmzZhEApF27dvIzC4FAgDRt2pQAINdcc43hVFRpEcb777/f8fm91E8//UQAkIyMDHLkyBFqGbPv9bfffiP33XcfSUlJIffff7/qf1ypTp06kfPOO4988MEHctqXX35JRo0aJf+OlNqxYwepWrUqueyyy+TfR0lJCbn77rtJ48aNydGjR51cbrkVhwiDpk6dSurUqUPq1KlDMjMziSiK8oAAgDRt2lR+kG/RokWkfv36pEqVKqRz587km2++IevXryc5OTnkH//4B/n666/J6tWrSa1ateT66enppEePHuTo0aPkvPPOU7V9zjnnkF9++YX861//Uq1kW7NmTfLss8+SvLw8Oc3n85E2bdoQQgi58MILiSAIcl6dOnXI2rVryYMPPkiysrLk9CpVqpCFCxeSYDBI7rvvPlKtWjVSt25dMmTIEPL777+T6dOnk8zMTNK7d2/5BxVt/ec//1H9DWiv/Px8VZ358+eTnJwcsmDBAl17U6ZMIXXr1lX9nZo1ayY/ECqpR48epE6dOuSSSy4hTzzxBCksLLTsayAQINOnTycNGzYkF1xwAWnVqhV5+OGH5UUPJf3www+kevXq5OKLLyYXXHCBg78KIc2bN1d9pw0bNiRr164lCxcuJCkpKXJ6Tk4OmThxolzvpZdeIg0aNJDzBUEgjRs3tgRkNNW6dWvd/2Hfvn115Wjf64IFC8g555xDzj33XDJkyBDy7bffmp7r5ZdfJo0aNSJpaWmkTZs25L777iPLli0zrbNjxw7yz3/+k9StW5e0bNmStGjRgowaNSphnpdKJHGIWOjo0aOkQ4cOugE0GAyS48ePk/Xr15NatWqRLVu2xKmHXMmkv/76i1x99dXx7gZXOVIgECBz5swhTZs2Jenp6aRJkybkpZdeYqq7bds2ct1115HLL7+ctG/fnqxYscL2+XlMxEIDBw5Efn6+Lv4hCAIqVaqEgoICXHrppQkVjOZKXC1evBjDhw+Pdze4ypGmTZuG77//HvPmzcOyZctQtWpVDBs2DE899ZRpvR07dqCgoAC33XYbNmzYgPnz5+PGG2/EqlWrbJ2fQ8REW7duxUcffYSMjAzDMkeOHEFeXp688BsXF02EELz++usoLi7GDTfcEO/ucJUTlZSU4O+//8bLL7+M9u3b49prr8WqVatw9tlnY+rUqbpFVpUaNWoU8vLy5JWxmzRpgn79+mHo0KGm9bTiEDHReeedh9atW2P27Nl47rnndE+nb9u2DfPnz8eTTz4Zpx5yJYu+/vprNG/eHA899FC8u8IVA505cwbFxcWOX6zbKhcXF+seBq1UqRK6deuG48ePG67mvGvXLqxYsQJXXXWVKr2goAD79u3DBx98wHyt/Il1E2VkZGDTpk144YUX8Nprr2HixImoXbs28vLycO6556Jz58548MEH491NriRQfn5+vLvAFSOdOXMGDTMzcchFG7Vr18bu3btNvSBAZPklrbKyspCTk2OY/8knnwDQb1TWtGlTAMD69evRp08fpr5yiFgoIyMD99xzD+655554d4WLiysJVFpaikMA9osCnKz5UAwg79AhHD16VLVqRHp6uuopejNt2rQJAwYMMNwoTdojSLv2W25uriqf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",
      "text/plain": [
       "<Figure size 400x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# train sample (32x32)\n",
    "x_l, x_u = -1, 1\n",
    "t_l, t_u = 0, 1\n",
    "xs = jnp.linspace(x_l, x_u, 32)\n",
    "ts = jnp.linspace(t_l, t_u, 32)\n",
    "x, t = jnp.meshgrid(xs, ts)\n",
    "x, t = x.reshape(-1, 1), t.reshape(-1, 1)\n",
    "x_train = np.hstack([x, t])\n",
    "x_bc = ((x == x_l) | (x == x_u)).flatten()\n",
    "t_ic = (t == t_l).flatten()\n",
    "xt_bc = ((x == x_l) | (x == x_u) | (t == 0)).flatten()\n",
    "print ('#BC sample = %d'%x_bc.sum())\n",
    "print ('#IC sample = %d'%t_ic.sum())\n",
    "# plot 1\n",
    "gamma, k1, k2, k3, k4 = train_task[0]\n",
    "u = eval_u(x, t, k2, k4)\n",
    "\n",
    "fig = plt.figure(figsize=(4, 4))\n",
    "u_plot = u.reshape(32, -1)\n",
    "\n",
    "ext = [t_l, t_u, x_l, x_u]     # plot boundary\n",
    "mesh2 = plt.imshow(u_plot.T, interpolation='bilinear', origin='lower', cmap='rainbow', extent=ext, aspect=0.5);\n",
    "plt.colorbar(mesh2); plt.xlabel('t'); plt.ylabel('x'); plt.title('gamma = %.3f,  k1 = %.3f\\nk2 = %.3f,  k3 = %.3f,  k4 = %.3f' % (gamma, k1, k2, k3, k4));"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e21554e",
   "metadata": {},
   "source": [
    "Baldwinian PINN (B-PINN)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b84fe638",
   "metadata": {},
   "outputs": [],
   "source": [
    "class PINN(nn.Module):\n",
    "    \"\"\"PINNs\"\"\"\n",
    "    def setup(self):\n",
    "        self.layers = [nn.Dense(n_nodes, kernel_init = jax.nn.initializers.he_uniform()),\n",
    "                       jnp.sin,\n",
    "                       nn.Dense(n_nodes, kernel_init = jax.nn.initializers.he_uniform()),\n",
    "                       jnp.sin]\n",
    "        #self.last_layer = nn.Dense(1, kernel_init = jax.nn.initializers.he_uniform(), use_bias=False)\n",
    "\n",
    "    @nn.compact\n",
    "    def __call__(self, inputs):\n",
    "\n",
    "        x, t = inputs[:,0:1], inputs[:,1:2]\n",
    "        def get_u(x, t):\n",
    "            f = jnp.hstack([x, t])\n",
    "            fs = []\n",
    "            for i, lyr in enumerate(self.layers):\n",
    "                f = lyr(f)\n",
    "                if (i == 0):\n",
    "                    f = 2 *jnp.pi *f\n",
    "                if (i%2 != 0):\n",
    "                    fs.append(f)\n",
    "            f = jnp.hstack(fs)\n",
    "            #u = self.last_layer(f)\n",
    "            return f\n",
    "\n",
    "        f = get_u(x, t)\n",
    "\n",
    "        def get_f_dir(get_u, x, t):\n",
    "            f_xx, f_t = jacfwd(jacfwd(get_u))(x, t), jacfwd(get_u, argnums=1)(x, t)\n",
    "            return f_xx, f_t\n",
    "\n",
    "        f_dir_vmap = vmap(get_f_dir, in_axes=(None, 0, 0))\n",
    "        f_xx, f_t = f_dir_vmap(get_u, x, t)\n",
    "        f_xx, f_t = f_xx[:,:,0,0], f_t[:,:,0]\n",
    "\n",
    "        outputs = jnp.hstack([f, f_xx, f_t])\n",
    "        return outputs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0fa8a424",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "np.int64(4352)"
      ]
     },
     "execution_count": 71,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# choose seed\n",
    "seed = 10\n",
    "key, rng = random.split(random.PRNGKey(seed))\n",
    "\n",
    "# dummy input\n",
    "a = random.normal(key, [1,2])\n",
    "\n",
    "# initialization call\n",
    "n_nodes = 64\n",
    "r_task = 8\n",
    "model = PINN()\n",
    "params = model.init(key, a)\n",
    "num_params, format_params_fn = get_params_format_fn(params)\n",
    "\n",
    "# flatten initial params\n",
    "params = jax.flatten_util.ravel_pytree(params)[0]\n",
    "num_params"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "id": "4a7214ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(4354,)"
      ]
     },
     "execution_count": 72,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# hyper-parameter \"lambs\" via SGD - append params to include 2 more parameters\n",
    "key, rng = random.split(rng) # update random generator\n",
    "ini_a = random.normal(key, [1,2])\n",
    "params = jnp.append(params, ini_a)\n",
    "params.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "364462c0",
   "metadata": {},
   "outputs": [],
   "source": [
    "from jax.scipy.linalg import solve\n",
    "import jax.numpy as jnp\n",
    "import jax\n",
    "import numpy as np\n",
    "n_step = 4\n",
    "\n",
    "def eval_loss(params, task):\n",
    "    gamma, k1, k2, k3, k4 = train_task[task]\n",
    "    u_bc = eval_u(x, t, k2, k4)\n",
    "    g = eval_q(x, t, gamma, k1, k2, k3, k4)\n",
    "    pred = model.apply(format_params_fn(params[:-2]), x_train)\n",
    "    f, f_xx, f_t = jnp.split(pred, 3, axis=1)\n",
    "    u = jnp.ones((f.shape[0], 1)) \n",
    "    lmbda = 10 ** (nn.sigmoid(params[-2]) * 8 - 6)   # (1e-6, 1e2)\n",
    "    lamb = 10 ** (nn.sigmoid(params[-1]) * 16 - 14)  # (1e-14, 1e2)\n",
    "\n",
    "    for i in range(n_step):\n",
    "        # u_t - ga*u_xx + k1*u*ln(k2*u) + k3*exp(k4*u) = q\n",
    "        pde = f_t - gamma * f_xx + k1 * f * jnp.log(k2 * u) \n",
    "        b_r = g - k3 * jnp.exp(k4 * u)\n",
    "        A = jnp.vstack([pde * lmbda, f[xt_bc]])\n",
    "        b = jnp.vstack([lmbda * b_r, u_bc[xt_bc]])\n",
    "        As = lamb * jnp.eye(A.shape[1]) + (A.T @ A)\n",
    "        bs = A.T @ b\n",
    "        w = solve(As, bs)\n",
    "        u = f @ w\n",
    "        u = jnp.clip(u, a_min=1e-6) \n",
    "    ssr = np.sum((b - A @ w) ** 2)\n",
    "    mse = jnp.mean((u_bc - u) ** 2)\n",
    "    rl2 = jnp.linalg.norm(u_bc - u) / jnp.linalg.norm(u_bc)\n",
    "    loss = mse\n",
    "    return loss, (ssr, mse, rl2)\n",
    "\n",
    "loss_grad = jax.jit(jax.value_and_grad(eval_loss, has_aux=True))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "41384e4d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# loss function--Newton linearization\n",
    "from jax.scipy.linalg import solve, solve_triangular\n",
    "n_step = 4\n",
    "def eval_loss(params, task):\n",
    "    gamma, k1, k2, k3, k4 = train_task[task]\n",
    "    u_bc = eval_u(x, t, k2, k4)\n",
    "    g = eval_q(x, t, gamma, k1, k2, k3, k4)\n",
    "    pred = model.apply(format_params_fn(params[:-2]), x_train)\n",
    "    f, f_xx, f_t = jnp.split(pred, 3, axis=1)\n",
    "    u = jnp.ones((f.shape[0], 1)) \n",
    "    lmbda = 10 ** (nn.sigmoid(params[-2]) * 8 - 6)   # (1e-6, 1e2)\n",
    "    lamb = 10 ** (nn.sigmoid(params[-1]) * 16 - 14)  # (1e-14, 1e2)\n",
    "\n",
    "    for i in range(n_step):\n",
    "        log_term = jnp.log(k2 * u)\n",
    "        exp_term = jnp.exp(k4 * u)\n",
    "        nonlinear1_grad = k1 * (log_term + 1.0)\n",
    "        nonlinear2_grad = k3 * k4 * exp_term\n",
    "        nonlinear1_const = -k1 * u\n",
    "        nonlinear2_const = k3 * (exp_term - k4 * exp_term * u)\n",
    "        nonlinear1_grad = k1 * (log_term + 1.0)\n",
    "        nonlinear2_grad = k3 * k4 * exp_term\n",
    "        pde = f_t - gamma * f_xx + (nonlinear1_grad + nonlinear2_grad) * f\n",
    "        b_r = g - nonlinear1_const - nonlinear2_const\n",
    "        A = jnp.vstack([pde * lmbda, f[xt_bc]])\n",
    "        b = jnp.vstack([lmbda * b_r, u_bc[xt_bc]])\n",
    "\n",
    "        As = lamb * jnp.eye(A.shape[1]) + A.T @ A\n",
    "        bs = A.T @ b\n",
    "        w = solve(As, bs)\n",
    "        u = f @ w\n",
    "        u = jnp.clip(u, a_min=1e-6) \n",
    "\n",
    "    ssr = jnp.sum((b - A @ w) ** 2)\n",
    "    mse = jnp.mean((u_bc - u) ** 2)\n",
    "    rl2 = jnp.linalg.norm(u_bc - u) / jnp.linalg.norm(u_bc)\n",
    "\n",
    "    return mse, (ssr, mse, rl2)\n",
    "\n",
    "loss_grad = jax.jit(jax.value_and_grad(eval_loss, has_aux=True))\n",
    "# loss_grad = jax.vmap(loss_grad)\n",
    "# loss_grad = jax.pmap(loss_grad)  # span multi GPUs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "id": "8b299160",
   "metadata": {},
   "outputs": [],
   "source": [
    "# weights update\n",
    "@jit\n",
    "def update(params, opt_state, key):\n",
    "    loss_all, ssr_all, mse_all, rl2_all, grad_all = 0, 0, 0, 0, 0\n",
    "    train_task = random.choice(key, n_task, (r_task,), replace=False)\n",
    "    # batch_train_task = random.choice(key, train_task_list, (r_task,), replace=False)\n",
    "    for task in train_task:\n",
    "        (loss, (ssr, mse, rl2)), grad = loss_grad(params, task)\n",
    "        loss_all += loss\n",
    "        ssr_all += ssr\n",
    "        mse_all += mse\n",
    "        rl2_all += rl2\n",
    "        grad_all += grad\n",
    "    loss_all /= r_task\n",
    "    ssr_all /= r_task\n",
    "    rl2_all /= r_task\n",
    "    mse_all /= r_task\n",
    "    grad_all /= r_task\n",
    "    updates, opt_state = optimizer.update(grad_all, opt_state)\n",
    "    params = optax.apply_updates(params, updates)\n",
    "    return params, opt_state, loss_all, ssr_all, mse_all, rl2_all"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3d81b0cb",
   "metadata": {},
   "outputs": [],
   "source": [
    "# optimizer\n",
    "max_iters = 2000\n",
    "max_lr = 5e-3\n",
    "lr_scheduler = optax.warmup_cosine_decay_schedule(init_value=max_lr, peak_value=max_lr, warmup_steps=int(max_iters*.4),\n",
    "                                                  decay_steps=max_iters, end_value=1e-6)\n",
    "optimizer = optax.adam(learning_rate=lr_scheduler) # Choose the method\n",
    "opt_state = optimizer.init(params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aeace165",
   "metadata": {},
   "outputs": [],
   "source": [
    "# training iteration\n",
    "runtime = 0\n",
    "train_iters = 0\n",
    "\n",
    "store = []\n",
    "\n",
    "while (train_iters <= max_iters) and (runtime < 18000):\n",
    "    # mini-batch update\n",
    "    start = time.time()\n",
    "    key, rng = random.split(rng) # update random generator\n",
    "    # print(key)\n",
    "    params, opt_state, loss, ssr, mse, rl2 = update(params, opt_state, key)\n",
    "    end = time.time()\n",
    "    runtime += (end-start)    \n",
    "    # append weights\n",
    "    if (train_iters % 100 == 0):\n",
    "        print ('iter. = %05d,  time = %03ds,  loss = %.2e  |  ssr = %.2e,  mse = %.2e,  rl2 = %.2e'%(train_iters, runtime, loss, ssr, mse, rl2))\n",
    "    store.append([train_iters, runtime, loss, ssr, mse, rl2])\n",
    "    train_iters += 1\n",
    "\n",
    "store = jnp.array(store)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a4744c23",
   "metadata": {},
   "outputs": [],
   "source": [
    "##   save weight\n",
    "import pickle\n",
    "import jax\n",
    "\n",
    "params_np = jax.device_get(params)\n",
    "store_np = jax.device_get(store)\n",
    "\n",
    "with open(\"Newton-PINet--heat-ln-exp.pkl\", \"wb\") as f:   ##   -newton\n",
    "    pickle.dump(params_np, f)\n",
    "with open(\"Newton-PINet--heat-ln-exp_store_np.pkl\", \"wb\") as f:\n",
    "    pickle.dump(store_np, f)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fdd99423",
   "metadata": {},
   "outputs": [],
   "source": [
    "###   load weight\n",
    "import pickle\n",
    "import jax\n",
    "\n",
    "with open(\"Newton-PINet--heat-ln-exp.pkl\", \"rb\") as f:\n",
    "    params_np = pickle.load(f)\n",
    "params = jax.numpy.asarray(params_np)\n",
    "\n",
    "# with open(\"PI-PINN-heat_store_np.pkl\", \"rb\") as f:\n",
    "#     store_np = pickle.load(f)\n",
    "# # 如果需要回到 JAX 数组\n",
    "# store = jax.numpy.asarray(store_np)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "082250f1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "#BC sample = 382\n"
     ]
    }
   ],
   "source": [
    "# test sample (128x128)\n",
    "xs = jnp.linspace(-1, 1, 128)\n",
    "ts = jnp.linspace(0, 1, 128)\n",
    "x, t = jnp.meshgrid(xs, ts)\n",
    "x, t = x.reshape(-1, 1), t.reshape(-1, 1)\n",
    "xt_bc = ((x == x_l) | (x == x_u) | (t == 0)).flatten()\n",
    "x_test = np.hstack([x, t])\n",
    "print ('#BC sample = %d'%xt_bc.sum())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "60e50377",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "one task runtime: 0.026022903621196747\n",
      "MEAN :  MSE = nan  RL2 = nan\n",
      "std  :  MSE = nan  RL2 = nan\n"
     ]
    }
   ],
   "source": [
    "# Performance of PINet on test tasks\n",
    "@jit\n",
    "def compute_ls_ntask(params, task):\n",
    "    gamma, k1, k2, k3, k4 = test_task[task]\n",
    "    u_bc = eval_u(x, t, k2, k4)\n",
    "    g = eval_q(x, t, gamma, k1, k2, k3, k4)\n",
    "    pred = model.apply(format_params_fn(params[:-2]), x_test)\n",
    "    f, f_xx, f_t = jnp.split(pred, 3, axis=1)\n",
    "    u = jnp.ones((f.shape[0], 1)) \n",
    "    lmbda = 10 ** (nn.sigmoid(params[-2]) * 8 - 6)   # (1e-6, 1e2)\n",
    "    lamb = 10 ** (nn.sigmoid(params[-1]) * 16 - 14)  # (1e-14, 1e2)\n",
    "\n",
    "    for i in range(n_step):\n",
    "        # u_t - ga*u_xx + k1*u*ln(k2*u) + k3*exp(k4*u) = q\n",
    "        pde = f_t - gamma * f_xx + k1 * f * jnp.log(k2 * u) \n",
    "        b_r = g - k3 * jnp.exp(k4 * u)\n",
    "        A = jnp.vstack([pde * lmbda, f[xt_bc]])\n",
    "        b = jnp.vstack([lmbda * b_r, u_bc[xt_bc]])\n",
    "        As = lamb * jnp.eye(A.shape[1]) + (A.T @ A)\n",
    "        bs = A.T @ b\n",
    "        w = solve(As, bs)\n",
    "        u = f @ w\n",
    "        u = jnp.clip(u, a_min=0.0)\n",
    "    ssr = np.sum((b - A @ w)**2)\n",
    "    mse = jnp.mean((u_bc - u)**2)\n",
    "    rl2 = jnp.linalg.norm(u_bc - u) / jnp.linalg.norm(u_bc)\n",
    "    return u, u_bc, ssr, mse, rl2\n",
    "\n",
    "\n",
    "mses, rl2s = [], []\n",
    "u_all = []\n",
    "label_all = []\n",
    "runtime=0\n",
    "start = time.time()\n",
    "for task in range(len(test_task)):\n",
    "    # problem\n",
    "    gamma, k1,k2,k3,k4 = test_task[task]\n",
    "    labels = eval_u(x, t, k2,k4)\n",
    "    # solve problem\n",
    "    u, u_bc_eval, ssr, mse, rl2 = compute_ls_ntask(params, task)\n",
    "    # print('%03d  a1, a2, w1, w2, w3, w4 = (%.1f, %.1f, %.1f, %.1f, %.1f, %.1f)\\\n",
    "    # | SSR = %.2e  MSE = %.2e  RL2 = %.2e'%(task, a1, a2, w1, w2, w3, w4, ssr, mse, rl2));\n",
    "    u_all.append(u)\n",
    "    label_all.append(labels)\n",
    "    mses.append(mse)\n",
    "    rl2s.append(rl2)\n",
    "\n",
    "mse_all, rl2_all = jnp.array(mses), jnp.array(rl2s)\n",
    "end = time.time()\n",
    "runtime += (end-start)\n",
    "print(\"one task runtime:\", runtime/len(test_task))\n",
    "u_all = jnp.stack(u_all)\n",
    "label_all = jnp.stack(label_all)\n",
    "print ('MEAN :  MSE = %.2e  RL2 = %.2e'%(mse_all.mean(), rl2_all.mean()))\n",
    "print ('std  :  MSE = %.2e  RL2 = %.2e'%(mse_all.std(), rl2_all.std()))\n",
    "\n",
    "np.savez(\n",
    "    \"PINet_u_label_plot_MSE_distribution.npz\",\n",
    "    u_total_plot=u_all,\n",
    "    label_total_plot=label_all,\n",
    "    results=mse_all\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c4c27bce",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "one task runtime: 0.026730071753263474\n",
      "MEAN :  MSE = 2.13e-06  RL2 = 1.39e-03\n",
      "std  :  MSE = 7.50e-06  RL2 = 2.43e-03\n"
     ]
    }
   ],
   "source": [
    "# Performance of Newton-PINet on test tasks\n",
    "@jit\n",
    "def compute_ls_ntask(params, task):\n",
    "    gamma, k1, k2, k3, k4 = test_task[task]\n",
    "    u_bc = eval_u(x, t, k2, k4)\n",
    "    g = eval_q(x, t, gamma, k1, k2, k3, k4)\n",
    "    pred = model.apply(format_params_fn(params[:-2]), x_test)\n",
    "    f, f_xx, f_t = jnp.split(pred, 3, axis=1)\n",
    "    u = jnp.ones((f.shape[0], 1)) \n",
    "    lmbda = 10 ** (nn.sigmoid(params[-2]) * 8 - 6)   # (1e-6, 1e2)\n",
    "    lamb = 10 ** (nn.sigmoid(params[-1]) * 16 - 14)  # (1e-14, 1e2)\n",
    "\n",
    "    for i in range(n_step):\n",
    "        log_term = jnp.log(k2 * u)\n",
    "        exp_term = jnp.exp(k4 * u)\n",
    "        nonlinear1_grad = k1 * (log_term + 1.0)\n",
    "        nonlinear2_grad = k3 * k4 * exp_term\n",
    "        nonlinear1_const = -k1 * u\n",
    "        nonlinear2_const = k3 * (exp_term - k4 * exp_term * u)\n",
    "        nonlinear1_grad = k1 * (log_term + 1.0)\n",
    "        nonlinear2_grad = k3 * k4 * exp_term\n",
    "        pde = f_t - gamma * f_xx + (nonlinear1_grad + nonlinear2_grad) * f\n",
    "        b_r = g - nonlinear1_const - nonlinear2_const\n",
    "        A = jnp.vstack([pde * lmbda, f[xt_bc]])\n",
    "        b = jnp.vstack([lmbda * b_r, u_bc[xt_bc]])\n",
    "\n",
    "        As = lamb * jnp.eye(A.shape[1]) + A.T @ A\n",
    "        bs = A.T @ b\n",
    "        w = solve(As, bs)\n",
    "        u = f @ w\n",
    "        u = jnp.clip(u, a_min=1e-6) \n",
    "    ssr = np.sum((b - A @ w)**2)\n",
    "    mse = jnp.mean((u_bc - u)**2)\n",
    "    rl2 = jnp.linalg.norm(u_bc - u) / jnp.linalg.norm(u_bc)\n",
    "    return u, u_bc, ssr, mse, rl2\n",
    "\n",
    "\n",
    "mses, rl2s = [], []\n",
    "u_all = []\n",
    "label_all = []\n",
    "runtime=0\n",
    "start = time.time()\n",
    "for task in range(len(test_task)):\n",
    "    # problem\n",
    "    gamma, k1,k2,k3,k4 = test_task[task]\n",
    "    labels = eval_u(x, t, k2,k4)\n",
    "    # solve problem\n",
    "    u, u_bc_eval, ssr, mse, rl2 = compute_ls_ntask(params, task)\n",
    "    # print('%03d  a1, a2, w1, w2, w3, w4 = (%.1f, %.1f, %.1f, %.1f, %.1f, %.1f)\\\n",
    "    # | SSR = %.2e  MSE = %.2e  RL2 = %.2e'%(task, a1, a2, w1, w2, w3, w4, ssr, mse, rl2));\n",
    "    u_all.append(u)\n",
    "    label_all.append(labels)\n",
    "    mses.append(mse)\n",
    "    rl2s.append(rl2)\n",
    "\n",
    "mse_all, rl2_all = jnp.array(mses), jnp.array(rl2s)\n",
    "end = time.time()\n",
    "runtime += (end-start)\n",
    "print(\"one task runtime:\", runtime/len(test_task))\n",
    "u_all = jnp.stack(u_all)\n",
    "label_all = jnp.stack(label_all)\n",
    "print ('MEAN :  MSE = %.2e  RL2 = %.2e'%(mse_all.mean(), rl2_all.mean()))\n",
    "print ('std  :  MSE = %.2e  RL2 = %.2e'%(mse_all.std(), rl2_all.std()))\n",
    "\n",
    "np.savez(\n",
    "    \"Newton-PINet_u_label_plot_MSE_distribution.npz\",\n",
    "    u_total_plot=u_all,\n",
    "    label_total_plot=label_all,\n",
    "    results=mse_all\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e13e9016",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as colors\n",
    "from matplotlib.ticker import FormatStrFormatter\n",
    "\n",
    "plt.rcParams['font.family'] = 'Times New Roman'\n",
    "plt.rcParams['font.size'] = 14\n",
    "\n",
    "t_dim, x_dim = 128, 128\n",
    "\n",
    "idx_max = np.argmax(mse_all)\n",
    "idx_median = 19\n",
    "\n",
    "def reshape_field(data, idx):\n",
    "    return data[idx, :, 0].reshape(t_dim, x_dim).T\n",
    "\n",
    "u_max, label_max = reshape_field(u_all, idx_max), reshape_field(label_all, idx_max)\n",
    "u_med, label_med = reshape_field(u_all, idx_median), reshape_field(label_all, idx_median)\n",
    "vmin_max = min(u_max.min(), label_max.min())\n",
    "vmax_max = max(u_max.max(), label_max.max())\n",
    "vmin_med = min(u_med.min(), label_med.min())\n",
    "vmax_med = max(u_med.max(), label_med.max())\n",
    "err_max = np.abs(u_max - label_max)\n",
    "err_med = np.abs(u_med - label_med)\n",
    "\n",
    "gamma_max, k1_max,k2_max,k3_max,k4_max = test_task[idx_max]\n",
    "gamma_med, k1_med,k2_med,k3_med,k4_med= test_task[idx_median]\n",
    "\n",
    "scale_r = 0.5\n",
    "fig = plt.figure(figsize=(12,6))\n",
    "gs = fig.add_gridspec(2, 4, width_ratios=[0.7,1,1,1], height_ratios=[1,1])\n",
    "\n",
    "ax_box = fig.add_subplot(gs[:,0])\n",
    "bp = ax_box.boxplot(\n",
    "    mse_all, widths=0.3, vert=True, patch_artist=True, showfliers=True,\n",
    "    flierprops=dict(marker=\"D\", markerfacecolor=\"black\", markersize=5)\n",
    ")\n",
    "for patch in bp[\"boxes\"]:\n",
    "    patch.set(facecolor=\"#d95f02\", edgecolor=\"black\", linewidth=1.5) \n",
    "for whisker in bp[\"whiskers\"]:\n",
    "    whisker.set(color=\"black\", linewidth=1.2)\n",
    "for cap in bp[\"caps\"]:\n",
    "    cap.set(color=\"black\", linewidth=1.2)\n",
    "for median in bp[\"medians\"]:\n",
    "    median.set(color=\"black\", linewidth=2) \n",
    "\n",
    "ax_box.set_yscale(\"log\")\n",
    "ax_box.set_ylabel(\"Test MSE\")\n",
    "ax_box.tick_params(axis='y', labelsize=16)\n",
    "ax_box.set_xticks([])\n",
    "ax_box.set_xticklabels([])\n",
    "ax_box.set_xlabel(\"Newton-PINet\")\n",
    "\n",
    "def format_axis(ax):\n",
    "    ax.set_xticks([])\n",
    "    ax.set_yticks([])\n",
    "    ax.set_xlabel(\"t\")\n",
    "    ax.set_ylabel(\"x\")\n",
    "    ax.xaxis.set_major_formatter(FormatStrFormatter('%.1f'))\n",
    "    ax.yaxis.set_major_formatter(FormatStrFormatter('%.1f'))\n",
    "\n",
    "ax_pred_max = fig.add_subplot(gs[0,1])\n",
    "im1 = ax_pred_max.imshow(u_max, cmap=\"jet\", origin=\"lower\",\n",
    "                         extent=[0,1,-1,1], vmin=vmin_med, vmax=vmax_med, aspect=scale_r)\n",
    "ax_pred_max.set_title(f\"Newton-PINet\")#γ, k1, k2, k3 = {gamma_med:.1f}, {k1_med:.1f}, {k2_med:.1f}, {k3_med:.1f}\\n\n",
    "format_axis(ax_pred_max)\n",
    "cbar1 = plt.colorbar(im1, ax=ax_pred_max, fraction=0.046, pad=0.04)\n",
    "# cbar1.set_ticks([-1,0,1])\n",
    "\n",
    "ax_truth_max = fig.add_subplot(gs[0,2])\n",
    "im2 = ax_truth_max.imshow(label_max, cmap=\"jet\", origin=\"lower\",\n",
    "                          extent=[0,1,-1,1], vmin=vmin_med, vmax=vmax_med, aspect=scale_r)\n",
    "ax_truth_max.set_title(\"Ground truth\")\n",
    "format_axis(ax_truth_max)\n",
    "cbar2 = plt.colorbar(im2, ax=ax_truth_max, fraction=0.046, pad=0.04)\n",
    "# cbar2.set_ticks([-1,0,1])\n",
    "\n",
    "ax_err_max = fig.add_subplot(gs[0,3])\n",
    "im3 = ax_err_max.imshow(err_max, cmap=\"jet\", origin=\"lower\",\n",
    "                        extent=[0,1,-1,1], norm=colors.LogNorm(vmin=1e-6, vmax=1e-2),\n",
    "                        aspect=scale_r)\n",
    "ax_err_max.set_title(r'$|u_{\\mathrm{predict}} - u_{\\mathrm{truth}}|$')\n",
    "format_axis(ax_err_max)\n",
    "plt.colorbar(im3, ax=ax_err_max, fraction=0.046, pad=0.04)\n",
    "\n",
    "ax_pred_med = fig.add_subplot(gs[1,1])\n",
    "im4 = ax_pred_med.imshow(u_med, cmap=\"jet\", origin=\"lower\",\n",
    "                         extent=[0,1,-1,1], vmin=vmin_med, vmax=vmax_med, aspect=scale_r)\n",
    "ax_pred_med.set_title(f\"Newton-PINet\") #γ, k1, k2, k3 = {gamma_max:.1f}, {k1_max:.1f}, {k2_max:.1f}, {k3_max:.1f}\\n\n",
    "format_axis(ax_pred_med)\n",
    "cbar4 = plt.colorbar(im4, ax=ax_pred_med, fraction=0.046, pad=0.04)\n",
    "# cbar4.set_ticks([-1,0,1])\n",
    "\n",
    "ax_truth_med = fig.add_subplot(gs[1,2])\n",
    "im5 = ax_truth_med.imshow(label_med, cmap=\"jet\", origin=\"lower\",\n",
    "                          extent=[0,1,-1,1], vmin=vmin_med, vmax=vmax_med, aspect=scale_r)\n",
    "ax_truth_med.set_title(\"Ground truth\")\n",
    "format_axis(ax_truth_med)\n",
    "cbar5 = plt.colorbar(im5, ax=ax_truth_med, fraction=0.046, pad=0.04)\n",
    "# cbar5.set_ticks([-1,0,1])\n",
    "\n",
    "ax_err_med = fig.add_subplot(gs[1,3])\n",
    "im6 = ax_err_med.imshow(err_med, cmap=\"rainbow\", origin=\"lower\",\n",
    "                        extent=[0,1,-1,1], norm=colors.LogNorm(vmin=1e-6, vmax=1e-2),\n",
    "                        aspect=scale_r)\n",
    "ax_err_med.set_title(r'$|u_{\\mathrm{predict}} - u_{\\mathrm{truth}}|$')\n",
    "format_axis(ax_err_med)\n",
    "plt.colorbar(im6, ax=ax_err_med, fraction=0.046, pad=0.04)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"heat_log_1_window_u_plot.png\", dpi=300, bbox_inches='tight')  # 可取消注释以保存\n",
    "plt.show()\n"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "provenance": []
  },
  "kernelspec": {
   "display_name": "star",
   "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.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
