Skip to content

Instantly share code, notes, and snippets.

@antosha417
Created June 6, 2018 11:16
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save antosha417/af26abc5d8f470c122edd9267d17693f to your computer and use it in GitHub Desktop.
Save antosha417/af26abc5d8f470c122edd9267d17693f to your computer and use it in GitHub Desktop.
This file has been truncated, but you can view the full file.
{
"cells": [
{
"cell_type": "markdown",
"metadata": {
"nbpresent": {
"id": "196b8a50-3d29-45c3-82b9-f4a09b49491d"
},
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"# Семинар 12\n",
"\n",
"# Введение в численные методы оптимизации (Ю. Е. Нестеров Введение в выпуклую оптимизацию, гл. 1 $\\S$ 1.1)\n",
"\n",
" 1. Обзор материала весеннего семестра\n",
" 2. Постановка задачи\n",
" 3. Общая схема решения\n",
" 4. Сравнение методов оптимизации\n",
" 5. Методы одномерной минимизации\n"
]
},
{
"cell_type": "markdown",
"metadata": {
"nbpresent": {
"id": "2a573842-172b-4931-b0dd-9c9d3c47a450"
},
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Обзор материала весеннего семестра\n",
"\n",
"Также на [странице курса](https://github.com/amkatrutsa/MIPT-Opt#Весенний-семестр).\n",
"\n",
"1. Методы решения задач **безусловной** оптимизации\n",
" - Одномерная минимизация (**уже сегодня!**)\n",
" - Градиентный спуск\n",
" - Метод Ньютона\n",
" - Квазиньютоновские методы\n",
" - Метод сопряжённых градиентов \n",
" - Опционально:\n",
" - Решение задачи наименьших квадратов\n",
" - Оптимальные методы и нижние оценки\n",
"2. Методы решения задач **условной** оптимизации\n",
" - Линейное программирование\n",
" - Методы проекции градиента и условного градиента\n",
" - Методы штрафных и барьерных функций\n",
" - Метод модифицированой функции Лагранжа"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Организация работы в семестре\n",
"\n",
"1. Семинар и лекция раз в неделю\n",
"2. Два задания в течение семестра\n",
"3. Midterm в середине семестра\n",
"4. Итоговая контрольная в конце семестра\n",
"5. Экзамен в конце семестра (схема выставлении оценки аналогична осеннему семестру)\n",
"6. Миниконтрольные в начале каждого семинара\n",
"7. Домашнее задание почти каждую неделю: $\\TeX$ или Jupyter Notebook"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Постановка задачи\n",
"\n",
"\\begin{equation}\n",
"\\begin{split}\n",
"& \\min_{x \\in S} f_0(x)\\\\\n",
"\\text{s.t. } & f_j(x) = 0, \\; j = 1,\\ldots,m\\\\\n",
"& g_k(x) \\leq 0, \\; k = 1,\\ldots,p\n",
"\\end{split}\n",
"\\end{equation}\n",
"где $S \\subseteq \\mathbb{R}^n$, $f_j: S \\rightarrow \\mathbb{R}, \\; j = 0,\\ldots,m$, $g_k: S \\rightarrow \\mathbb{R}, \\; k=1,\\ldots,p$\n",
"\n",
"Все функции как минимум непрерывны. \n",
"\n",
"Важный факт</span>: задачи **нелинейной** оптимизации \n",
"\n",
"в их самой общей форме являются **численно неразрешимыми**!"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Аналитические результаты\n",
"- Необходимое условие первого порядка: \n",
"\n",
"если $x^*$ точка локального минимума дифференцируемой функции $f(x)$, тогда \n",
"$$\n",
"f'(x^*) = 0\n",
"$$\n",
"- Необходимое условие второго порядка \n",
"\n",
"если $x^*$ точка локального минимума дважды дифференцируемой функции $f(x)$, тогда \n",
"\n",
"$$\n",
"f'(x^*) = 0 \\quad \\text{и} \\quad f''(x^*) \\succeq 0\n",
"$$\n",
"- Достаточное условие: \n",
"\n",
"пусть $f(x)$ дважды дифференцируемая функция, и пусть точка $x^*$ удовлетворяет условиям\n",
"\n",
"$$\n",
"f'(x^*) = 0 \\quad f''(x^*) \\succ 0,\n",
"$$\n",
"тогда $x^*$ является точкой строго локального минимума функции $f(x)$.\n",
"\n",
"**Замечание**: убедитесь, что Вы понимаете, как доказывать эти\n",
"\n",
"результаты!"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Особенности численного решения\n",
"\n",
"1. Точно решить задачу принципиально невозможно из-за погрешности машинной арифметики\n",
"2. Необходимо задать критерий обнаружения решения\n",
"3. Необходимо определить, какую информацию о задаче использовать"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Общая итеративная схема\n",
"\n",
"Дано: начальное приближение $x$, требуемая точность $\\varepsilon$.\n",
"\n",
"```python\n",
"def GeneralScheme(x, epsilon):\n",
" \n",
" while StopCriterion(x) > epsilon:\n",
" \n",
" OracleResponse = RequestOracle(x)\n",
" \n",
" UpdateInformation(I, x, OracleResponse)\n",
" \n",
" x = NextPoint(I, x)\n",
" \n",
" return x\n",
"```"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Вопросы\n",
"1. Какие критерии остановки могут быть?\n",
"2. Что такое оракул и зачем он нужен?\n",
"3. Что такое информационная модель?\n",
"4. Как вычисляется новая точка?"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"#### Критерии остановки\n",
"1. Сходимость по аргументу: \n",
"$$\n",
"\\| x_k - x^* \\|_2 < \\varepsilon\n",
"$$ \n",
"2. Сходимость по функции: \n",
"$$\n",
"\\| f_k - f^* \\|_2 < \\varepsilon\n",
"$$ \n",
"3. Выполнение необходимого условия \n",
"$$\n",
"\\| f'(x_k) \\|_2 < \\varepsilon\n",
"$$\n",
"\n",
"Но ведь $x^*$ неизвестна!\n",
"\n",
"Тогда\n",
"$$\n",
"\\|x_{k+1} - x_k \\| = \\|x_{k+1} - x_k + x^* - x^* \\| \\leq \\|x_{k+1} - x^* \\| + \\| x_k - x^* \\| \\leq 2\\varepsilon\n",
"$$\n",
"\n",
"Аналогично для сходимости по функции...\n",
"\n",
"**Замечание**: лучше использовать относительные изменения этих величин! Например $\\dfrac{\\|x_{k+1} - x_k \\|_2}{\\| x_k \\|_2}$\n"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"#### Что такое оракул?\n",
"**Определение**: оракулом называют некоторое абстрактное \n",
"\n",
"устройство, которое отвечает на последовательные вопросы \n",
"\n",
"метода\n",
"\n",
"Аналогия из ООП: \n",
"\n",
"- оракул - это виртуальный метод базового класса\n",
"- каждая задача - производный класс\n",
"- оракул определяется для каждой задачи отдельно согласно общему определению в базовом классе\n",
"\n",
"**Концепция чёрного ящика**\n",
"1. Единственной информацией, получаемой в ходе работы итеративного метода, являются ответы оракула\n",
"2. Ответы оракула являются *локальными*"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"#### Информация о задаче\n",
"1. Каждый ответ оракула даёт локальную информацию о поведении функции в точке\n",
"2. Агрегируя все полученные ответы оракула, обновляем информацию о глобальном виде целевой функции:\n",
" - кривизна\n",
" - направление убывания\n",
" - etc"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"#### Вычисление следующей точки\n",
"\n",
"$$\n",
"x_{k+1} = x_{k} + \\alpha_k h_k\n",
"$$\n",
"\n",
"- **Линейный поиск**: фиксируется направление $h_k$ и производится поиск по этому направлению \"оптимального\" значения $\\alpha_k$\n",
"- **Метод доверительных областей**: фиксируется допустимый размер *области* по некоторой норме $\\| \\cdot \\| \\leq \\alpha$ и *модель* целевой функции, которая хорошо её аппроксимирует в выбранной области. \n",
" Далее производится поиск направления $h_k$, минимизирующего модель целевой функции и не выводящего точку $x_k + h_k$ за пределы доверительной области\n",
"\n",
"Вопросы:\n",
"1. Как выбрать $\\alpha_k$?\n",
"2. Как выбрать $h_k$?\n",
"3. Как выбрать модель?\n",
"4. Как выбрать область?\n",
"5. Как выбрать размер области? \n",
"\n",
"<span style=\"color:red\">\n",
" В курсе рассматривается только линейный поиск!</span> \n",
" \n",
"Однако несколько раз копцепция метода доверительных областей \n",
"\n",
"будет использована."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Как сравнивать методы оптимизации?\n",
"Для заданного класса задач сравнивают следующие величины:\n",
"1. Сложность\n",
" - аналитическая: число обращений к оракулу для решения задачи с точностью $\\varepsilon$\n",
" - арифметическая: общее число всех вычислений, необходимых для решения задачи с точностью $\\varepsilon$\n",
"2. Скорость сходимости\n",
"3. Эксперименты"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Скорости сходимости \n",
"1. Сублинейная\n",
"$$\n",
"\\| x_{k+1} - x^* \\|_2 \\leq C k^{\\alpha},\n",
"$$\n",
"где $\\alpha < 0$ и $ 0 < C < \\infty$\n",
"2. Линейная (геометрическая прогрессия)\n",
"$$\n",
"\\| x_{k+1} - x^* \\|_2 \\leq Cq^k, \n",
"$$\n",
"где $q \\in (0, 1)$ и $ 0 < C < \\infty$\n",
"\n",
"3. Сверхлинейная \n",
"$$\n",
"\\| x_{k+1} - x^* \\|_2 \\leq Cq^{k^2}, \n",
"$$\n",
"где $q \\in (0, 1)$ и $ 0 < C < \\infty$\n",
"4. Квадратичная\n",
"$$\n",
"\\| x_{k+1} - x^* \\|_2 \\leq C\\| x_k - x^* \\|^2_2, \\qquad \\text{или} \\qquad \\| x_{k+1} - x^* \\|_2 \\leq C q^{2^k}\n",
"$$\n",
"где $q \\in (0, 1)$ и $ 0 < C < \\infty$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Оптимальные методы: can we do better?\n",
"- Доказывают нижние оценки скоростей сходимости для класса задач и методов фиксированного порядка\n",
"- Предлагают методы, на которых эти нижние оценки достигаются $\\Rightarrow$ доказана оптимальность \n",
"- Ниже про значение теорем сходимости\n",
"\n",
"Оптимальным методам и нижним оценкам будет, возможно, \n",
"\n",
"посвящён отдельный семинар."
]
},
{
"cell_type": "code",
"execution_count": 1,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"outputs": [
{
"data": {
"image/png": "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zIABAyp1gbx9+3Zmz57NrFmzCtoeeOAB+vfvz9y5cwuKgOTkZCZNmkRERAQ7duygRYsWBf23bt3KHXfcwfTp09m4cWOR43/77be0adOmSNulS5cYMGAAzz33HJMnTy5yLIBdu3bx5z//mWeffbbCn6uwUaNG8cUXXzB//nwSE23DhX7/+9/TsmVLPvzwQ77//nvi4+MpqxieN28ep06dcvicnTt3ZoiFt5IPHjwIQFRUVInb27VrR1JSEsnJyQ4XAceOHWPMmDFF2lq3bs3KlSvp3bt35QIrpVQt4d2nD4063UDG4iX4DRmCi4eH1ZEaJGfHBOyoriBKFdalSxfWrVvH9OnTWbduXcGjLYGBgfTq1Yvx48czcODAKjmXn58ff/3rX4u0RUdHM3r0aFavXs3GjRuJi4ur8PHDw8N58skni7TdeeedhIWFsW/fvoK2xYsXc/nyZebPn1/sov22225j0KBBbNq0iezsbHx8fAq2XV0AAHh4eDB16lS2bt3Kli1bGDt2bJHtzZo1K1KUVMYnn3zCxx9/TM+ePYs8Fx8cHExERARbt27F09Oz2O/x1ebNm0daWprD542Li7O0d6h08QAAIABJREFUCMif9re0AeH57Y4WNuPGjaNnz5507NgRHx8fDh8+zMKFC1m6dCkDBgzg888/p1OnTlUTXimlLGSMISQxkZ8eephTb75JYD16VLQu0WHZdUhVfMNel4wYMYKhQ4eybds2du3axVdffcWuXbt4++23efvttxk7diyrVq3CGFOp89x4441FLqrz9enTh9WrV/PVV19Vqgjo3LlziY9ztGrVis8//7zg5/xf79ixgy+++KJY/+PHj5Obm0tycjJdu3YtaE9PT+f5559ny5YtpKenk5OTU2S/X375pdixOnXqhKenZ4U/U2Fr164FYMaMGcX+WzRq1AiASZMm0apVqzKPk5qaWiV5ShMREeFUkTF69OgSx1U4Kn9VTEf/fF5dlF1//fUsWbIEb29vXnzxRWbPnl3sLpBSStVVXt270zi6K5lLXsN/2DBc7P9eqJpTqSLA2P51aw/krxtwAjgouia0qiLu7u7ExsYSGxsL2KYOfeuttxg/fjxr1qxh6NChlf42uFmzZiW258/NXtmF3vz9/Utsd3NzI6/QgimZmZkARR5LKsnZs2cLfn348GG6devGyZMn6dmzJ7Gx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dWndujVjx44l3MBw+avvrF+/nqFDh9KyZUsOO+ww3nnnnTLFMHHiRNq1awfAvHnzClRHmTt3bpHjFeeDDz5g6NChtG7dmsTERJKTk+nXrx/PP/98seXL4sCBA0ybNo3U1FTq1q1Lq1atGDt2LD/99BONGjWiS5cuFT5GWWzcuJHFixeTkpLC9ddfX2Dd3XffTf369XnmmWfYs2dPtcYlIiI1U7Orr+awevX4fsZjQYdSbnoTIFFpzpw5HDhwgMsuu4xf/OIXYctdffXVTJo0if/85z+89957nHrqqeU+5ujRo+nSpQtnnHEGycnJZGZm8vrrr3PllVfyxRdfMGnSpALlb7rpJqZNm0ZycjLXXnstCQkJvPLKK7z//vvs37+fOnXqFHucDRs2cMopp9CpUycuv/xy9u7dS8OGDcsUQ58+fdi1axdTp04lNTWViy+++ND+u3fvXuq5PvXUU4wePZq4uDguvPBCOnbsyPbt21m9ejUzZ87ksssuK+/XyP79+xk4cCCLFy+mR48ejBs3jh07djB79mw2btxIdnZ2tVffevvttwHo168fhx1W8NlHUlISvXr1YvHixbz33nucddZZEe1z3759zJ8/n4yMDOrXr8/xxx/PGWecQVxcXKXHLyIi0SW+SROaDLuKzMdnkbPuOurWwDfvSgJqkK333ce+z9cFHUaJEo/rTKs//rHC+3n33XcBOPvss0ssFx8fT58+ffjHP/7B8uXLK5QErFmzhg4dOhRYtn//fvr3788DDzzAqFGjaN26NQArV65k2rRpdOjQgQ8++ICmTZsCcO+999K3b1++++472rZtG/bc7rjjDu67775yx9CnTx9SUlKYOnUq3bt3L1O1lLVr1zJmzBgaNmxIWlpakbcs33zzTcT7Ks7YsWNZvHgxDz30ELfddtuh5cOGDaNPnz4AnHjiiSXu49FHH2XXrl0RH7N79+4FEqHCvvjiCwA6depU7PqOHTuyePFi1q9fH3ESsHXrVq688soCy9q1a8ecOXM488wzI4xcRERqqmbDh5M1/+98P206R8+seW8ElARUMTMbCAw8pobWFwvKd999B8DRRx9datlQmYrevBa++QaoU6cO119/PW+//TZLlizhqquuArw3FQB33nnnoQQAoG7dutx///307ds37HGOOOIIJkyYUOEYyuvxxx/nwIED/OlPfyq2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\n",
"text/plain": [
"<matplotlib.figure.Figure at 0x10f2c5fd0>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"%matplotlib inline\n",
"import matplotlib.pyplot as plt\n",
"import numpy as np\n",
"C = 10\n",
"alpha = -0.5\n",
"q = 0.9\n",
"num_iter = 7\n",
"sublinear = np.array([C * k**alpha for k in range(1, num_iter + 1)])\n",
"linear = np.array([C * q**k for k in range(1, num_iter + 1)])\n",
"superlinear = np.array([C * q**(k**2) for k in range(1, num_iter + 1)])\n",
"quadratic = np.array([C * q**(2**k) for k in range(1, num_iter + 1)])\n",
"plt.figure(figsize=(12,8))\n",
"plt.semilogy(np.arange(1, num_iter+1), sublinear, \n",
" label=r\"Sublinear, $\\alpha = -0.5$\")\n",
"plt.semilogy(np.arange(1, num_iter+1), superlinear, \n",
" label=r\"Superlinear, $q = 0.5$\")\n",
"plt.semilogy(np.arange(1, num_iter+1), linear, \n",
" label=r\"Linear, $q = 0.5$\")\n",
"plt.semilogy(np.arange(1, num_iter+1), quadratic, \n",
" label=r\"Quadratic, $q = 0.5$\")\n",
"plt.xlabel(\"Number of iteration, $k$\", fontsize=24)\n",
"plt.ylabel(\"Error rate upper bound\", fontsize=24)\n",
"plt.legend(loc=\"best\", fontsize=20)\n",
"plt.xticks(fontsize = 20)\n",
"_ = plt.yticks(fontsize = 20)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Значение теорем сходимости (Б.Т. Поляк Введение в оптимизацию, гл. 1, $\\S$ 6)\n",
"1. Что дают теоремы сходимости\n",
" - класс задач, для которых можно рассчитывать на применимость метода (важно не завышать условия!)\n",
" - выпуклость\n",
" - гладкость\n",
" - качественное поведение метода\n",
" - существенно ли начальное приближение\n",
" - по какому функционалу есть сходимость\n",
" - оценку скорости сходимости\n",
" - теоретическая оценка поведения метода без проведения экспериментов\n",
" - определение факторов, которые влияют на сходимость (обусловленность, размерность, etc)\n",
" - иногда заранее можно выбрать число итераций для достижения заданной точности \n",
"\n",
"2. Что **НЕ** дают теоремы сходимости\n",
" - сходимость метода **ничего не говорит** о целесообразности его применения\n",
" - оценки сходимости зависят от неизвестных констант - неконструктивный характер\n",
" - учёт ошибок округления и точности решения вспомогательных задач\n",
" \n",
"**Мораль**: нужно проявлять разумную осторожность \n",
"\n",
"и здравый смысл!"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Классификация задач\n",
"1. Безусловная оптимизация\n",
" - целевая функция липшицева\n",
" - градиент целевой функции липшицев\n",
"2. Условная оптимизация\n",
" - многогранник\n",
" - множество простой структуры\n",
" - общего вида"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Классификация методов\n",
"1. Методы нулевого порядка: оракул возвращает только значение функции $f(x)$\n",
"\n",
"2. Методы первого порядка: оракул возвращает значение функции $f(x)$ и её градиент $f'(x)$\n",
"\n",
"3. Методы второго порядка: оракул возвращает значение функции $f(x)$, её градиент $f'(x)$ и гессиан $f''(x)$.\n",
"\n",
"**Вопрос**: существуют ли методы более высокого порядка?\n",
"\n",
"1. Одношаговые методы \n",
"$$\n",
"x_{k+1} = \\Phi(x_k)\n",
"$$\n",
"2. Многошаговые методы\n",
"$$\n",
"x_{k+1} = \\Phi(x_k, x_{k-1}, ...)\n",
"$$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Одномерная минимизация\n",
"**Определение**. Функция $f(x)$ называется унимодальной на $[a, b]$, если существует такая точка $x^* \\in [a, b]$, что \n",
"- $f(x_1) > f(x_2)$ для любых $a \\leq x_1 < x_2 < x^*$, \n",
"\n",
"и \n",
"- $f(x_1) < f(x_2)$ для любых $x^* < x_1 < x_2 \\leq b$.\n",
"\n",
"**Вопрос**: какая геометрия унимодальных функций?"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Метод дихотомии\n",
"\n",
"Идея из информатики первого семестра: делим отрезок $[a,b]$ на две равные части пока не найдём минимум унимодальной функции.\n",
"\n",
"Пусть $N$ - число вычислений функции $f$, тогда можно выполнить $K = \\frac{N - 1}{2}$ итераций и справедлива следующая оценка\n",
"$$\n",
"|x_{K+1} - x^*| \\leq \\frac{b_{K+1} - a_{K+1}}{2} = \\left( \\frac{1}{2} \\right)^{\\frac{N-1}{2}} (b - a) \\approx 0.5^{K} (b - a) \n",
"$$"
]
},
{
"cell_type": "code",
"execution_count": 2,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"def binary_search(f, a, b, epsilon, callback=None):\n",
" c = (a + b) / 2.0\n",
" while abs(b - a) > epsilon:\n",
"# Check left subsegment\n",
" y = (a + c) / 2.0\n",
" if f(y) <= f(c):\n",
" b = c\n",
" c = y\n",
" else:\n",
"# Check right subsegment\n",
" z = (b + c) / 2.0\n",
" if f(c) <= f(z):\n",
" a = y\n",
" b = z\n",
" else:\n",
" a = c\n",
" c = z\n",
" if callback is not None:\n",
" callback(a, b)\n",
" return c"
]
},
{
"cell_type": "code",
"execution_count": 3,
"metadata": {},
"outputs": [],
"source": [
"def my_callback(a, b, left_bound, right_bound, approximation):\n",
" left_bound.append(a)\n",
" right_bound.append(b)\n",
" approximation.append((a + b) / 2.0)"
]
},
{
"cell_type": "code",
"execution_count": 4,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"9.313225746154785e-10\n"
]
},
{
"data": {
"text/plain": [
"Text(0.5,1,'Objective function')"
]
},
"execution_count": 4,
"metadata": {},
"output_type": "execute_result"
},
{
"data": {
"image/png": "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\n",
"text/plain": [
"<matplotlib.figure.Figure at 0x11a918588>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"# %matplotlib inline\n",
"import numpy as np\n",
"# import matplotlib.pyplot as plt\n",
"\n",
"left_boud_bs = []\n",
"right_bound_bs = []\n",
"approximation_bs = []\n",
"\n",
"callback_bs = lambda a, b: my_callback(a, b, \n",
" left_boud_bs, right_bound_bs, approximation_bs)\n",
"\n",
"# Target unimodal function on given segment\n",
"f = lambda x: (x - 2) * x * (x + 2)**2 # np.power(x+2, 2)\n",
"# f = lambda x: -np.sin(x)\n",
"x_true = -2\n",
"# x_true = np.pi / 2.0\n",
"a = -3\n",
"b = -1.5\n",
"epsilon = 1e-8\n",
"x_opt = binary_search(f, a, b, epsilon, callback_bs)\n",
"print(np.abs(x_opt - x_true))\n",
"plt.figure(figsize=(10,6))\n",
"plt.plot(np.linspace(a,b), f(np.linspace(a,b)))\n",
"plt.title(\"Objective function\", fontsize=18)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Метод золотого сечения\n",
"Идея: делить отрезок $[a,b]$ не на две равные насти, а в пропорции \"золотого сечения\".\n",
"\n",
"Оценим скорость сходимости аналогично методу дихотомии:\n",
"$$\n",
"|x_{K+1} - x^*| \\leq b_{K+1} - a_{K+1} = \\left( \\frac{1}{\\tau} \\right)^{N-1} (b - a) \\approx 0.618^K(b-a),\n",
"$$\n",
"где $\\tau = \\frac{\\sqrt{5} + 1}{2}$.\n",
"\n",
"- Константа геометрической прогрессии **больше**, чем у метода дихотомии\n",
"- Количество вызовов функции **меньше**, чем у метода дихотомии"
]
},
{
"cell_type": "code",
"execution_count": 5,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"def golden_search(f, a, b, tol=1e-5, callback=None):\n",
" tau = (np.sqrt(5) + 1) / 2.0\n",
" y = a + (b - a) / tau**2\n",
" z = a + (b - a) / tau\n",
" while b - a > tol:\n",
" if f(y) <= f(z):\n",
" b = z\n",
" z = y\n",
" y = a + (b - a) / tau**2\n",
" else:\n",
" a = y\n",
" y = z\n",
" z = a + (b - a) / tau\n",
" if callback is not None:\n",
" callback(a, b)\n",
" return (a + b) / 2.0"
]
},
{
"cell_type": "code",
"execution_count": 6,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"6.93889390875399e-18\n",
"9.549014390504221e-18\n",
"9.313225746154785e-10\n"
]
}
],
"source": [
"left_boud_gs = []\n",
"right_bound_gs = []\n",
"approximation_gs = []\n",
"\n",
"cb_gs = lambda a, b: my_callback(a, b, left_boud_gs, right_bound_gs, approximation_gs)\n",
"x_gs = golden_search(f, a, b, epsilon, cb_gs)\n",
"\n",
"print(f(x_opt))\n",
"print(f(x_gs))\n",
"print(np.abs(x_opt - x_true))"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Сравнение методов одномерной минимизации"
]
},
{
"cell_type": "code",
"execution_count": 7,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"image/png": "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\n",
"text/plain": [
"<matplotlib.figure.Figure at 0x11b8c9630>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"plt.figure(figsize=(10,6))\n",
"plt.semilogy(np.arange(1, len(approximation_bs) + 1), np.abs(x_true - np.array(approximation_bs, dtype=np.float64)), label=\"Binary search\")\n",
"plt.semilogy(np.arange(1, len(approximation_gs) + 1), np.abs(x_true - np.array(approximation_gs, dtype=np.float64)), label=\"Golden search\")\n",
"plt.xlabel(r\"Number of iteration, $k$\", fontsize=20)\n",
"plt.ylabel(\"Error rate upper bound\", fontsize=18)\n",
"plt.legend(loc=\"best\", fontsize=20)\n",
"plt.xticks(fontsize = 20)\n",
"_ = plt.yticks(fontsize = 20)"
]
},
{
"cell_type": "code",
"execution_count": 8,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"22.6 µs ± 1.13 µs per loop (mean ± std. dev. of 7 runs, 10000 loops each)\n",
"84.1 µs ± 6.64 µs per loop (mean ± std. dev. of 7 runs, 10000 loops each)\n"
]
}
],
"source": [
"%timeit binary_search(f, a, b, epsilon)\n",
"%timeit golden_search(f, a, b, epsilon)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Пример иного поведения методов\n",
"\n",
"$$\n",
"f(x) = \\sin(\\sin(\\sin(\\sqrt{x}))), \\; x \\in [2, 60]\n",
"$$"
]
},
{
"cell_type": "code",
"execution_count": 9,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"f = lambda x: np.sin(np.sin(np.sin(np.sqrt(x))))\n",
"x_true = (3 * np.pi / 2)**2\n",
"a = 2\n",
"b = 60\n",
"epsilon = 1e-8\n",
"# plt.plot(np.linspace(a,b), f(np.linspace(a,b)))"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Сравнение скорости сходимости и времени работы методов"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Метод дихотомии"
]
},
{
"cell_type": "code",
"execution_count": 10,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"2.1968899233115735e-07\n"
]
}
],
"source": [
"left_boud_bs = []\n",
"right_bound_bs = []\n",
"approximation_bs = []\n",
"\n",
"callback_bs = lambda a, b: my_callback(a, b, \n",
" left_boud_bs, right_bound_bs, approximation_bs)\n",
"\n",
"x_opt = binary_search(f, a, b, epsilon, callback_bs)\n",
"print(np.abs(x_opt - x_true))"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Метод золотого сечения"
]
},
{
"cell_type": "code",
"execution_count": 11,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"2.1968899233115735e-07\n"
]
}
],
"source": [
"left_boud_gs = []\n",
"right_bound_gs = []\n",
"approximation_gs = []\n",
"\n",
"cb_gs = lambda a, b: my_callback(a, b, left_boud_gs, right_bound_gs, approximation_gs)\n",
"x_gs = golden_search(f, a, b, epsilon, cb_gs)\n",
"\n",
"print(np.abs(x_opt - x_true))"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Сходимость"
]
},
{
"cell_type": "code",
"execution_count": 12,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"image/png": "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\n",
"text/plain": [
"<matplotlib.figure.Figure at 0x11b18f400>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"plt.figure(figsize=(8,6))\n",
"plt.semilogy(np.abs(x_true - np.array(approximation_bs, dtype=np.float64)), label=\"Binary\")\n",
"plt.semilogy(np.abs(x_true - np.array(approximation_gs, dtype=np.float64)), label=\"Golden\")\n",
"plt.legend(fontsize=16)\n",
"plt.xticks(fontsize=20)\n",
"_ = plt.yticks(fontsize=20)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Время работы"
]
},
{
"cell_type": "code",
"execution_count": 13,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"455 µs ± 35.8 µs per loop (mean ± std. dev. of 7 runs, 1000 loops each)\n",
"416 µs ± 26.2 µs per loop (mean ± std. dev. of 7 runs, 1000 loops each)\n"
]
}
],
"source": [
"%timeit binary_search(f, a, b, epsilon)\n",
"%timeit golden_search(f, a, b, epsilon)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Резюме\n",
"1. Введение в численные методы оптимизации\n",
"2. Общая схема работы метода\n",
"3. Способы сравнивнения методов оптимизации\n",
"4. Зоопарк задач и методов\n",
"5. Одномерная минимизация"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"# Семинар 13\n",
"\n",
"# Методы спуска (Descent methods)\n",
"# Пять способов получить градиентный спуск"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## На прошлом семинаре...\n",
"1. Введение в численные методы оптимизации\n",
"2. Общая схема работы метода\n",
"3. Как сравнивать методы оптимизации?\n",
"4. Зоопарк задач и методов\n",
"5. Одномерная минимизация"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Что такое методы спуска?\n",
"\n",
"Последовательность $x_k$ генерируется по правилу\n",
"$$\n",
"x_{k+1} = x_k + \\alpha_k h_k\n",
"$$\n",
"так что\n",
"$$\n",
"f(x_{k+1}) < f(x_k)\n",
"$$\n",
"Направление $h_k$ называется *направлением убывания*.\n",
"\n",
"**Замечание**: существуют методы, которые не требуют монотонного убывания функции от итерации к итерации."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"```python\n",
"def DescentMethod(f, x0, epsilon, **kwargs):\n",
" x = x0\n",
" while StopCriterion(x, f, **kwargs) > epsilon:\n",
" h = ComputeDescentDirection(x, f, **kwargs)\n",
" alpha = SelectStepSize(x, h, f, **kwargs)\n",
" x = x + alpha * h\n",
" return x\n",
"\n",
"```"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Способ 1: направление убывания\n",
"Рассмотрим линейную аппроксимацию дифференцируемой функции $f$ вдоль некоторого направления убывания $h, \\|h\\|_2 = 1$:\n",
"$$\n",
"f(x + \\alpha h) = f(x) + \\alpha \\langle f'(x), h \\rangle + o(\\alpha)\n",
"$$\n",
"Из условия убывания\n",
"$$\n",
"f(x) + \\alpha \\langle f'(x), h \\rangle + o(\\alpha) < f(x)\n",
"$$\n",
"и переходя к пределу при $\\alpha \\rightarrow 0$:\n",
"$$\n",
"\\langle f'(x), h \\rangle \\leq 0\n",
"$$\n",
"Также из неравенства Коши-Буняковского-Шварца\n",
"$$\n",
"\\langle f'(x), h \\rangle \\geq -\\| f'(x) \\|_2 \\| h \\|_2 = -\\| f'(x) \\|_2\n",
"$$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"Таким образом, направление антиградиента \n",
"$$\n",
"h = -\\dfrac{f'(x)}{\\|f'(x)\\|_2}\n",
"$$\n",
"даёт направление **наискорейшего локального** убывания функции$~f$.\n",
"\n",
"В итоге метод имеет вид\n",
"$$\n",
"x_{k+1} = x_k - \\alpha f'(x_k)\n",
"$$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Способ 2: схема Эйлера решения ОДУ\n",
"\n",
"Рассмотрим обыкновенное диференциальное уравнение вида:\n",
"$$\n",
"\\frac{dx}{dt} = -f'(x(t))\n",
"$$\n",
"и дискретизуем его на равномерной сетке с шагом $\\alpha$:\n",
"$$\n",
"\\frac{x_{k+1} - x_k}{\\alpha} = -f'(x_k),\n",
"$$\n",
"где $x_k \\equiv x(t_k)$ и $\\alpha = t_{k+1} - t_k$ - шаг сетки.\n",
"\n",
"Отсюда получаем выражение для $x_{k+1}$\n",
"$$\n",
"x_{k+1} = x_k - \\alpha f'(x_k),\n",
"$$\n",
"которое в точности совпадает с выражением для градиентного спуска.\n",
"\n",
"Такая схема называется явной или прямой схемой Эйлера.\n",
"\n",
"**Вопрос:** какая схема называется неявной или обратной?"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Способ 3: манипуляции с необходимым условием\n",
"\n",
"\\begin{align*}\n",
"& f'(x) = 0\\\\\n",
"& -\\alpha f'(x) = 0\\\\\n",
"& x - \\alpha f'(x) = x\\\\\n",
"& x_k - \\alpha f'(x_k) = x_{k+1}\n",
"\\end{align*}\n",
"Это **HE** доказательство корректности, только формальное получение выражения!"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Способ 4: минимизация квадратичной оценки сверху \n",
"#### (А. В. Гасников Метод универсального градиентного спуска https://arxiv.org/abs/1711.00394)\n",
"\n",
"Глобальная оценка сверху на функцию $f$ в точке $x_k$:\n",
"$$\n",
"f(y) \\leq f(x_k) + \\langle f'(x_k), y - x_k \\rangle + \\frac{L}{2} \\|y - x_k \\|_2^2 = g(y), \n",
"$$\n",
"где $\\lambda_{\\max}(f''(x)) \\leq L$ для всех допустимых $x$.\n",
"\n",
"Справа &mdash; квадратичная форма, точка минимума которой имеет аналитическое выражение:\n",
"\\begin{align*}\n",
"& g'(y^*) = 0 \\\\\n",
"& f'(x_k) + L (y^* - x_k) = 0 \\\\\n",
"& y^* = x_k - \\frac{1}{L}f'(x_k) = x_{k+1}\n",
"\\end{align*}\n",
"Этот способ позволяет оценить значение шага как $\\frac{1}{L}$. Однако часто константа $L$ неизвестна."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Способ 5: концепция доверительных областей\n",
"\n",
"Минимизация линейной аппроксимации в шаре радиуса $r$ с центром в точке $x_k$\n",
"\\begin{align}\n",
"& \\min \\; f(x_k) + \\langle f'(x_k), x - x_k \\rangle\\\\\n",
"\\text{s.t. } & \\| x - x_k \\|^2_2 \\leq r^2\n",
"\\end{align}\n",
"\n",
"Используя метод множителей Лагранжа и условия ККТ, получим\n",
"$$\n",
"x_{k+1} = x_k - \\frac{1}{2\\lambda} f'(x_k),\n",
"$$\n",
"где $\\lambda$ &mdash; множитель Лагранжа, и\n",
"$$\n",
"\\lambda = \\frac{\\| f'(x_k)\\|_2}{2r}\n",
"$$\n",
"\n",
"**Вопрос:** как выбрать $r$?\n",
"\n",
"**Вопрос:** какое соотношение между константой Липшица $L$ градиента $f'(x_k)$ из способа 4 и $\\lambda$?"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Итого: метод градиентного спуска &mdash; дёшево и сердито\n",
"```python\n",
"def GradientDescentMethod(f, x0, epsilon, **kwargs):\n",
" x = x0\n",
" while StopCriterion(x, f, **kwargs) > epsilon:\n",
" h = ComputeGradient(x, f, **kwargs)\n",
" alpha = SelectStepSize(x, h, f, **kwargs)\n",
" x = x - alpha * h\n",
" return x\n",
"\n",
"```"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Как выбрать шаг $\\alpha_k$? (J. Nocedal, S. Wright Numerical Optimization, $\\S$ 3.1.)\n",
"\n",
"Список подходов:\n",
"- Постоянный шаг \n",
"$$\n",
"\\alpha_k = \\overline{\\alpha}\n",
"$$\n",
"- Априорно заданная последовательность, например\n",
"$$\n",
"\\alpha_k = \\dfrac{\\overline{\\alpha}}{\\sqrt{k+1}}\n",
"$$\n",
"- Наискорейший спуск\n",
"$$\n",
"\\alpha_k = \\arg\\min_{\\alpha \\geq 0} f(x_k - \\alpha f'(x_k))\n",
"$$\n",
"- Требование **достаточного** убывания, требование **существенного** убывания и условие кривизны: для некоторых $\\beta_1, \\beta_2$, таких что $0 < \\beta_1 < \\beta_2 < 1$ найти $x_{k+1}$ такую что\n",
"\n",
" - Достаточное убывание: $f(x_{k+1}) \\leq f(x_k) - \\beta_1 \\alpha_k \\langle f'(x_k), h_k \\rangle$ или\n",
" $ f(x_k) - f(x_{k+1}) \\geq \\beta_1 \\alpha_k \\langle f'(x_k), h_k \\rangle\n",
" $\n",
" - Существенное убывание: $f(x_{k+1}) \\geq f(x_k) - \\beta_2 \\alpha_k \\langle f'(x_k), h_k \\rangle$ или\n",
" $\n",
" f(x_k) - f(x_{k+1}) \\leq \\beta_2 \\alpha_k \\langle f'(x_k), h_k \\rangle\n",
" $\n",
" - Условие кривизны: $\\langle f'(x_{k+1}), h_k \\rangle \\geq \\beta_2 \\langle f'(x_k), h_k \\rangle$\n",
"\n",
"Обычно коэффициенты выбирают так: $\\beta_1 \\in (0, 0.3)$, а $\\beta_2 \\in (0.9, 1)$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Анализ и мотивация подходов к выбору шага $\\alpha_k$\n",
"- Постоянный шаг: самое простое и неэффективное решение\n",
"- Априорно заданная последовательность: немногим лучше постоянного шага\n",
"- Наискорейший спуск: самое лучшее решение, но применимо только если вспомогательная задача решается аналитически или ооооооочень быстро. <br></br>\n",
"То есть почти всегда неприменимо :)\n",
"- Требование достаточного убывания, требование существенного убывания и условие кривизны:\n",
" - требование достаточного убывания гарантирует, что функция в точке $x_{k+1}$ не превосходит линейной аппроксимации с коэффициентом наклона $\\beta_1$\n",
" - требование существенного убывания гарантирует, что функция в точке $x_{k+1}$ убывает не меньше, чем линейная аппроксимация c коэффициентом наклона $\\beta_2$\n",
" - условие кривизны гарантирует, что угол наклона касательной в точке $x_{k+1}$ не меньше, чем угол наклона касательной в точке $x_k$, <br></br>\n",
"умноженный на $\\beta_2$ \n",
"\n",
"Требование существенного убывания и условие кривизны обеспечивают убывание функции по выбранному направлению $h_k$. Обычно выбирают одно из них.\n",
"[comment]: <> (<img src=\"Goldstein.png\", style=\"width: 600px;\">)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"#### Альтернативные названия\n",
"- Требование достаточного убывания $\\equiv$ правило Армихо\n",
"- Требование достаточного убывания + условие кривизны $\\equiv$ правило Вольфа\n",
"- Требование достаточного убывания + требование существенного убывания $\\equiv$ правило Гольдштейна"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Зачем нужно условие существенного убывания?"
]
},
{
"cell_type": "code",
"execution_count": 1,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"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\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\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=\"640\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"application/vnd.jupyter.widget-view+json": {
"model_id": "42430f4391914c83b9e16e2af707e7a1",
"version_major": 2,
"version_minor": 0
},
"text/html": [
"<p>Failed to display Jupyter Widget of type <code>interactive</code>.</p>\n",
"<p>\n",
" If you're reading this message in the Jupyter Notebook or JupyterLab Notebook, it may mean\n",
" that the widgets JavaScript is still loading. If this message persists, it\n",
" likely means that the widgets JavaScript library is either not installed or\n",
" not enabled. See the <a href=\"https://ipywidgets.readthedocs.io/en/stable/user_install.html\">Jupyter\n",
" Widgets Documentation</a> for setup instructions.\n",
"</p>\n",
"<p>\n",
" If you're reading this message in another frontend (for example, a static\n",
" rendering on GitHub or <a href=\"https://nbviewer.jupyter.org/\">NBViewer</a>),\n",
" it may mean that your frontend doesn't currently support widgets.\n",
"</p>\n"
],
"text/plain": [
"interactive(children=(FloatSlider(value=1.5, description='Initial point', max=4.0, min=-4.0), FloatSlider(value=0.8, description='Step', max=1.2), Output()), _dom_classes=('widget-interact',))"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"%matplotlib notebook\n",
"import matplotlib.pyplot as plt\n",
"import ipywidgets as ipywidg\n",
"import numpy as np\n",
"\n",
"def gd(f, gradf, x0, step, eps=1e-6, maxiter=10):\n",
" x = x0\n",
" x_hist = [x]\n",
" n_iter = 0\n",
" while abs(gradf(x)) > eps and n_iter < maxiter:\n",
" x = x - step * gradf(x)\n",
" x_hist.append(x)\n",
" n_iter += 1\n",
" return x, x_hist\n",
"\n",
"f = lambda x: x**2\n",
"gradf = lambda x: 2 * x\n",
"\n",
"fig = plt.figure()\n",
"ax = fig.add_subplot(1, 1, 1)\n",
"plt.rc(\"text\", usetex=True)\n",
"\n",
"def update(x0, step):\n",
" _, x_hist = gd(f, gradf, x0, step)\n",
" x = np.linspace(-np.max(x_hist) - 1, np.max(x_hist) + 1)\n",
" ax.clear()\n",
" ax.plot(x, f(x), color=\"r\", label=\"$f(x) = x^2$\")\n",
" y_hist = np.array([f(x) for x in x_hist])\n",
" x_hist = np.array(x_hist)\n",
" plt.quiver(x_hist[:-1], y_hist[:-1], x_hist[1:]-x_hist[:-1], y_hist[1:]-y_hist[:-1], \n",
" scale_units='xy', angles='xy', scale=1, width=0.005, color=\"green\", label=\"Descent path\")\n",
" ax.legend()\n",
" fig.canvas.draw()\n",
"\n",
"step_slider = ipywidg.FloatSlider(value=0.8, min=0, max=1.2, step=0.1, description=\"Step\")\n",
"x0_slider = ipywidg.FloatSlider(value=1.5, min=-4, max=4, step=0.1, description=\"Initial point\")\n",
"_ = ipywidg.interact(update, x0=x0_slider, step=step_slider)"
]
},
{
"cell_type": "code",
"execution_count": 2,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"outputs": [],
"source": [
"def plot_alpha(f, x, h, alphas):\n",
" df = np.zeros_like(alphas)\n",
" for i, alpha in enumerate(alphas):\n",
" df[i] = f(x + alpha * h)\n",
" plt.plot(alphas, df)\n",
" plt.xlabel(r\"$\\alpha$\", fontsize=18)\n",
" plt.ylabel(r\"$f(x + \\alpha h)$\", fontsize=18)"
]
},
{
"cell_type": "code",
"execution_count": 4,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"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\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\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=\"640\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
}
],
"source": [
"plot_alpha(lambda x: x**2, 0.5, -1, np.linspace(1e-3, 1.01, 10))"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Backtracking \n",
"\n",
"```python\n",
"def SelectStepSize(x, f, h, rho, alpha0, beta1, beta2):\n",
" # 0 < rho < 1\n",
" # alpha0 - initial guess of step size\n",
" # beta1 and beta2 - constants from conditions\n",
" alpha = alpha0\n",
" # Check violating sufficient decrease and curvature conditions \n",
" while (f(x - alpha * h) >= f(x) + beta1 * alpha grad_f(x_k).dot(h)) and \n",
" (grad_f(x - alpha * h).dot(h) <= beta2 * grad_f(x_k).dot(h)):\n",
" alpha *= rho\n",
" return alpha\n",
"```"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Теоремы сходимости (Б.Т. Поляк Введение в оптимизацию, гл. 1, $\\S$ 4; гл. 3, $\\S$ 1; Ю.Е. Нестеров Введение в выпуклую оптимизацию, $\\S$ 2.2)\n",
"От общего к частному:"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"**Теорема 1.** \n",
"Пусть \n",
"\n",
"- $f(x)$ дифференцируема на $\\mathbb{R}^n$, \n",
"- градиент $f(x)$ удовлетворяет условию Липшица с константой $L$\n",
"- $f(x)$ ограничена снизу\n",
"- $\\alpha = const$ и $0 < \\alpha < \\frac{2}{L}$\n",
"\n",
"Тогда для градиентного метода выполнено:\n",
"$$\n",
"\\lim\\limits_{k \\to \\infty} f'(x_k) = 0,\n",
"$$\n",
"а функция монотонно убывает $f(x_{k+1}) < f(x_k)$."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"** Теорема 2.**\n",
"Пусть\n",
"- $f(x)$ дифференцируема на $\\mathbb{R}^n$, \n",
"- градиент $f(x)$ непрерывен\n",
"- множество $\\{ x: f(x) \\leq f(x_0) \\}$ ограничено\n",
"- $\\alpha_k = \\arg\\min\\limits_{\\alpha \\geq 0} f(x_k - \\alpha f'(x_k))$\n",
"\n",
"Тогда \n",
"$$\n",
"f'(x_k) \\to 0, \\; k \\to \\infty \\qquad x_{k_i} \\to x^*\n",
"$$\n"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"**Теорема 3.** Пусть\n",
"- $f(x)$ дифференцируема на $\\mathbb{R}^n$\n",
"- $f(x)$ выпукла \n",
"- $f'(x)$ удовлетворяет условию Липшица с константой $L$\n",
"- $\\alpha = \\dfrac{1}{L}$\n",
"\n",
"Тогда \n",
"$$\n",
"f(x_k) - f^* \\leq \\dfrac{2L \\| x_0 - x^*\\|^2_2}{k+4}\n",
"$$\n"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"**Теорема 4.** \n",
"Пусть\n",
"\n",
"- $f(x)$ дифференцируема на $\\mathbb{R}^n$, \n",
"- градиент $f(x)$ удовлетворяет условию Липшица с константой $L$\n",
"- $f(x)$ является сильно выпуклой с константой $l$\n",
"- $\\alpha = const$ и $0 < \\alpha < \\frac{2}{L}$\n",
"\n",
"Тогда градиентный метод сходится к единственной точке глобального минимума $x^*$ с линейной скоростью:\n",
"$$\n",
"\\| x_k - x^* \\|_2 \\leq cq^k, \\qquad 0 \\leq q < 1\n",
"$$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"**Теорема 5.**\n",
"Пусть\n",
"\n",
"- $f(x)$ дифференцируема на $\\mathbb{R}^n$, \n",
"- градиент $f(x)$ удовлетворяет условию Липшица с константой $L$\n",
"- $f(x)$ является сильно выпуклой с константой $l$\n",
"- $\\alpha = \\dfrac{2}{l + L}$\n",
"\n",
"Тогда для градиентного метода выполнено:\n",
"$$\n",
"\\| x_k - x^* \\|^2_2 \\leq \\left( \\dfrac{M - 1}{M + 1} \\right)^k \\|x_0 - x^*\\|^2_2 \\qquad f(x_k) - f^* \\leq \\dfrac{L}{2} \\left( \\dfrac{M - 1}{M + 1} \\right)^{2k} \\| x_0 - x^*\\|^2_2,\n",
"$$\n",
"где $M = \\frac{L}{l}$"
]
},
{
"cell_type": "markdown",
"metadata": {
"collapsed": true,
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"**Теорема 6.**\n",
"Пусть \n",
"- $f(x)$ дважды дифференцируема и $l\\mathbf{I} \\preceq f''(x) \\preceq L\\mathbf{I}$ для всех $x$\n",
"- $\\alpha = const$ и $0 < \\alpha < \\frac{2}{L}$\n",
"\n",
"Тогда \n",
"$$\n",
"\\| x_k - x^*\\|_2 \\leq \\|x_0 - x^*\\|_2 q^k, \\qquad q = \\max(|1 - \\alpha l|, |1 - \\alpha L|) < 1\n",
"$$\n",
"и минимальное $q^* = \\dfrac{L - l}{L + l}$ при $\\alpha^* = \\dfrac{2}{L + l}$"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### От чего зависит $q^*$ и как это использовать?\n",
"Из Теорем 5 и 6 имеем \n",
"$$\n",
"q^* = \\dfrac{L - l}{L + l} = \\dfrac{L/l - 1}{L/l + 1} = \\dfrac{M - 1}{M + 1},\n",
"$$\n",
"где $M$ - оценка числа обусловленности $f''(x)$.\n",
"\n",
"**Вопрос**: что такое число обусловленности матрицы?\n",
"\n",
"- При $M \\gg 1$, $q^* \\to 1 \\Rightarrow$ оооочень **медленная** сходимости градиентного метода. Например при $M = 100$: $q^* \\approx 0.98 $\n",
"- При $M \\simeq 1$, $q^* \\to 0 \\Rightarrow$ **ускорение** сходимости градиентного метода. Например при $M = 4$: $q^* = 0.6 $\n",
"\n",
"**Вопрос**: какая геометрия у этого требования?\n",
"\n",
"**Мораль**: необходимо сделать оценку $M$ как можно ближе к 1!\n",
"\n",
"О том, как это сделать, Вам будет предложено подумать в домашнем задании :)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Выбор начального приближения\n",
"\n",
"Обратите внимание, что ни в одной из теорем приведённых выше не было условий на начальное приближение $x_0$!\n",
"\n",
"- Для невыпуклой функции градиентный спуск сойдётся к некоторой стационарной точке, которая может быть как локальным минимумом, так и седловой точкой \n",
"- Для выпуклых функций стационарная точка будет точкой минимума\n",
"- Способы выбора начального приближения:\n",
" - случайная точка\n",
" - безградиентные методы случайного поиска, например [эволюционный алгоритм](https://docs.scipy.org/doc/scipy-0.18.1/reference/generated/scipy.optimize.differential_evolution.html), алгоритм муравьиной колонии (ant colony) и т.д.\n",
" - обычно эти методы могут дать лишь незначительное уменьшение функции, однако сильно отойти от локальных минимумов"
]
},
{
"cell_type": "markdown",
"metadata": {
"collapsed": true,
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Вычислительный аспект и эксперименты\n",
"1. Для каждого шага метода нужно хранить только текущую точку и вектор градиента: $O(n)$ памяти\n",
"2. Поиск $\\alpha_k$:\n",
" - дан априори\n",
" - ищется из аналитического решения задачи наискорейшего спуска\n",
" - заканчивается за конечное число шагов\n",
"3. Для каждого шага метода нужно вычислять линейную комбинацию векторов: $O(n)$ вычислений + высокопроизводительные реализации"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Pеализация градиентного спуска"
]
},
{
"cell_type": "code",
"execution_count": 5,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"import numpy as np\n",
"\n",
"def GradientDescent(f, gradf, x0, epsilon, num_iter, line_search, \n",
" disp=False, callback=None, **kwargs):\n",
" x = x0.copy()\n",
" iteration = 0\n",
" opt_arg = {\"f\": f, \"grad_f\": gradf}\n",
" for key in kwargs:\n",
" opt_arg[key] = kwargs[key]\n",
" while True:\n",
" gradient = gradf(x)\n",
" alpha = line_search(x, -gradient, **opt_arg)\n",
" x = x - alpha * gradient\n",
" if callback is not None:\n",
" callback(x)\n",
" iteration += 1\n",
" if disp:\n",
" print(\"Current function val =\", f(x))\n",
" print(\"Current gradient norm = \", np.linalg.norm(gradf(x)))\n",
" if np.linalg.norm(gradf(x)) < epsilon:\n",
" break\n",
" if iteration >= num_iter:\n",
" break\n",
" res = {\"x\": x, \"num_iter\": iteration, \"tol\": np.linalg.norm(gradf(x))}\n",
" return res"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Выбор шага"
]
},
{
"cell_type": "code",
"execution_count": 6,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"def constant_step(x, gradient, **kwargs):\n",
" return 1e-3\n",
"\n",
"# For quadratic function x'Ax\n",
"def exact_linesearch(x, gradient, **kwargs):\n",
" return max(0, -x.dot(A.dot(gradient)) / gradient.dot(A.dot(gradient)))\n",
"\n",
"def backtracking(x, descent_dir, **kwargs):\n",
" f = kwargs[\"f\"]\n",
" grad_f = kwargs[\"grad_f\"] \n",
" if kwargs[\"method\"] == \"Armijo\":\n",
" beta1 = kwargs[\"beta1\"]\n",
" rho = kwargs[\"rho\"]\n",
" alpha = 1\n",
" while f(x + alpha * descent_dir) >= f(x) + beta1 * alpha * grad_f(x).dot(descent_dir) or \\\n",
" np.isnan(f(x + alpha * descent_dir)):\n",
" alpha *= rho\n",
" if alpha < 1e-16:\n",
" break\n",
" return alpha"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Зависимость от обусловленности матрицы $f''(x)$\n",
"Рассмотрим задачу \n",
"$$\n",
"\\min f(x),\n",
"$$ \n",
"где\n",
"$$\n",
"f(x) = x^{\\top}Ax, \\; A = \n",
"\\begin{bmatrix}\n",
"1 & 0\\\\\n",
"0 & \\gamma\n",
"\\end{bmatrix}\n",
"$$\n",
"\n",
"$$\n",
"f'(x) = 2Ax\n",
"$$"
]
},
{
"cell_type": "code",
"execution_count": 7,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"def my_f(x, A):\n",
" return x.dot(A.dot(x))\n",
"\n",
"def my_gradf(x, A):\n",
" return 2 * A.dot(x)"
]
},
{
"cell_type": "code",
"execution_count": 8,
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"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\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\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=\"800\">"
],
"text/plain": [
"<IPython.core.display.HTML object>"
]
},
"metadata": {},
"output_type": "display_data"
},
{
"data": {
"text/plain": [
"Text(0,0.5,'Number of iterations with $\\\\varepsilon = 10^{-7}$')"
]
},
"execution_count": 8,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"plt.rc(\"text\", usetex=True)\n",
"\n",
"gammas = [0.1, 0.5, 1, 2, 3, 4, 5, 10, 20, 50, 100, 1000, 5000, 10000]\n",
"num_iter_converg = []\n",
"for g in gammas:\n",
" A = np.array([[1, 0], \n",
" [0, g]], dtype=np.float64)\n",
" f = lambda x: my_f(x, A)\n",
" gradf = lambda x: my_gradf(x, A)\n",
"# x0 = np.random.rand(A.shape[0])\n",
"# x0 = np.sort(x0)\n",
"# x0 = x0[::-1]\n",
" x0 = np.array([g, 1], dtype=np.float64)\n",
"# print x0[1] / x0[0]\n",
" res = GradientDescent(f, gradf, x0, 1e-7, 10000, exact_linesearch)\n",
" num_iter_converg.append(res[\"num_iter\"])\n",
"\n",
"plt.figure(figsize=(8, 6))\n",
"plt.loglog(gammas, num_iter_converg)\n",
"plt.xticks(fontsize = 20)\n",
"plt.yticks(fontsize = 20)\n",
"plt.xlabel(r\"$\\gamma$\", fontsize=20)\n",
"plt.ylabel(r\"Number of iterations with $\\varepsilon = 10^{-7}$\", fontsize=20)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"- При неудачном начальном приближении сходимость для плохо обусловенной задачи очень медленная\n",
"- При случайном начальном приближении сходимость может быть гораздо быстрее теоретических оценок\n",
"- **Open problem:** получить оценки сходимости, использующие случайное начальное приближение, а не худшее из возможных"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Эксперимент на многомерной задаче\n",
"Пусть $A \\in \\mathbb{R}^{m \\times n}$. Рассмотрим систему линейных неравенств: $Ax \\leq 1$ при условии $|x_i| \\leq 1$ для всех $i$.\n",
"\n",
"**Определение.** Аналитическим центром системы неравенств $Ax \\leq 1$ при условии $|x_i| \\leq 1$ является решение задачи\n",
"$$\n",
"f(x) = - \\sum_{i=1}^m \\log(1 - a_i^{\\top}x) - \\sum_{i = 1}^n \\log (1 - x^2_i) \\to \\min_x\n",
"$$\n",
"$$\n",
"f'(x) - ?\n",
"$$"
]
},
{
"cell_type": "code",
"execution_count": 9,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"import numpy as np\n",
"n = 100\n",
"m = 200\n",
"x0 = np.zeros(n)\n",
"A = np.random.rand(n, m)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Точное решение с помощью CVXPy"
]
},
{
"cell_type": "code",
"execution_count": 10,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"['ECOS', 'ECOS_BB', 'SCS', 'LS']\n",
"----------------------------------------------------------------------------\n",
"\tSCS v1.2.6 - Splitting Conic Solver\n",
"\t(c) Brendan O'Donoghue, Stanford University, 2012-2016\n",
"----------------------------------------------------------------------------\n",
"Lin-sys: sparse-indirect, nnz in A = 20700, CG tol ~ 1/iter^(2.00)\n",
"eps = 1.00e-03, alpha = 1.50, max_iters = 2500, normalize = 1, scale = 1.00\n",
"Variables n = 500, constraints m = 1200\n",
"Cones:\tsoc vars: 300, soc blks: 100\n",
"\texp vars: 900, dual exp vars: 0\n",
"Setup time: 2.32e-03s\n",
"----------------------------------------------------------------------------\n",
" Iter | pri res | dua res | rel gap | pri obj | dua obj | kap/tau | time (s)\n",
"----------------------------------------------------------------------------\n",
" 0| 1.31e+02 2.67e+03 7.93e-01 -3.31e+04 -3.82e+03 0.00e+00 5.26e-03 \n",
" 80| 4.89e-05 6.44e-04 3.52e-05 -6.49e+02 -6.49e+02 1.43e-13 2.09e-01 \n",
"----------------------------------------------------------------------------\n",
"Status: Solved\n",
"Timing: Solve time: 2.09e-01s\n",
"\tLin-sys: avg # CG iterations: 2.89, avg solve time: 2.59e-04s\n",
"\tCones: avg projection time: 2.29e-03s\n",
"----------------------------------------------------------------------------\n",
"Error metrics:\n",
"dist(s, K) = 1.7077e-05, dist(y, K*) = 2.2204e-16, s'y/|s||y| = -9.3095e-12\n",
"|Ax + s - b|_2 / (1 + |b|_2) = 4.8928e-05\n",
"|A'y + c|_2 / (1 + |c|_2) = 6.4429e-04\n",
"|c'x + b'y| / (1 + |c'x| + |b'y|) = 3.5160e-05\n",
"----------------------------------------------------------------------------\n",
"c'x = -648.8600, -b'y = -648.8143\n",
"============================================================================\n",
"Optimal value = -648.8599505343861\n"
]
}
],
"source": [
"import cvxpy as cvx\n",
"print(cvx.installed_solvers())\n",
"x = cvx.Variable(n, 1)\n",
"\n",
"obj = cvx.Minimize(cvx.sum_entries(-cvx.log(1 - A.T * x)) - \n",
" cvx.sum_entries(cvx.log(1 - cvx.square(x))))\n",
"prob = cvx.Problem(obj)\n",
"prob.solve(solver=\"SCS\", verbose=True)\n",
"x = x.value\n",
"print(\"Optimal value =\", prob.value)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"### Решение с помощью градиентного спуска"
]
},
{
"cell_type": "code",
"execution_count": 11,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"f = lambda x: -np.sum(np.log(1 - A.T.dot(x))) - np.sum(np.log(1 - x*x))\n",
"grad_f = lambda x: np.sum(A.dot(np.diagflat(1 / (1 - A.T.dot(x)))), \\\n",
" axis=1) + 2 * x / (1 - np.power(x, 2))\n",
"def my_callback(x, array):\n",
" array.append(x)"
]
},
{
"cell_type": "code",
"execution_count": 12,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"name": "stdout",
"output_type": "stream",
"text": [
"Gradient norm = 6.001795377638296e-05 and optimal value = -648.8919249928828\n"
]
},
{
"name": "stderr",
"output_type": "stream",
"text": [
"/Users/alex/anaconda3/envs/cvxpy/lib/python3.6/site-packages/ipykernel_launcher.py:1: RuntimeWarning: invalid value encountered in log\n",
" \"\"\"Entry point for launching an IPython kernel.\n"
]
},
{
"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\", \"pdf\", \"png\", \"ps\", \"raw\", \"svg\"];\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=\"
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment