Skip to content

Instantly share code, notes, and snippets.

@aakashns
Created May 14, 2018 06:55
Show Gist options
  • Save aakashns/82db9df1e6c20eb13523903507dbd537 to your computer and use it in GitHub Desktop.
Save aakashns/82db9df1e6c20eb13523903507dbd537 to your computer and use it in GitHub Desktop.
Visualizing linear regression
Display the source blob
Display the rendered blob
Raw
{
"cells": [
{
"cell_type": "markdown",
"metadata": {},
"source": [
"# Training a single linear neuron with fixed input size"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"Todo:\n",
"\n",
"1. Generate and visualize some training data (for the network)\n",
"2. Train a network and collect the weights\n",
"3. Save the data to file for later use"
]
},
{
"cell_type": "code",
"execution_count": 168,
"metadata": {},
"outputs": [],
"source": [
"import numpy as np\n",
"import matplotlib.pyplot as plt\n",
"from matplotlib.animation import FuncAnimation\n",
"import seaborn as sns\n",
"import pandas as pd\n",
"%matplotlib inline"
]
},
{
"cell_type": "code",
"execution_count": 169,
"metadata": {},
"outputs": [],
"source": [
"sns.set_style(style='whitegrid')\n",
"plt.rcParams[\"patch.force_edgecolor\"] = True"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"## Create a single training set"
]
},
{
"cell_type": "code",
"execution_count": 170,
"metadata": {},
"outputs": [],
"source": [
"m = 2 # slope\n",
"c = 3 # intercept"
]
},
{
"cell_type": "code",
"execution_count": 171,
"metadata": {},
"outputs": [],
"source": [
"x = np.random.rand(256)"
]
},
{
"cell_type": "code",
"execution_count": 172,
"metadata": {},
"outputs": [],
"source": [
"noise = np.random.randn(256) / 4"
]
},
{
"cell_type": "code",
"execution_count": 173,
"metadata": {},
"outputs": [],
"source": [
"y = x * m + c + noise"
]
},
{
"cell_type": "code",
"execution_count": 174,
"metadata": {},
"outputs": [],
"source": [
"df = pd.DataFrame()\n",
"df['x'] = x\n",
"df['y'] = y"
]
},
{
"cell_type": "code",
"execution_count": 175,
"metadata": {},
"outputs": [
{
"data": {
"text/plain": [
"<seaborn.axisgrid.FacetGrid at 0x1a1b91c2e8>"
]
},
"execution_count": 175,
"metadata": {},
"output_type": "execute_result"
},
{
"data": {
"image/png": "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\n",
"text/plain": [
"<matplotlib.figure.Figure at 0x1a1b91ca20>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"sns.lmplot(x='x', y='y', data=df)"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"## Implement Linear Regression"
]
},
{
"cell_type": "code",
"execution_count": 176,
"metadata": {},
"outputs": [],
"source": [
"import torch\n",
"import torch.nn as nn\n",
"from torch.autograd import Variable"
]
},
{
"cell_type": "code",
"execution_count": 177,
"metadata": {},
"outputs": [],
"source": [
"x_train = x.reshape(-1, 1).astype('float32')"
]
},
{
"cell_type": "code",
"execution_count": 178,
"metadata": {},
"outputs": [],
"source": [
"y_train = y.reshape(-1, 1).astype('float32')"
]
},
{
"cell_type": "code",
"execution_count": 179,
"metadata": {},
"outputs": [],
"source": [
"class LinearRegressionModel(nn.Module):\n",
" def __init__(self, input_dim, output_dim):\n",
" super(LinearRegressionModel, self).__init__()\n",
" self.linear = nn.Linear(input_dim, output_dim)\n",
" \n",
" def forward(self, x):\n",
" out = self.linear(x)\n",
" return out"
]
},
{
"cell_type": "code",
"execution_count": 180,
"metadata": {},
"outputs": [
{
"data": {
"text/plain": [
"(1, 1)"
]
},
"execution_count": 180,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"input_dim = x_train.shape[1]\n",
"output_dim = y_train.shape[1]\n",
"input_dim, output_dim"
]
},
{
"cell_type": "code",
"execution_count": 181,
"metadata": {},
"outputs": [],
"source": [
"model = LinearRegressionModel(input_dim, output_dim)"
]
},
{
"cell_type": "code",
"execution_count": 182,
"metadata": {},
"outputs": [],
"source": [
"criterion = nn.MSELoss()"
]
},
{
"cell_type": "code",
"execution_count": 183,
"metadata": {},
"outputs": [],
"source": [
"[w, b] = model.parameters()"
]
},
{
"cell_type": "code",
"execution_count": 184,
"metadata": {},
"outputs": [],
"source": [
"def get_param_values():\n",
" return w.data[0][0], b.data[0]"
]
},
{
"cell_type": "code",
"execution_count": 185,
"metadata": {},
"outputs": [],
"source": [
"def plot_current_fit(title=\"\"):\n",
" plt.figure(figsize=(12,4))\n",
" plt.title(title)\n",
" plt.scatter(x, y, s=8)\n",
" w1 = w.data[0][0]\n",
" b1 = b.data[0]\n",
" x1 = np.array([0., 1.])\n",
" y1 = x1 * w1 + b1\n",
" plt.plot(x1, y1, 'r', label='Current Fit ({:.3f}, {:.3f})'.format(w1, b1))\n",
" plt.xlabel('x (input)')\n",
" plt.ylabel('y (target)')\n",
" plt.legend()\n",
" plt.show()"
]
},
{
"cell_type": "code",
"execution_count": 186,
"metadata": {},
"outputs": [
{
"data": {
"image/png": "iVBORw0KGgoAAAANSUhEUgAAAsgAAAEPCAYAAABfrjLnAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADl0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uIDIuMS4yLCBodHRwOi8vbWF0cGxvdGxpYi5vcmcvNQv5yAAAIABJREFUeJzt3Xl4FFWiNvC3urORBSQkgQCGJUCMDMgqYmQX8QLiFWUxEQERFQFFnWFzgUEuiwqKLGHAnYsCKjjIx4iKCIKCECSAN4CAiEBCgoGEDmTrPt8fsdvqTu9dXdVJv7/nmWdIqrvq1DkV+63Tp86RhBACREREREQEANBpXQAiIiIiokDCgExEREREJMOATEREREQkw4BMRERERCTDgExEREREJMOATEREREQkw4BMROSGc+fOITU1Fffeey/uvfde3HPPPRg2bBiysrJcvtdgMGDkyJEYNGgQvvzyS8XLdvXqVTz88MMev2/79u2YO3eu09dcvHgRI0eO9LZoREQ1UojWBSAiqikiIiLw73//2/Lz1q1bMWPGDJehNycnB3/88Qe++uorv5SrqKgIR44c8fh9/fr1Q79+/Zy+pmHDhli3bp23RSMiqpEYkImIvHTlyhXEx8dbfv7mm2+QmZmJiooKREREYNq0aahXrx5mzpyJixcv4t5778X69euxe/duLFu2DCaTCVFRUZgxYwbat2+PpUuX4tChQ8jPz0dKSgpee+01ZGZm4ssvv4TJZEKTJk0wa9YsNGzY0KocM2bMQGlpKe69915s3LgRt9xyC/r164djx47htddew/Hjx7F+/XpUVFSgqKgI48ePR3p6OjZu3Iht27bhX//6F0aNGoUOHTrg4MGDyM3NRffu3fHyyy/jwoULuOeee/DTTz9h6dKlOH/+PAoKCnD+/Hk0bNgQr776KhISEnD48GHMnj0bFRUVSEpKwoULFzB9+nR069ZN7WYhIvKdICIil37//Xdx0003iSFDhoghQ4aI3r17i7Zt24pvv/1WCCHEr7/+KgYPHiwKCwuFEEKcOHFCpKWliZKSErF3714xaNAgIYQQJ0+eFLfffrs4e/asEEKI77//XqSlpYmrV6+KN998UwwYMEBUVFQIIYTYtGmTmDJliuXndevWiUcffdRu2Tp06GD5uU2bNmLTpk1CCCEMBoMYPny4pVw//fST5bWffvqpeOyxx4QQQjz00EPiqaeeEkajUVy9elXccccd4ocffrDa95tvvin69esnrl69KoQQ4vHHHxdLliwRFRUVomfPnpa6+OGHH0RKSorYu3evInVPRKQ29iATEbnJdojF999/j4kTJ2Lz5s3Ys2cP8vPzMWbMGMt2SZJw9uxZq33s3bsXt912G2688UYAQPfu3REbG4ujR48CADp06ICQkKr/NO/YsQNHjhzB/fffDwAwmUy4fv26W2Xt0qULACAqKgorV67Ezp07cebMGRw7dgzXrl2z+54+ffpAp9MhOjoazZo1Q1FREZo2bWr1mltvvRXR0dEAgJtvvhlFRUU4ceIEAKBXr14AgNtuuw2tW7d2q5xERIGIAZmIyEu33347kpKScOTIEZhMJnTv3h1vvPGGZXtubi4SEhJw4MABy+9MJhMkSbLajxAClZWVAIDIyEir1z766KNIT08HAJSXl6OoqMitspn3k5eXhxEjRmD48OHo3Lkz7r77buzYscPueyIiIiz/liQJQgi3XqPX66u9Vq/Xu1VOIqJAxFksiIi89Ouvv+L8+fNITU1F9+7dsWfPHpw6dQoAsHPnTgwZMgSlpaVW7+nevTt2796N33//HQDwww8/IDc3F7fccku1/d9xxx345JNPYDAYAABLlizB1KlTq70uJCQERqPRbqA9evQoYmNj8eSTT+KOO+6whGOj0ejbycskJycjLCwMu3btAgAcPnwYJ06cqHYjQERUU7AHmYjITeYH4cxMJhPmzJmDFi1aAADmzJmDZ599FkIIhISEIDMzE1FRUVb7aNWqFWbNmoVJkybBaDQiIiICK1euRExMTLXjDRs2DBcvXsTw4cMhSRISExOxYMGCaq+Lj49H+/btMWjQIKxdu9ZqW1paGj755BPcfffdkCQJt956K2JjY/Hbb78pUSUAqgL60qVLMWvWLCxevBjNmzdHXFycVW8zEVFNIgl7XQ5EREQeWLhwIcaNG4e4uDjk5ubi3nvvxddff426detqXTQiIo+xB5mIiHzWpEkTjBkzBiEhIRBCYO7cuQzHRFRjsQeZiIiIiEiGD+kREREREckwIBMRERERyTAgExERERHJBNRDellZWVoXgYiIiIiCROfOne3+PqACMuC4oP6Wk5OD1NRUTY5N6mAbBwe2c3BgOwcHtnPtp2Ub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Z+0shMCTTz6JlJQUzJkzB3q9Xqtiko+ctfPUqVPx8ccfY82aNbjvvvswZswYhuMayFkb/+1vf8OJEydQWFiIyspKZGdno1WrVloVlXzgrJ3r1auHiIgIhIWFITw8HDExMSguLtaqqOQnycnJ+O2333DlyhWUl5fjwIED6Nixo6ZlCqoe5P79+2PPnj0YOXIkhBCYN28e3n33XSQlJaFfv34YNWoU0tPTIYTAM888w7GpNZSzdjaZTPjxxx9RXl6O7777DgDw7LPPav6HSJ5z9fdMNZ+rNn7uuefw6KOPAgDuvvtudmrUUK7a+fvvv8fw4cOh0+nQqVMnpKWlaV1kUsjnn3+Oa9euYcSIEZg+fTrGjRsHIQTuv/9+NGzYUNOycSU9IiIiIiKZoBpiQURERETkCgMyEREREZEMAzIRERERkQwDMhERERGRDAMyEREREZEMAzIRUQA7cOAA3n//feTk5GDZsmWK7fd///d/AQAFBQWYM2eOYvslIqoNGJCJiAKUEAJLly7Fgw8+iNTUVEyaNEmxfWdmZgIA4uPjERUVhR9//FGxfRMR1XQMyEREGlm7di2ee+45AMC0adOwdu1aq+179uxBq1atEBYWhn379uGZZ54BANx1112YPn06RowYgSeffBJGoxEbN27ExIkTMXr0aAwZMgTbtm0DAPTt2xdlZWUAgNdeew0bN25EZmYmioqKMHv2bADA4MGD8cEHH6h01kREgY8BmYhIIxkZGbh+/TqmT5+OiooKZGRkWG3/8ccfkZKSUu19v//+O55++mmsX78ehYWFOHLkCADg2rVrePfdd/HOO+9gwYIFqKystHvcCRMmoF69epaA3KpVKxw8eFDZkyMiqsEYkImINPTYY49h06ZNGDduXLVtly9fRoMGDar9vn79+khMTAQAJCYmWnqIu3btCp1Oh7i4ONStWxeFhYVW73O0cKper4der4fJZPL1dIiIagUGZCIijZSXl2PevHmYM2cOZs+ejfLycqvtsbGxuHr1arX3SZJkd38///wzAODSpUswGAxo0KABwsLCkJ+fDyEEjh07ZnmtPCwLIRASEgKdjh8JREQAAzIRkWZee+019O7dGyNGjEDPnj2xaNEiq+3dunVDdna22/u7dOkSRo8ejcceewyzZs2CXq/Ho48+isceewzjx49H3bp1La9NTk7G3//+dwDA8ePH0aFDB2VOioioFpCEo+/ciIhIUyaTCaNHj8bbb7+NsLAwp6/duHEjTp8+bQm9nnjllVfQt29fdOnSxduiEhHVKuxBJiIKUDqdDhMnTsSHH37ot2MUFBTAYDAwHBMRybAHmYiIiIhIhj3IREREREQyDMhERERERDIMyEREREREMgzIREREREQyDMhERERERDIMyEREREREMv8ft9RzGPYaXvMAAAAASUVORK5CYII=\n",
"text/plain": [
"<matplotlib.figure.Figure at 0x1a1b904dd8>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"plot_current_fit('Before training')"
]
},
{
"cell_type": "code",
"execution_count": 187,
"metadata": {},
"outputs": [],
"source": [
"learning_rate = 0.01\n",
"optimizer = torch.optim.SGD(model.parameters(), lr=learning_rate)"
]
},
{
"cell_type": "code",
"execution_count": 188,
"metadata": {},
"outputs": [],
"source": [
"def run_epoch(epoch):\n",
" # Convert from numpy array to torch tensors\n",
" inputs = Variable(torch.from_numpy(x_train))\n",
" labels = Variable(torch.from_numpy(y_train))\n",
" \n",
" # Clear the gradients w.r.t. parameters\n",
" optimizer.zero_grad()\n",
" \n",
" # Forward to get the outputs\n",
" outputs = model(inputs)\n",
" \n",
" # Calcuate loss\n",
" loss = criterion(outputs, labels)\n",
" \n",
" # Getting gradients from parameters\n",
" loss.backward()\n",
" \n",
" # Updating parameters\n",
" optimizer.step()\n",
" \n",
"# print('epoch {}, loss {}'.format(epoch, loss.data[0]))\n",
" \n",
" return loss"
]
},
{
"cell_type": "code",
"execution_count": 190,
"metadata": {},
"outputs": [
{
"data": {
"application/javascript": [
"/* Put everything inside the global mpl namespace */\n",
"window.mpl = {};\n",
"\n",
"\n",
"mpl.get_websocket_type = function() {\n",
" if (typeof(WebSocket) !== 'undefined') {\n",
" return WebSocket;\n",
" } else if (typeof(MozWebSocket) !== 'undefined') {\n",
" return MozWebSocket;\n",
" } else {\n",
" alert('Your browser does not have WebSocket support.' +\n",
" 'Please try Chrome, Safari or Firefox ≥ 6. ' +\n",
" 'Firefox 4 and 5 are also supported but you ' +\n",
" 'have to enable WebSockets in about:config.');\n",
" };\n",
"}\n",
"\n",
"mpl.figure = function(figure_id, websocket, ondownload, parent_element) {\n",
" this.id = figure_id;\n",
"\n",
" this.ws = websocket;\n",
"\n",
" this.supports_binary = (this.ws.binaryType != undefined);\n",
"\n",
" if (!this.supports_binary) {\n",
" var warnings = document.getElementById(\"mpl-warnings\");\n",
" if (warnings) {\n",
" warnings.style.display = 'block';\n",
" warnings.textContent = (\n",
" \"This browser does not support binary websocket messages. \" +\n",
" \"Performance may be slow.\");\n",
" }\n",
" }\n",
"\n",
" this.imageObj = new Image();\n",
"\n",
" this.context = undefined;\n",
" this.message = undefined;\n",
" this.canvas = undefined;\n",
" this.rubberband_canvas = undefined;\n",
" this.rubberband_context = undefined;\n",
" this.format_dropdown = undefined;\n",
"\n",
" this.image_mode = 'full';\n",
"\n",
" this.root = $('<div/>');\n",
" this._root_extra_style(this.root)\n",
" this.root.attr('style', 'display: inline-block');\n",
"\n",
" $(parent_element).append(this.root);\n",
"\n",
" this._init_header(this);\n",
" this._init_canvas(this);\n",
" this._init_toolbar(this);\n",
"\n",
" var fig = this;\n",
"\n",
" this.waiting = false;\n",
"\n",
" this.ws.onopen = function () {\n",
" fig.send_message(\"supports_binary\", {value: fig.supports_binary});\n",
" fig.send_message(\"send_image_mode\", {});\n",
" if (mpl.ratio != 1) {\n",
" fig.send_message(\"set_dpi_ratio\", {'dpi_ratio': mpl.ratio});\n",
" }\n",
" fig.send_message(\"refresh\", {});\n",
" }\n",
"\n",
" this.imageObj.onload = function() {\n",
" if (fig.image_mode == 'full') {\n",
" // Full images could contain transparency (where diff images\n",
" // almost always do), so we need to clear the canvas so that\n",
" // there is no ghosting.\n",
" fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height);\n",
" }\n",
" fig.context.drawImage(fig.imageObj, 0, 0);\n",
" };\n",
"\n",
" this.imageObj.onunload = function() {\n",
" fig.ws.close();\n",
" }\n",
"\n",
" this.ws.onmessage = this._make_on_message_function(this);\n",
"\n",
" this.ondownload = ondownload;\n",
"}\n",
"\n",
"mpl.figure.prototype._init_header = function() {\n",
" var titlebar = $(\n",
" '<div class=\"ui-dialog-titlebar ui-widget-header ui-corner-all ' +\n",
" 'ui-helper-clearfix\"/>');\n",
" var titletext = $(\n",
" '<div class=\"ui-dialog-title\" style=\"width: 100%; ' +\n",
" 'text-align: center; padding: 3px;\"/>');\n",
" titlebar.append(titletext)\n",
" this.root.append(titlebar);\n",
" this.header = titletext[0];\n",
"}\n",
"\n",
"\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(canvas_div) {\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._init_canvas = function() {\n",
" var fig = this;\n",
"\n",
" var canvas_div = $('<div/>');\n",
"\n",
" canvas_div.attr('style', 'position: relative; clear: both; outline: 0');\n",
"\n",
" function canvas_keyboard_event(event) {\n",
" return fig.key_event(event, event['data']);\n",
" }\n",
"\n",
" canvas_div.keydown('key_press', canvas_keyboard_event);\n",
" canvas_div.keyup('key_release', canvas_keyboard_event);\n",
" this.canvas_div = canvas_div\n",
" this._canvas_extra_style(canvas_div)\n",
" this.root.append(canvas_div);\n",
"\n",
" var canvas = $('<canvas/>');\n",
" canvas.addClass('mpl-canvas');\n",
" canvas.attr('style', \"left: 0; top: 0; z-index: 0; outline: 0\")\n",
"\n",
" this.canvas = canvas[0];\n",
" this.context = canvas[0].getContext(\"2d\");\n",
"\n",
" var backingStore = this.context.backingStorePixelRatio ||\n",
"\tthis.context.webkitBackingStorePixelRatio ||\n",
"\tthis.context.mozBackingStorePixelRatio ||\n",
"\tthis.context.msBackingStorePixelRatio ||\n",
"\tthis.context.oBackingStorePixelRatio ||\n",
"\tthis.context.backingStorePixelRatio || 1;\n",
"\n",
" mpl.ratio = (window.devicePixelRatio || 1) / backingStore;\n",
"\n",
" var rubberband = $('<canvas/>');\n",
" rubberband.attr('style', \"position: absolute; left: 0; top: 0; z-index: 1;\")\n",
"\n",
" var pass_mouse_events = true;\n",
"\n",
" canvas_div.resizable({\n",
" start: function(event, ui) {\n",
" pass_mouse_events = false;\n",
" },\n",
" resize: function(event, ui) {\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" stop: function(event, ui) {\n",
" pass_mouse_events = true;\n",
" fig.request_resize(ui.size.width, ui.size.height);\n",
" },\n",
" });\n",
"\n",
" function mouse_event_fn(event) {\n",
" if (pass_mouse_events)\n",
" return fig.mouse_event(event, event['data']);\n",
" }\n",
"\n",
" rubberband.mousedown('button_press', mouse_event_fn);\n",
" rubberband.mouseup('button_release', mouse_event_fn);\n",
" // Throttle sequential mouse events to 1 every 20ms.\n",
" rubberband.mousemove('motion_notify', mouse_event_fn);\n",
"\n",
" rubberband.mouseenter('figure_enter', mouse_event_fn);\n",
" rubberband.mouseleave('figure_leave', mouse_event_fn);\n",
"\n",
" canvas_div.on(\"wheel\", function (event) {\n",
" event = event.originalEvent;\n",
" event['data'] = 'scroll'\n",
" if (event.deltaY < 0) {\n",
" event.step = 1;\n",
" } else {\n",
" event.step = -1;\n",
" }\n",
" mouse_event_fn(event);\n",
" });\n",
"\n",
" canvas_div.append(canvas);\n",
" canvas_div.append(rubberband);\n",
"\n",
" this.rubberband = rubberband;\n",
" this.rubberband_canvas = rubberband[0];\n",
" this.rubberband_context = rubberband[0].getContext(\"2d\");\n",
" this.rubberband_context.strokeStyle = \"#000000\";\n",
"\n",
" this._resize_canvas = function(width, height) {\n",
" // Keep the size of the canvas, canvas container, and rubber band\n",
" // canvas in synch.\n",
" canvas_div.css('width', width)\n",
" canvas_div.css('height', height)\n",
"\n",
" canvas.attr('width', width * mpl.ratio);\n",
" canvas.attr('height', height * mpl.ratio);\n",
" canvas.attr('style', 'width: ' + width + 'px; height: ' + height + 'px;');\n",
"\n",
" rubberband.attr('width', width);\n",
" rubberband.attr('height', height);\n",
" }\n",
"\n",
" // Set the figure to an initial 600x600px, this will subsequently be updated\n",
" // upon first draw.\n",
" this._resize_canvas(600, 600);\n",
"\n",
" // Disable right mouse context menu.\n",
" $(this.rubberband_canvas).bind(\"contextmenu\",function(e){\n",
" return false;\n",
" });\n",
"\n",
" function set_focus () {\n",
" canvas.focus();\n",
" canvas_div.focus();\n",
" }\n",
"\n",
" window.setTimeout(set_focus, 100);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items) {\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) {\n",
" // put a spacer in here.\n",
" continue;\n",
" }\n",
" var button = $('<button/>');\n",
" button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' +\n",
" 'ui-button-icon-only');\n",
" button.attr('role', 'button');\n",
" button.attr('aria-disabled', 'false');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
"\n",
" var icon_img = $('<span/>');\n",
" icon_img.addClass('ui-button-icon-primary ui-icon');\n",
" icon_img.addClass(image);\n",
" icon_img.addClass('ui-corner-all');\n",
"\n",
" var tooltip_span = $('<span/>');\n",
" tooltip_span.addClass('ui-button-text');\n",
" tooltip_span.html(tooltip);\n",
"\n",
" button.append(icon_img);\n",
" button.append(tooltip_span);\n",
"\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" var fmt_picker_span = $('<span/>');\n",
"\n",
" var fmt_picker = $('<select/>');\n",
" fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content');\n",
" fmt_picker_span.append(fmt_picker);\n",
" nav_element.append(fmt_picker_span);\n",
" this.format_dropdown = fmt_picker[0];\n",
"\n",
" for (var ind in mpl.extensions) {\n",
" var fmt = mpl.extensions[ind];\n",
" var option = $(\n",
" '<option/>', {selected: fmt === mpl.default_extension}).html(fmt);\n",
" fmt_picker.append(option)\n",
" }\n",
"\n",
" // Add hover states to the ui-buttons\n",
" $( \".ui-button\" ).hover(\n",
" function() { $(this).addClass(\"ui-state-hover\");},\n",
" function() { $(this).removeClass(\"ui-state-hover\");}\n",
" );\n",
"\n",
" var status_bar = $('<span class=\"mpl-message\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"}\n",
"\n",
"mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) {\n",
" // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client,\n",
" // which will in turn request a refresh of the image.\n",
" this.send_message('resize', {'width': x_pixels, 'height': y_pixels});\n",
"}\n",
"\n",
"mpl.figure.prototype.send_message = function(type, properties) {\n",
" properties['type'] = type;\n",
" properties['figure_id'] = this.id;\n",
" this.ws.send(JSON.stringify(properties));\n",
"}\n",
"\n",
"mpl.figure.prototype.send_draw_message = function() {\n",
" if (!this.waiting) {\n",
" this.waiting = true;\n",
" this.ws.send(JSON.stringify({type: \"draw\", figure_id: this.id}));\n",
" }\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" var format_dropdown = fig.format_dropdown;\n",
" var format = format_dropdown.options[format_dropdown.selectedIndex].value;\n",
" fig.ondownload(fig, format);\n",
"}\n",
"\n",
"\n",
"mpl.figure.prototype.handle_resize = function(fig, msg) {\n",
" var size = msg['size'];\n",
" if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) {\n",
" fig._resize_canvas(size[0], size[1]);\n",
" fig.send_message(\"refresh\", {});\n",
" };\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_rubberband = function(fig, msg) {\n",
" var x0 = msg['x0'] / mpl.ratio;\n",
" var y0 = (fig.canvas.height - msg['y0']) / mpl.ratio;\n",
" var x1 = msg['x1'] / mpl.ratio;\n",
" var y1 = (fig.canvas.height - msg['y1']) / mpl.ratio;\n",
" x0 = Math.floor(x0) + 0.5;\n",
" y0 = Math.floor(y0) + 0.5;\n",
" x1 = Math.floor(x1) + 0.5;\n",
" y1 = Math.floor(y1) + 0.5;\n",
" var min_x = Math.min(x0, x1);\n",
" var min_y = Math.min(y0, y1);\n",
" var width = Math.abs(x1 - x0);\n",
" var height = Math.abs(y1 - y0);\n",
"\n",
" fig.rubberband_context.clearRect(\n",
" 0, 0, fig.canvas.width, fig.canvas.height);\n",
"\n",
" fig.rubberband_context.strokeRect(min_x, min_y, width, height);\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_figure_label = function(fig, msg) {\n",
" // Updates the figure title.\n",
" fig.header.textContent = msg['label'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_cursor = function(fig, msg) {\n",
" var cursor = msg['cursor'];\n",
" switch(cursor)\n",
" {\n",
" case 0:\n",
" cursor = 'pointer';\n",
" break;\n",
" case 1:\n",
" cursor = 'default';\n",
" break;\n",
" case 2:\n",
" cursor = 'crosshair';\n",
" break;\n",
" case 3:\n",
" cursor = 'move';\n",
" break;\n",
" }\n",
" fig.rubberband_canvas.style.cursor = cursor;\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_message = function(fig, msg) {\n",
" fig.message.textContent = msg['message'];\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_draw = function(fig, msg) {\n",
" // Request the server to send over a new figure.\n",
" fig.send_draw_message();\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_image_mode = function(fig, msg) {\n",
" fig.image_mode = msg['mode'];\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Called whenever the canvas gets updated.\n",
" this.send_message(\"ack\", {});\n",
"}\n",
"\n",
"// A function to construct a web socket function for onmessage handling.\n",
"// Called in the figure constructor.\n",
"mpl.figure.prototype._make_on_message_function = function(fig) {\n",
" return function socket_on_message(evt) {\n",
" if (evt.data instanceof Blob) {\n",
" /* FIXME: We get \"Resource interpreted as Image but\n",
" * transferred with MIME type text/plain:\" errors on\n",
" * Chrome. But how to set the MIME type? It doesn't seem\n",
" * to be part of the websocket stream */\n",
" evt.data.type = \"image/png\";\n",
"\n",
" /* Free the memory for the previous frames */\n",
" if (fig.imageObj.src) {\n",
" (window.URL || window.webkitURL).revokeObjectURL(\n",
" fig.imageObj.src);\n",
" }\n",
"\n",
" fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL(\n",
" evt.data);\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
" else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == \"data:image/png;base64\") {\n",
" fig.imageObj.src = evt.data;\n",
" fig.updated_canvas_event();\n",
" fig.waiting = false;\n",
" return;\n",
" }\n",
"\n",
" var msg = JSON.parse(evt.data);\n",
" var msg_type = msg['type'];\n",
"\n",
" // Call the \"handle_{type}\" callback, which takes\n",
" // the figure and JSON message as its only arguments.\n",
" try {\n",
" var callback = fig[\"handle_\" + msg_type];\n",
" } catch (e) {\n",
" console.log(\"No handler for the '\" + msg_type + \"' message type: \", msg);\n",
" return;\n",
" }\n",
"\n",
" if (callback) {\n",
" try {\n",
" // console.log(\"Handling '\" + msg_type + \"' message: \", msg);\n",
" callback(fig, msg);\n",
" } catch (e) {\n",
" console.log(\"Exception inside the 'handler_\" + msg_type + \"' callback:\", e, e.stack, msg);\n",
" }\n",
" }\n",
" };\n",
"}\n",
"\n",
"// from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas\n",
"mpl.findpos = function(e) {\n",
" //this section is from http://www.quirksmode.org/js/events_properties.html\n",
" var targ;\n",
" if (!e)\n",
" e = window.event;\n",
" if (e.target)\n",
" targ = e.target;\n",
" else if (e.srcElement)\n",
" targ = e.srcElement;\n",
" if (targ.nodeType == 3) // defeat Safari bug\n",
" targ = targ.parentNode;\n",
"\n",
" // jQuery normalizes the pageX and pageY\n",
" // pageX,Y are the mouse positions relative to the document\n",
" // offset() returns the position of the element relative to the document\n",
" var x = e.pageX - $(targ).offset().left;\n",
" var y = e.pageY - $(targ).offset().top;\n",
"\n",
" return {\"x\": x, \"y\": y};\n",
"};\n",
"\n",
"/*\n",
" * return a copy of an object with only non-object keys\n",
" * we need this to avoid circular references\n",
" * http://stackoverflow.com/a/24161582/3208463\n",
" */\n",
"function simpleKeys (original) {\n",
" return Object.keys(original).reduce(function (obj, key) {\n",
" if (typeof original[key] !== 'object')\n",
" obj[key] = original[key]\n",
" return obj;\n",
" }, {});\n",
"}\n",
"\n",
"mpl.figure.prototype.mouse_event = function(event, name) {\n",
" var canvas_pos = mpl.findpos(event)\n",
"\n",
" if (name === 'button_press')\n",
" {\n",
" this.canvas.focus();\n",
" this.canvas_div.focus();\n",
" }\n",
"\n",
" var x = canvas_pos.x * mpl.ratio;\n",
" var y = canvas_pos.y * mpl.ratio;\n",
"\n",
" this.send_message(name, {x: x, y: y, button: event.button,\n",
" step: event.step,\n",
" guiEvent: simpleKeys(event)});\n",
"\n",
" /* This prevents the web browser from automatically changing to\n",
" * the text insertion cursor when the button is pressed. We want\n",
" * to control all of the cursor setting manually through the\n",
" * 'cursor' event from matplotlib */\n",
" event.preventDefault();\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" // Handle any extra behaviour associated with a key event\n",
"}\n",
"\n",
"mpl.figure.prototype.key_event = function(event, name) {\n",
"\n",
" // Prevent repeat events\n",
" if (name == 'key_press')\n",
" {\n",
" if (event.which === this._key)\n",
" return;\n",
" else\n",
" this._key = event.which;\n",
" }\n",
" if (name == 'key_release')\n",
" this._key = null;\n",
"\n",
" var value = '';\n",
" if (event.ctrlKey && event.which != 17)\n",
" value += \"ctrl+\";\n",
" if (event.altKey && event.which != 18)\n",
" value += \"alt+\";\n",
" if (event.shiftKey && event.which != 16)\n",
" value += \"shift+\";\n",
"\n",
" value += 'k';\n",
" value += event.which.toString();\n",
"\n",
" this._key_event_extra(event, name);\n",
"\n",
" this.send_message(name, {key: value,\n",
" guiEvent: simpleKeys(event)});\n",
" return false;\n",
"}\n",
"\n",
"mpl.figure.prototype.toolbar_button_onclick = function(name) {\n",
" if (name == 'download') {\n",
" this.handle_save(this, null);\n",
" } else {\n",
" this.send_message(\"toolbar_button\", {name: name});\n",
" }\n",
"};\n",
"\n",
"mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) {\n",
" this.message.textContent = tooltip;\n",
"};\n",
"mpl.toolbar_items = [[\"Home\", \"Reset original view\", \"fa fa-home icon-home\", \"home\"], [\"Back\", \"Back to previous view\", \"fa fa-arrow-left icon-arrow-left\", \"back\"], [\"Forward\", \"Forward to next view\", \"fa fa-arrow-right icon-arrow-right\", \"forward\"], [\"\", \"\", \"\", \"\"], [\"Pan\", \"Pan axes with left mouse, zoom with right\", \"fa fa-arrows icon-move\", \"pan\"], [\"Zoom\", \"Zoom to rectangle\", \"fa fa-square-o icon-check-empty\", \"zoom\"], [\"\", \"\", \"\", \"\"], [\"Download\", \"Download plot\", \"fa fa-floppy-o icon-save\", \"download\"]];\n",
"\n",
"mpl.extensions = [\"eps\", \"jpeg\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\", \"tif\"];\n",
"\n",
"mpl.default_extension = \"png\";var comm_websocket_adapter = function(comm) {\n",
" // Create a \"websocket\"-like object which calls the given IPython comm\n",
" // object with the appropriate methods. Currently this is a non binary\n",
" // socket, so there is still some room for performance tuning.\n",
" var ws = {};\n",
"\n",
" ws.close = function() {\n",
" comm.close()\n",
" };\n",
" ws.send = function(m) {\n",
" //console.log('sending', m);\n",
" comm.send(m);\n",
" };\n",
" // Register the callback with on_msg.\n",
" comm.on_msg(function(msg) {\n",
" //console.log('receiving', msg['content']['data'], msg);\n",
" // Pass the mpl event to the overriden (by mpl) onmessage function.\n",
" ws.onmessage(msg['content']['data'])\n",
" });\n",
" return ws;\n",
"}\n",
"\n",
"mpl.mpl_figure_comm = function(comm, msg) {\n",
" // This is the function which gets called when the mpl process\n",
" // starts-up an IPython Comm through the \"matplotlib\" channel.\n",
"\n",
" var id = msg.content.data.id;\n",
" // Get hold of the div created by the display call when the Comm\n",
" // socket was opened in Python.\n",
" var element = $(\"#\" + id);\n",
" var ws_proxy = comm_websocket_adapter(comm)\n",
"\n",
" function ondownload(figure, format) {\n",
" window.open(figure.imageObj.src);\n",
" }\n",
"\n",
" var fig = new mpl.figure(id, ws_proxy,\n",
" ondownload,\n",
" element.get(0));\n",
"\n",
" // Call onopen now - mpl needs it, as it is assuming we've passed it a real\n",
" // web socket which is closed, not our websocket->open comm proxy.\n",
" ws_proxy.onopen();\n",
"\n",
" fig.parent_element = element.get(0);\n",
" fig.cell_info = mpl.find_output_cell(\"<div id='\" + id + \"'></div>\");\n",
" if (!fig.cell_info) {\n",
" console.error(\"Failed to find cell for figure\", id, fig);\n",
" return;\n",
" }\n",
"\n",
" var output_index = fig.cell_info[2]\n",
" var cell = fig.cell_info[0];\n",
"\n",
"};\n",
"\n",
"mpl.figure.prototype.handle_close = function(fig, msg) {\n",
" var width = fig.canvas.width/mpl.ratio\n",
" fig.root.unbind('remove')\n",
"\n",
" // Update the output cell to use the data from the current canvas.\n",
" fig.push_to_output();\n",
" var dataURL = fig.canvas.toDataURL();\n",
" // Re-enable the keyboard manager in IPython - without this line, in FF,\n",
" // the notebook keyboard shortcuts fail.\n",
" IPython.keyboard_manager.enable()\n",
" $(fig.parent_element).html('<img src=\"' + dataURL + '\" width=\"' + width + '\">');\n",
" fig.close_ws(fig, msg);\n",
"}\n",
"\n",
"mpl.figure.prototype.close_ws = function(fig, msg){\n",
" fig.send_message('closing', msg);\n",
" // fig.ws.close()\n",
"}\n",
"\n",
"mpl.figure.prototype.push_to_output = function(remove_interactive) {\n",
" // Turn the data on the canvas into data in the output cell.\n",
" var width = this.canvas.width/mpl.ratio\n",
" var dataURL = this.canvas.toDataURL();\n",
" this.cell_info[1]['text/html'] = '<img src=\"' + dataURL + '\" width=\"' + width + '\">';\n",
"}\n",
"\n",
"mpl.figure.prototype.updated_canvas_event = function() {\n",
" // Tell IPython that the notebook contents must change.\n",
" IPython.notebook.set_dirty(true);\n",
" this.send_message(\"ack\", {});\n",
" var fig = this;\n",
" // Wait a second, then push the new image to the DOM so\n",
" // that it is saved nicely (might be nice to debounce this).\n",
" setTimeout(function () { fig.push_to_output() }, 1000);\n",
"}\n",
"\n",
"mpl.figure.prototype._init_toolbar = function() {\n",
" var fig = this;\n",
"\n",
" var nav_element = $('<div/>')\n",
" nav_element.attr('style', 'width: 100%');\n",
" this.root.append(nav_element);\n",
"\n",
" // Define a callback function for later on.\n",
" function toolbar_event(event) {\n",
" return fig.toolbar_button_onclick(event['data']);\n",
" }\n",
" function toolbar_mouse_event(event) {\n",
" return fig.toolbar_button_onmouseover(event['data']);\n",
" }\n",
"\n",
" for(var toolbar_ind in mpl.toolbar_items){\n",
" var name = mpl.toolbar_items[toolbar_ind][0];\n",
" var tooltip = mpl.toolbar_items[toolbar_ind][1];\n",
" var image = mpl.toolbar_items[toolbar_ind][2];\n",
" var method_name = mpl.toolbar_items[toolbar_ind][3];\n",
"\n",
" if (!name) { continue; };\n",
"\n",
" var button = $('<button class=\"btn btn-default\" href=\"#\" title=\"' + name + '\"><i class=\"fa ' + image + ' fa-lg\"></i></button>');\n",
" button.click(method_name, toolbar_event);\n",
" button.mouseover(tooltip, toolbar_mouse_event);\n",
" nav_element.append(button);\n",
" }\n",
"\n",
" // Add the status bar.\n",
" var status_bar = $('<span class=\"mpl-message\" style=\"text-align:right; float: right;\"/>');\n",
" nav_element.append(status_bar);\n",
" this.message = status_bar[0];\n",
"\n",
" // Add the close button to the window.\n",
" var buttongrp = $('<div class=\"btn-group inline pull-right\"></div>');\n",
" var button = $('<button class=\"btn btn-mini btn-primary\" href=\"#\" title=\"Stop Interaction\"><i class=\"fa fa-power-off icon-remove icon-large\"></i></button>');\n",
" button.click(function (evt) { fig.handle_close(fig, {}); } );\n",
" button.mouseover('Stop Interaction', toolbar_mouse_event);\n",
" buttongrp.append(button);\n",
" var titlebar = this.root.find($('.ui-dialog-titlebar'));\n",
" titlebar.prepend(buttongrp);\n",
"}\n",
"\n",
"mpl.figure.prototype._root_extra_style = function(el){\n",
" var fig = this\n",
" el.on(\"remove\", function(){\n",
"\tfig.close_ws(fig, {});\n",
" });\n",
"}\n",
"\n",
"mpl.figure.prototype._canvas_extra_style = function(el){\n",
" // this is important to make the div 'focusable\n",
" el.attr('tabindex', 0)\n",
" // reach out to IPython and tell the keyboard manager to turn it's self\n",
" // off when our div gets focus\n",
"\n",
" // location in version 3\n",
" if (IPython.notebook.keyboard_manager) {\n",
" IPython.notebook.keyboard_manager.register_events(el);\n",
" }\n",
" else {\n",
" // location in version 2\n",
" IPython.keyboard_manager.register_events(el);\n",
" }\n",
"\n",
"}\n",
"\n",
"mpl.figure.prototype._key_event_extra = function(event, name) {\n",
" var manager = IPython.notebook.keyboard_manager;\n",
" if (!manager)\n",
" manager = IPython.keyboard_manager;\n",
"\n",
" // Check for shift+enter\n",
" if (event.shiftKey && event.which == 13) {\n",
" this.canvas_div.blur();\n",
" event.shiftKey = false;\n",
" // Send a \"J\" for go to next cell\n",
" event.which = 74;\n",
" event.keyCode = 74;\n",
" manager.command_mode();\n",
" manager.handle_keydown(event);\n",
" }\n",
"}\n",
"\n",
"mpl.figure.prototype.handle_save = function(fig, msg) {\n",
" fig.ondownload(fig, null);\n",
"}\n",
"\n",
"\n",
"mpl.find_output_cell = function(html_output) {\n",
" // Return the cell and output element which can be found *uniquely* in the notebook.\n",
" // Note - this is a bit hacky, but it is done because the \"notebook_saving.Notebook\"\n",
" // IPython event is triggered only after the cells have been serialised, which for\n",
" // our purposes (turning an active figure into a static one), is too late.\n",
" var cells = IPython.notebook.get_cells();\n",
" var ncells = cells.length;\n",
" for (var i=0; i<ncells; i++) {\n",
" var cell = cells[i];\n",
" if (cell.cell_type === 'code'){\n",
" for (var j=0; j<cell.output_area.outputs.length; j++) {\n",
" var data = cell.output_area.outputs[j];\n",
" if (data.data) {\n",
" // IPython >= 3 moved mimebundle to data attribute of output\n",
" data = data.data;\n",
" }\n",
" if (data['text/html'] == html_output) {\n",
" return [cell, data, j];\n",
" }\n",
" }\n",
" }\n",
" }\n",
"}\n",
"\n",
"// Register the function which deals with the matplotlib target/channel.\n",
"// The kernel may be null if the page has been refreshed.\n",
"if (IPython.notebook.kernel != null) {\n",
" IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm);\n",
"}\n"
],
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"text/html": [
"<img src=\"\" width=\"864\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"%matplotlib notebook\n",
"import matplotlib.pyplot as plt\n",
"from matplotlib.animation import FuncAnimation\n",
"\n",
"fig, (ax1) = plt.subplots(1, figsize=(12, 6))\n",
"ax1.scatter(x, y, s=8)\n",
"\n",
"w1, b1 = get_param_values()\n",
"x1 = np.array([0., 1.])\n",
"y1 = x1 * w1 + b1\n",
"fit, = ax1.plot(x1, y1, 'r', label='Predicted'.format(w1, b1))\n",
"ax1.plot(x1, x1 * m + c, 'g', label='Real')\n",
"ax1.legend()\n",
"ax1.set_title('Linear Regression')\n",
"\n",
"def init():\n",
" ax1.set_ylim(0, 6)\n",
" return fit,\n",
"\n",
"def animate(i):\n",
" loss = run_epoch(i)\n",
" w1, b1 = get_param_values()\n",
" y1 = x1 * w1 + b1\n",
" fit.set_ydata(y1)\n",
"\n",
"epochs = np.arange(1, 250)\n",
"ani = FuncAnimation(fig, animate, epochs, init_func=init, interval=100, blit=True, repeat=False)\n",
"plt.show()"
]
},
{
"cell_type": "code",
"execution_count": 25,
"metadata": {},
"outputs": [
{
"data": {
"text/plain": [
"(1.9343229532241821, 2.986394166946411)"
]
},
"execution_count": 25,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"get_param_values()"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {},
"outputs": [],
"source": []
}
],
"metadata": {
"kernelspec": {
"display_name": "Python 3",
"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.6.4"
}
},
"nbformat": 4,
"nbformat_minor": 2
}
@Karlheinzniebuhr
Copy link

Getting lots of this error: TypeError: unsupported operand type(s) for *: 'numpy.ndarray' and 'Tensor'

@Eng-Hercules
Copy link

No animation is generated by the animation function. Jupyter cannot recognize %matplotlib notebook

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment