{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Writing efficient DOcplex code\n",
    "\n",
    "In this notebook, we show how to improve efficiency of DOcplex models using five simple rules.\n",
    "\n",
    ">This notebook is part of **[Prescriptive Analytics for Python](http://ibmdecisionoptimization.github.io/docplex-doc/)**\n",
    ">\n",
    ">It requires either an [installation of CPLEX Optimizers](http://ibmdecisionoptimization.github.io/docplex-doc/getting_started.html) or it can be run on [IBM Watson Studio Cloud](https://www.ibm.com/cloud/watson-studio/) (Sign up for a [free IBM Cloud account](https://dataplatform.cloud.ibm.com/registration/stepone?context=wdp&apps=all>)\n",
    "and you can start using Watson Studio Cloud right away)."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## A simple timing tool\n",
    "\n",
    "To measure performance we need a simple timing tool. For this purpose, we chose to implement a Python context manager object (see https://docs.python.org/3/reference/datamodel.html#context-managers) for details.\n",
    "\n",
    "This object stores the start time when entering a block and reports time spent when exiting the block. Python's `with` statement avoids cluttering code with intrusive prints."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin fibonacci 30\n",
      "fibonacci(30) = 832040\n",
      "<-- end fibonacci 30,  time: 154 ms\n"
     ]
    }
   ],
   "source": [
    "import time\n",
    "import math\n",
    "\n",
    "class ContextTimer(object):\n",
    "    def __init__(self, msg):\n",
    "        self.msg = msg\n",
    "        self.start = 0\n",
    "        \n",
    "    def __enter__(self):\n",
    "        self.start = time.time()\n",
    "        print('--> begin {0}'.format(self.msg))\n",
    "        return self  # return value is value of with ()\n",
    "        \n",
    "    def __exit__(self, *args):\n",
    "        elapsed = time.time() - self.start\n",
    "        self.msecs = math.ceil(1000* elapsed)\n",
    "        print('<-- end {0},  time: {1:.0f} ms'.format(self.msg, self.msecs))   \n",
    "        \n",
    "# try our timer on computing fibonacci numbers\n",
    "def fib(n):\n",
    "    return 1 if n <= 2 else  fib(n-1) + fib(n-2)\n",
    "\n",
    "# timing fibonacci(30)\n",
    "with ContextTimer(\"fibonacci 30\"):\n",
    "    n = 30\n",
    "    f = fib(n)\n",
    "    print(\"fibonacci({0}) = {1}\".format(n, f))\n",
    "        "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## The benchmark model\n",
    "\n",
    "To compare various implementations, we need a simple, scalable benchmark model. The model has no real business meaning, but is simple to grasp\n",
    "and can be scaled by changing one `size` parameter.\n",
    "Note that we'll be comparing only the _build_ time of the model, not the _solve_ time.\n",
    "\n",
    "Note that the model has _n_ constraints, all with expressions of size _N_, so we expect the underlying matrix _size_ to grow as $O(N^2)$."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Description\n",
    "\n",
    "Let $N$ be an integer (the size of the problem).\n",
    "\n",
    "$$\n",
    "minimize \\sum_{k=0}^{k=N-1} (k+1) * y_{k}\\\\\n",
    "s.t.\\\\\n",
    "\\forall\\ \\ m\\ in \\{0..N-1\\}\\ \\ \\sum_{l=0}^{l=N-1} (y_{l} * (l+ (l+m) \\%3) \\ge l\\\\\n",
    "y_{k} = 0, 1\\\\\n",
    "\\\\\n",
    "\\sum_{l} y_{l} \\ge 2\\\\\n",
    "$$"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## A beginners's implementation of the model\n",
    "\n",
    "In this section we show a Python/Docplex beginner's implementation of this model.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from docplex.mp.model import Model\n",
    "\n",
    "def build_bench_model1(size=10):\n",
    "    m = Model(name=\"bench1\")\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"my_yvar\")\n",
    "    # create constraints\n",
    "    k = {(i,j) : (i + (i+j) %3) for i in rsize for j in rsize}\n",
    "    for i in rsize:\n",
    "        m.add(m.sum(ys[i] * k[i,j] for j in rsize) >= i, \"ct_sum_yjs_%d\" %i)\n",
    "    # for minimize, create a list of coefficients\n",
    "    rsize1 = [i+1 for i in rsize]\n",
    "    m.minimize(m.sum(ys[k] * rsize[k] for k in rsize))\n",
    "    return m"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Lets run our context timer with N=1000; we expect a model with 1000 variables and 1000 constraints:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench1_size_1000\n",
      "<-- end bench1_size_1000,  time: 3333 ms\n",
      "Model: bench1\n",
      " - number of variables: 1000\n",
      "   - binary=1000, integer=0, continuous=0\n",
      " - number of constraints: 1000\n",
      "   - linear=1000\n",
      " - parameters: defaults\n",
      " - problem type is: MILP\n"
     ]
    }
   ],
   "source": [
    "with ContextTimer(\"bench1_size_1000\"):\n",
    "    m11k = build_bench_model1(1000)\n",
    "m11k.print_information()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "As expected the model has 1000 variables and 1000 constraints and the build time is significant.\n",
    "For N=3000, we can expect an increase in buid time by a factor of 9, so you might as well go grab a coffee while this cell executes.."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench1 size=3000\n",
      "<-- end bench1 size=3000,  time: 24649 ms\n"
     ]
    }
   ],
   "source": [
    "N=3000\n",
    "with ContextTimer(\"bench1 size={0}\".format(N)):\n",
    "    build_bench_model1(N)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Pitfall #1 : use Model.sum(), not Python sum()\n",
    "\n",
    "In the above code, we compute the sum of variables using `Model.sum()`, not Python builtin function `sum`. One could wonder why Docplex had to redefine a specific sum function?\n",
    "\n",
    "Python's um function calls the `+` operator repeatedly, that is, sum([x,y,z, t]) is evaluated as ((x+y)+z)+t. Calling `sum` with `N` arguments then creates `N` intermediate expressions, but there's more. Each intermediate expression has to be copied for the next sum, and as the size of the intermediate expressions grow, each copy takes an O(n) time, and th efinal sum has time in O(N^2).\n",
    "On the opposite, `Model.sum()` creates only _one_ expression and incrementally adds each argument to it.\n",
    "\n",
    "**Summary**: never use Python's builtin sum, instead use `Model.sum()` to compute expresssions in Docplex.\n",
    "\n",
    "Let's experiment with our benchmark model. First we define a variant function to build the same model as above with Python `sum`, and measure the two build times."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin python_sum_too_slow_n=1000\n",
      "<-- end python_sum_too_slow_n=1000,  time: 5435 ms\n",
      "--> begin same_model_with_model_sum_n=1000\n",
      "<-- end same_model_with_model_sum_n=1000,  time: 2546 ms\n"
     ]
    }
   ],
   "source": [
    "def build_bench_sum(size=10):\n",
    "    m = Model(name=\"bench1\")\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"my_yvar\")\n",
    "    # create constraints\n",
    "    k = {(i,j) : (i + (i+j) %3) for i in rsize for j in rsize}\n",
    "    for i in rsize:\n",
    "        m.add(sum(ys[i] * k[i,j] for j in rsize) >= i, \"ct_sum_yjs_%d\" %i)\n",
    "    # for minimize, create a list of coefficients\n",
    "    rsize1 = [i+1 for i in rsize]\n",
    "    m.minimize(sum(ys[k] * rsize[k] for k in rsize))\n",
    "    return m\n",
    "\n",
    "s = 1000\n",
    "with ContextTimer(\"python_sum_too_slow_n=%d\" % s):\n",
    "    build_bench_sum(size=s)\n",
    "\n",
    "with ContextTimer(\"same_model_with_model_sum_n=%d\" % s):\n",
    "    build_bench_model1(size=s)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Building the Python sum model takes roughly twice as much time as the model with `Model.sum()'."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#  Common tips to improve DOcplex code efficiency"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Tip #1: Use scalar product\n",
    "\n",
    "When building large expressions,  scalar product (`Model.scal_prod()`) is \n",
    "an efficient way to combine a sequence of variables (or expressions)\n",
    "and a sequence of coefficients.\n",
    "Try using `scalar_prod` instead of using `for` loops in expressions."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench2 size=3000\n",
      "<-- end bench2 size=3000,  time: 15698 ms\n"
     ]
    }
   ],
   "source": [
    "def build_bench_model2(size=10):\n",
    "    m = Model(name=\"bench2\")\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"y\")\n",
    "    # create a matrix of coefficients\n",
    "    k = {(i,j) : (i + (i+j) %3) for i in rsize for j in rsize}\n",
    "    for i in rsize:\n",
    "        m.add(m.scal_prod([ys[i1] for i1 in rsize], [k[i,j] for j in rsize]) >= i, \"ct_%d\" % i)\n",
    "    # for minimize, create a list of coefficients\n",
    "    rsize1 = [i+1 for i in rsize]\n",
    "    m.minimize(m.scal_prod([ys[k] for k in rsize], rsize1))\n",
    "    return m\n",
    "\n",
    "with ContextTimer(\"bench2 size={0}\".format(N)):\n",
    "    build_bench_model2(N)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Tip #2 (variant) : try functional scalar product (a.k.a. dotf)\n",
    "\n",
    "Since DOcplex 2.9, Scalar product has a _functional_ variant, in which  coefficients are computed on the fly, with no need to prepare a (possibly large) container of numbers. In some cases, this can save a significant time.\n",
    "\n",
    "The method is `Model.dotf` takes two arguments:\n",
    "\n",
    "  - a dictionary of variables, as create by the Model.<type>_var_dict methods\n",
    "  - a Python function that takes a variable key and returns a float, the coefficient\n",
    "  \n",
    "In our example, keys are the integer from 0 to `size-1` .\n",
    "The coefficient for y_j in the i_th constraint is i+(i+j)%3: here we do not need\n",
    "to precompute a list or a comprehension, but only use a lambda function to compute this.\n",
    "\n",
    "We also leverage `dotf` for the objective, where the cost coefficient for y_j is (j+1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench3 size=3000\n",
      "<-- end bench3 size=3000,  time: 9409 ms\n"
     ]
    }
   ],
   "source": [
    "def build_bench_model3(size=10):\n",
    "    m = Model(name=\"bench3\")\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"y\")\n",
    "    # no need to build an explicit matrix\n",
    "    #k = {(i,j) : (i + (i+j) %3) for i in rsize for j in rsize}\n",
    "    for i in rsize:\n",
    "        # the ith function is a function oj returning (i+(i+j)%3)\n",
    "        fi = lambda j_: i + (i+j_) % 3\n",
    "        m.add(m.dotf(ys, fi) >= i, \"ct_%d\" % i)\n",
    "    m.minimize(m.dotf(ys, lambda j_: j_ +1))\n",
    "    return m\n",
    "\n",
    "with ContextTimer(\"bench3 size={0}\".format(N)):\n",
    "    build_bench_model3(N)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Tip #3: Add constraints in batches\n",
    "\n",
    "Adding constraints to the model by batches using `Model.add_constraints()`\n",
    "is usually more efficient.\n",
    "Try grouping constraints in lists or comprehensions (both work).\n",
    "\n",
    "If the constraints are named, pass a second argument with the collection of constraint names.\n",
    "\n",
    "Note that `Model.add` runs `Model.add_constraints` if passed an iterable."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench4 size=3000\n",
      "<-- end bench4 size=3000,  time: 9946 ms\n"
     ]
    }
   ],
   "source": [
    "def build_bench_model4(size=10):\n",
    "    m = Model(name=\"bench4\")\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"y\")\n",
    "    # create constraints\n",
    "    m.add_constraints((m.dotf(ys, lambda j_: i + (i+j_) % 3) >= i for i in rsize),\n",
    "         (\"ct_%d\" % i for i in rsize))\n",
    "    m.minimize(m.dotf(ys, lambda j_: j_ +1))\n",
    "    return m\n",
    "\n",
    "with ContextTimer(\"bench4 size={0}\".format(N)):\n",
    "    build_bench_model4(N)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Tip #4: take control of name generation\n",
    "\n",
    "Naming variables and/or constraints is useful to generate readable LP files. However, generating large numbers of strings may have a significant cost in Python, especially for large models. \n",
    "\n",
    "Passing `ignore_names=True` at model creation time, disables all name generation (variables and constraint names alike) everywhere in the model. \n",
    "\n",
    "By default, names are active and this flag is  `False`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench5 size=3000\n",
      "<-- end bench5 size=3000,  time: 11013 ms\n"
     ]
    }
   ],
   "source": [
    "def build_bench_model5(size=10):\n",
    "    m = Model(name=\"bench5\", ignore_names=True)\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"y\")\n",
    "    # create constraints\n",
    "    m.add((m.dotf(ys, lambda j_: i + (i+j_) % 3) >= i for i in rsize),\n",
    "         (\"ct_%d\" % i for i in rsize))\n",
    "    m.minimize(m.dotf(ys, lambda j_: j_ +1))\n",
    "    return m\n",
    "\n",
    "with ContextTimer(\"bench5 size={0}\".format(N)):\n",
    "    build_bench_model5(N)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Tip #5: take control of argument checking\n",
    "\n",
    "DOcplex usually checks the argumens passed to methods. As this can be useful when writing the model to avoid errors, this checking comes with a runtime cost. When running a deployed model that has been thoroughly tested and tuned, you can remove all checks by adding the `checker=\"off\"` keyword argument to the model constructor.\n",
    "\n",
    "Again, the next version is identical to the previous one , except that type-checking has been disabled."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--> begin bench6 size=3000\n",
      "<-- end bench6 size=3000,  time: 9232 ms\n"
     ]
    }
   ],
   "source": [
    "def build_bench_model6(size=10):\n",
    "    m = Model(name=\"bench6\", ignore_names=True, checker='off')\n",
    "    rsize = range(size)\n",
    "    # create variables as a dictionary indexed by the range\n",
    "    ys = m.binary_var_dict(rsize, name=\"y\")\n",
    "    # create constraints\n",
    "    m.add((m.dotf(ys, lambda j_: i + (i+j_) % 3) >= i for i in rsize),\n",
    "         (\"ct_%d\" % i for i in rsize))\n",
    "    m.add(m.sum(ys) >= 2, \"sum_ys_ge_2\")\n",
    "    m.minimize(m.dotf(ys, lambda j_: j_ +1))\n",
    "    return m\n",
    "\n",
    "with ContextTimer(\"bench6 size={0}\".format(N)):\n",
    "    build_bench_model6(N)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Summary\n",
    "\n",
    "From version 1 to version 6 , model build time has decreased from 35s to 4s (on our platform). Results may well differ on other platforms, but still, this demonstrates that the way the model is built can greatly influence the performance.\n",
    "\n",
    "Here is a list of tricks to try to improve model building time:\n",
    "\n",
    " - Use Model.scal_prod wherever possible\n",
    " - Try using Model.dotf when applicable.\n",
    " - Add constraints in batches, not one by one\n",
    " - Try ignoring name generation (for large models)\n",
    " - Try disabling argument checking\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Plotting the trend\n",
    "\n",
    "In this section we compute the times to build different model versions on various sizes, and\n",
    "plot the result on a graph.\n",
    "This code requires the `matplotlib` library to run.\n",
    "\n",
    "**Note**: the next cell might take a significant time to run, as it\n",
    "runs a lot of (size, model_build_function) combinations..."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "* start computing performance data\n",
      "* start computing results...\n",
      "--> begin [1/36] use build_bench_model1 with size=100\n",
      "<-- end [1/36] use build_bench_model1 with size=100,  time: 42 ms\n",
      "--> begin [2/36] use build_bench_model2 with size=100\n",
      "<-- end [2/36] use build_bench_model2 with size=100,  time: 28 ms\n",
      "--> begin [3/36] use build_bench_model3 with size=100\n",
      "<-- end [3/36] use build_bench_model3 with size=100,  time: 23 ms\n",
      "--> begin [4/36] use build_bench_model4 with size=100\n",
      "<-- end [4/36] use build_bench_model4 with size=100,  time: 21 ms\n",
      "--> begin [5/36] use build_bench_model5 with size=100\n",
      "<-- end [5/36] use build_bench_model5 with size=100,  time: 19 ms\n",
      "--> begin [6/36] use build_bench_model6 with size=100\n",
      "<-- end [6/36] use build_bench_model6 with size=100,  time: 17 ms\n",
      "--> begin [7/36] use build_bench_model1 with size=300\n",
      "<-- end [7/36] use build_bench_model1 with size=300,  time: 254 ms\n",
      "--> begin [8/36] use build_bench_model2 with size=300\n",
      "<-- end [8/36] use build_bench_model2 with size=300,  time: 151 ms\n",
      "--> begin [9/36] use build_bench_model3 with size=300\n",
      "<-- end [9/36] use build_bench_model3 with size=300,  time: 117 ms\n",
      "--> begin [10/36] use build_bench_model4 with size=300\n",
      "<-- end [10/36] use build_bench_model4 with size=300,  time: 101 ms\n",
      "--> begin [11/36] use build_bench_model5 with size=300\n",
      "<-- end [11/36] use build_bench_model5 with size=300,  time: 97 ms\n",
      "--> begin [12/36] use build_bench_model6 with size=300\n",
      "<-- end [12/36] use build_bench_model6 with size=300,  time: 67 ms\n",
      "--> begin [13/36] use build_bench_model1 with size=600\n",
      "<-- end [13/36] use build_bench_model1 with size=600,  time: 1021 ms\n",
      "--> begin [14/36] use build_bench_model2 with size=600\n",
      "<-- end [14/36] use build_bench_model2 with size=600,  time: 892 ms\n",
      "--> begin [15/36] use build_bench_model3 with size=600\n",
      "<-- end [15/36] use build_bench_model3 with size=600,  time: 417 ms\n",
      "--> begin [16/36] use build_bench_model4 with size=600\n",
      "<-- end [16/36] use build_bench_model4 with size=600,  time: 410 ms\n",
      "--> begin [17/36] use build_bench_model5 with size=600\n",
      "<-- end [17/36] use build_bench_model5 with size=600,  time: 418 ms\n",
      "--> begin [18/36] use build_bench_model6 with size=600\n",
      "<-- end [18/36] use build_bench_model6 with size=600,  time: 251 ms\n",
      "--> begin [19/36] use build_bench_model1 with size=1000\n",
      "<-- end [19/36] use build_bench_model1 with size=1000,  time: 3026 ms\n",
      "--> begin [20/36] use build_bench_model2 with size=1000\n",
      "<-- end [20/36] use build_bench_model2 with size=1000,  time: 1701 ms\n",
      "--> begin [21/36] use build_bench_model3 with size=1000\n",
      "<-- end [21/36] use build_bench_model3 with size=1000,  time: 1092 ms\n",
      "--> begin [22/36] use build_bench_model4 with size=1000\n",
      "<-- end [22/36] use build_bench_model4 with size=1000,  time: 1207 ms\n",
      "--> begin [23/36] use build_bench_model5 with size=1000\n",
      "<-- end [23/36] use build_bench_model5 with size=1000,  time: 1125 ms\n",
      "--> begin [24/36] use build_bench_model6 with size=1000\n",
      "<-- end [24/36] use build_bench_model6 with size=1000,  time: 704 ms\n",
      "--> begin [25/36] use build_bench_model1 with size=3000\n",
      "<-- end [25/36] use build_bench_model1 with size=3000,  time: 23946 ms\n",
      "--> begin [26/36] use build_bench_model2 with size=3000\n",
      "<-- end [26/36] use build_bench_model2 with size=3000,  time: 15195 ms\n",
      "--> begin [27/36] use build_bench_model3 with size=3000\n",
      "<-- end [27/36] use build_bench_model3 with size=3000,  time: 9592 ms\n",
      "--> begin [28/36] use build_bench_model4 with size=3000\n",
      "<-- end [28/36] use build_bench_model4 with size=3000,  time: 9393 ms\n",
      "--> begin [29/36] use build_bench_model5 with size=3000\n",
      "<-- end [29/36] use build_bench_model5 with size=3000,  time: 10426 ms\n",
      "--> begin [30/36] use build_bench_model6 with size=3000\n",
      "<-- end [30/36] use build_bench_model6 with size=3000,  time: 7888 ms\n",
      "--> begin [31/36] use build_bench_model1 with size=5000\n",
      "<-- end [31/36] use build_bench_model1 with size=5000,  time: 69977 ms\n",
      "--> begin [32/36] use build_bench_model2 with size=5000\n",
      "<-- end [32/36] use build_bench_model2 with size=5000,  time: 45892 ms\n",
      "--> begin [33/36] use build_bench_model3 with size=5000\n",
      "<-- end [33/36] use build_bench_model3 with size=5000,  time: 28273 ms\n",
      "--> begin [34/36] use build_bench_model4 with size=5000\n",
      "<-- end [34/36] use build_bench_model4 with size=5000,  time: 27294 ms\n",
      "--> begin [35/36] use build_bench_model5 with size=5000\n",
      "<-- end [35/36] use build_bench_model5 with size=5000,  time: 29352 ms\n",
      "--> begin [36/36] use build_bench_model6 with size=5000\n",
      "<-- end [36/36] use build_bench_model6 with size=5000,  time: 18474 ms\n",
      "* end computing results\n"
     ]
    }
   ],
   "source": [
    "# various sizes to sample performance\n",
    "sizes = [100, 300, 600, 1000, 3000, 5000]\n",
    "\n",
    "# a lits of tuples (fn, label) to build model and an explanatory label\n",
    "builders = [(build_bench_model1, \"initial\"), \n",
    "            (build_bench_model2, \"scal_prod\"),\n",
    "            (build_bench_model3, \"dotf\"),\n",
    "            (build_bench_model4, \"batch_cts\"),\n",
    "            (build_bench_model5, \"ignore_names\"),\n",
    "            (build_bench_model6, \"names_checker_off\")\n",
    "           ]\n",
    "print(\"* start computing performance data\")\n",
    "res = {}\n",
    "print(\"* start computing results...\")\n",
    "nb_runs = len(sizes) * len(builders)\n",
    "r = 0\n",
    "for s in sizes:\n",
    "    for b, (bf, _) in enumerate(builders):\n",
    "        r +=1 \n",
    "        with ContextTimer(\"[{2}/{3}] use {0} with size={1}\"\n",
    "                          .format(bf.__name__, s, r, nb_runs)) as tt:\n",
    "            m = bf(s)\n",
    "            m.end()\n",
    "        elapsed = tt.msecs\n",
    "        res[b, s] = tt.msecs\n",
    "print(\"* end computing results\")\n",
    "# now we have a dict of (#builder, size) -> time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA54AAAGbCAYAAACs3U99AAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAgAElEQVR4nOzdd1hW9eP/8ecBkaXiwIEbLRURQcU90iytXLkrK83MrI/WL7NsZ8vGp0+amqYtG1au1GybZZobFITEibhwIAiy1/3+/YH5bThQgcN4Pa7LCzic8brhxut+3e9z3scyxiAiIiIiIiJSWJzsDiAiIiIiIiKlm4qniIiIiIiIFCoVTxERERERESlUKp4iIiIiIiJSqFQ8RUREREREpFCVK8qDeXt7m4YNGxblIUVERERERKSIhIaGnjLGVP/n8iItng0bNiQkJKQoDykiIiIiIiJFxLKsg+dbrlNtRUREREREpFCpeIqIiIiIiEihUvEUERERERGRQlWk13ieT3Z2NkeOHCEjI8PuKGIzNzc36tati4uLi91RRERERESkANlePI8cOULFihVp2LAhlmXZHUdsYowhPj6eI0eO4Ovra3ccEREREREpQLafapuRkUG1atVUOss4y7KoVq2aRr5FREREREoh24snoNIpgJ4HIiIiIiKlVbEoniIiIiIiIlJ6qXgCnTp1uuQ6Y8aMYefOnQBMnTr1srevUKHClYUTEREREREp4SxjTJEdLDg42ISEhPxtWVRUFH5+fkWWoSBUqFCBlJSUQt+mLCqJzwcREREREcljWVaoMSb4n8s14sn/jUauWbOG7t27M2TIEJo1a8aIESP4s5h3796dkJAQnnjiCdLT0wkKCmLEiBF/2z4lJYWePXvSunVrAgICWLFihT0PSEREREREpBix/XYqf/XCyj/YGXumQPfZvHYlnu/nn+/1t2/fzh9//EHt2rXp3Lkz69evp0uXLue+/9prrzFr1izCwsL+ta2bmxvLli2jUqVKnDp1ig4dOtC/f39NmiMiIiIiImXaJUc8LctqallW2F/+nbEs6/9ZllXVsqxVlmXtPfuxSlEELmzt2rWjbt26ODk5ERQURExMTL63Ncbw1FNP0bJlS2644QaOHj3KiRMnCi+siIiIiIhICXDJEU9jzG4gCMCyLGfgKLAMeAJYbYx5zbKsJ85+PflqwlzOyGRhcXV1Pfe5s7MzOTk5+d52wYIFxMXFERoaiouLCw0bNtR9KUVEREREpMy73Gs8ewL7jTEHgQHAx2eXfwzcWpDBijMXFxeys7P/tTwpKYkaNWrg4uLCr7/+ysGDB21IJyIiIiIipU1iWhZ7TiTbHeOKXW7xvA344uznNY0xxwDOfqxxvg0syxprWVaIZVkhcXFxV560GBk7diwtW7Y8N7nQn0aMGEFISAjBwcEsWLCAZs2a2ZRQRERERERKi9/2xNF7+loeXLCNXEfR3ZWkIOX7diqWZZUHYgF/Y8wJy7ISjTGV//L908aYi17nWVpupyKFR88HEREREZE8aVk5TP0uis82HaJJzQq8NSyIFnW87I51URe6ncrlzGp7M7DNGPPnbDknLMvyMcYcsyzLBzhZEEFFRERERETKutCDCUxcFM6hhDTGdmvExBub4ObibHesK3Y5xfN2/u80W4CvgZHAa2c/6qaVIiIiIiIiVyEzJ5fpP+9l7m/7qV3ZnS/v60D7RtXsjnXV8lU8LcvyAG4E7v/L4teARZZl3QscAoYWfDwREREREZGyIerYGR5ZGMau48nc1rYez/RtTgXXyxkrLL7y9SiMMWlAtX8siydvllsRERERERG5QrkOw7y10by1ajde7uX5YGQwPf1q2h2rQJWO+iwiIiIiIlICHYxP5dFF4YQcPM0tAbV4+dYAqnqWtztWgVPxFBERERERKWLGGBZsPsTU76Io52Tx9m1B9A+sjWVZdkcrFCqeIiIiIiIiRejEmQweX7KD3/bE0fVab94Y0hIfL3e7YxUqJ7sDlHQxMTG0aNGiyI43f/58xo8fX2THExERERGRgvN1eCy9pq1l84F4Xhrgzyej25X60gka8SwWjDEYY3By0vsAIiIiIiKlUWJaFs8sj+SbHcdoVb8ybw0Lwtfb0+5YRaZ4Fc/vn4DjEQW7z1oBcPNrF/x2amoqw4YN48iRI+Tm5vLss8/SqFEjHn74YVJTU3F1dWX16tXEx8dz1113kZqaCsCsWbPo1KnTJQ8/f/58li1bRmZmJgcOHOCOO+7g+eefJyYmhptvvpkePXqwceNGli9fzoYNG5g6dSrGGPr06cPrr78OwEcffcSrr76Kj48PTZo0wdXVtWB+NiIiIiIiUuh+3X2SyUt2cDoti8d6N+X+bo0o51y2Bp2KV/G0wQ8//EDt2rX59ttvAUhKSqJVq1YsXLiQtm3bcubMGdzd3alRowarVq3Czc2NvXv3cvvttxMSEpKvY2zZsoXIyEg8PDxo27Ytffr0wdvbm927d/PRRx8xe/ZsYmNjmTx5MqGhoVSpUoVevXqxfPly2rdvz/PPP09oaCheXl706NGDVq1aFeaPRERERERECkBqZg6vfBfF55sP0bRmRT66py3+tb3sjmWL4lU8LzIyWVgCAgKYNGkSkydPpm/fvlSuXBkfHx/atm0LQKVKlYC8kdHx48cTFhaGs7Mze/bsyfcxbrzxRqpVy7sN6qBBg/j999+59dZbadCgAR06dABg69atdO/enerVqwMwYsQI1q5dC/C35cOHD7+sY4uIiIiISNELiUlg4qJwDp9O4/7rGjHxxia4lnO2O5ZtilfxtEGTJk0IDQ3lu+++48knn6RXr17nncJ42rRp1KxZk/DwcBwOB25ubvk+xj/39+fXnp7/d063MSbf24uIiIiISPGUmZPLW6v2MG9tNHWruLNwbEfa+Va1O5btytaJxecRGxuLh4cHd955J5MmTWLTpk3ExsaydetWAJKTk8nJySEpKQkfHx+cnJz49NNPyc3NzfcxVq1aRUJCAunp6SxfvpzOnTv/a5327dvz22+/cerUKXJzc/niiy+47rrraN++PWvWrCE+Pp7s7GwWL15cYI9dREREREQKzh+xSQyYtZ65v0VzW9v6fP9wN5XOs8r8iGdERASPPfYYTk5OuLi4MGfOHIwxTJgwgfT0dNzd3fn555958MEHGTx4MIsXL6ZHjx5/G628lC5dunDXXXexb98+7rjjDoKDg4mJifnbOj4+Prz66qv06NEDYwy33HILAwYMAGDKlCl07NgRHx8fWrdufVmlV0RERERECldOroO5a6OZ/vMeKnuU56NRbenRrIbdsYoV62KneBa04OBg888JeaKiovDz8yuyDEVt/vz5hISEMGvWLLujlAil/fkgIiIiIqXLgVOpPLoojG2HEunT0oeXB7Sgimd5u2PZxrKsUGNM8D+Xl/kRTxERERERkctljOGzzYeY+m0ULs4WM25vRf/A2nbHKrZUPAvIjz/+yOTJk/+2zNfXl2XLljFq1Ch7QomIiIiISIE7npTB40t3sHZPHN2aVOeNwS2p5ZX/yUfLIhXPAtK7d296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\n",
      "text/plain": [
       "<Figure size 1152x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "try:\n",
    "    import matplotlib.pyplot as plt\n",
    "    %matplotlib inline\n",
    "except ImportError:\n",
    "    print(\"try install matplotlib: pip install matplotlib\")\n",
    "    raise\n",
    "\n",
    "\n",
    "labels = [ bl for (_, bl) in builders]\n",
    "plt.figure(figsize=(16,7))\n",
    "for b in range(len(builders)):\n",
    "    bts = [res[b,s]/1000 for s in sizes]\n",
    "    plt.plot(sizes, bts, label=labels[b])\n",
    "    plt.legend()\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Average improvement\n",
    "\n",
    "In the next cell, we compute the geometric mean of improvment between the first and last versions. Of course, results may differ depending on platform (and the model, too) but the idea is, applying the above rules may yield a significant improvement"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "* geometric mean of time improvement is 3.5\n"
     ]
    }
   ],
   "source": [
    "# compute geomerical mean for all sizes\n",
    "nb_builders = len(builders)\n",
    "ratios = {}\n",
    "for s in sizes:\n",
    "    initial = res[0, s]\n",
    "    # compute best over all builds\n",
    "    best = min(res[b, s] for b in range(nb_builders))\n",
    "    r = (initial/best)\n",
    "    ratios[s] = r\n",
    "import math\n",
    "rgm = math.exp(sum(math.log(r) for r in ratios.values()) / float(nb_builders))\n",
    "print(\"* geometric mean of time improvement is {0:.1f}\".format(rgm))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "collapsed": true
   },
   "source": [
    "Copyright &copy; 2017-2019 IBM. IPLA licensed Sample Materials."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
