<!DOCTYPE html>
<html lang="en-us">

  <head>
  <link href="http://gmpg.org/xfn/11" rel="profile">
  <meta http-equiv="content-type" content="text/html; charset=utf-8">

  <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1">

  <title>
    
      The ICLR Blog Track &middot; 
    
  </title>

  
  <link rel="canonical" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/blog/">
  

  <link rel="stylesheet" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/public/css/poole.css">
  <link rel="stylesheet" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/public/css/syntax.css">
  <link rel="stylesheet" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/public/css/lanyon.css">
  <link rel="stylesheet" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/public/css/custom.css">
  <link rel="stylesheet" href="https://fonts.googleapis.com/css?family=PT+Serif:400,400italic,700%7CPT+Sans:400">

  <link rel="apple-touch-icon-precomposed" sizes="144x144" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/public/apple-touch-icon-precomposed.png">
  <link rel="shortcut icon" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/public/favicon.ico">

  <link rel="alternate" type="application/rss+xml" title="RSS" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/atom.xml">

  

  <script src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.1/MathJax.js?config=TeX-AMS-MML_HTMLorMML" type="text/javascript" ></script>
 <!-- <script type="text/x-mathjax-config"> MathJax.Hub.Config({ TeX: { equationNumbers: { autoNumber: "AMS" } } }); </script> -->
  <script type="text/x-mathjax-config">
      MathJax.Hub.Config({
        tex2jax: { inlineMath: [ ['$','$'], ["\\(","\\)"] ],
         processEscapes: false
        }
      });
</script>
</head>


  <body>

    <!-- Target for toggling the sidebar `.sidebar-checkbox` is for regular
     styles, `#sidebar-checkbox` for behavior. -->
<input type="checkbox" class="sidebar-checkbox" id="sidebar-checkbox">
<!-- <input type="checkbox" class="sidebar-checkbox" id="sidebar-checkbox" > -->

<!-- Toggleable sidebar -->
<div class="sidebar" id="sidebar">
  <div class="sidebar-item">
    <p>For short-term, peer-sourced tests of time, generalizations, specializations, reproductions, etc.!</p>
  </div>

  <nav class="sidebar-nav">

    

    
    
      
        
          <a class="sidebar-nav-item" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/">ICLR 2022 Blog Track</a>
        
      
    
      
        
      
    
      
        
          <a class="sidebar-nav-item" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/about/">About</a>
        
      
    
      
    
      
        
      
    
      
        
          <a class="sidebar-nav-item" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/submitting/">Submitting</a>
        
      
    
      
        
          <a class="sidebar-nav-item" href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/tags/">Tags</a>
        
      
    

    <a class="sidebar-nav-item" href="https://github.com/iclr-blog-track/iclr-blog-track.github.io">GitHub project</a>
    <span class="sidebar-nav-item">Currently vICLR Spring 2021</span>
  </nav>

  <div class="sidebar-item">
    <p>
      &copy; 2022. All rights reserved.
    </p>
  </div>
</div>


    <!-- Wrap is the content to shift when toggling the sidebar. We wrap the
         content to avoid any CSS collisions with our real content. -->
    <div class="wrap">
      <div class="masthead">
        <div class="container">
          <h3 class="masthead-title">
            <a href="/" title="Home">The ICLR Blog Track</a>
            <small></small>
          </h3>
        </div>
      </div>

      <div class="container content">
        <div class="posts">
  
  <div >
    <h2 class="post-title">
      <a href="https://iclr.iro.umontreal.ca/1e85d3e5-33b7-4854-9868-77bcb5bb27b0_1643793688/2021/09/01/stochastic-autograd/">
        Stochastic Autograd
      </a>
    </h2>

    <span class="post-date">01 Sep 2021 | 
      <a class="content-tag" href="/tags/#auto-grad"> auto-grad </a>
        
      <a class="content-tag" href="/tags/#backpropagation"> backpropagation </a>
        
      <a class="content-tag" href="/tags/#tutorial"> tutorial </a>
        
    </span>
    <span class="post-date">If you know them, let us know</span>

    <!-- <ul>
  <li><a href="#background">Background</a>
    <ul>
      <li><a href="#manual-calculations">Manual Calculations</a></li>
      <li><a href="#using-programs">Using Programs</a></li>
    </ul>
  </li>
  <li><a href="#automatic-differentiation">Automatic Differentiation</a>
    <ul>
      <li><a href="#chain-rule">Chain Rule</a></li>
      <li><a href="#forward-accumulation">Forward Accumulation</a></li>
      <li><a href="#reverse-accumulation">Reverse Accumulation</a></li>
    </ul>
  </li>
  <li><a href="#automatic-differentiation-in-practice">Automatic Differentiation in Practice</a>
    <ul>
      <li><a href="#jacobians">Jacobians</a></li>
      <li><a href="#jacobian-calculation-as-computational-graph">Jacobian Calculation as Computational Graph</a></li>
      <li><a href="#first-order-optimization-methods">First-order Optimization Methods</a></li>
    </ul>
  </li>
  <li><a href="#here-comes-the-challenger">Here Comes the Challenger</a>
    <ul>
      <li><a href="#implementation">Implementation</a></li>
      <li><a href="#questions">Questions</a></li>
    </ul>
  </li>
  <li><a href="#akcnowledgments">Akcnowledgments</a></li>
  <li><a href="#references">References</a></li>
</ul>

<h2 id="background">Background</h2>

<p>Pioneered by the <a href="https://www.nature.com/articles/323533a0">seminal work of Rumelhart and Hinton in 1986</a>, the majority of current machine learning optimization methods use derivatives and hence there is a pressing need for their efficient calculation.</p>

<h3 id="manual-calculations">Manual Calculations</h3>

<blockquote>
  <p>Implementing backpropagation by hand is like programming in assembly language. You will probably never do it, but it is important for having a mental model of how everything works. $-$ Roger Grosse</p>
</blockquote>

<p>Time to admit something: I knew about neural networks since 2008, but it wasn’t until 2020 when I could implement it by hand (!!). It’s funny how it took a blink of second to realize how easy it was, and I couldn’t agree more with Prof. Grosse about its importance for having a <em>mental model</em> of neural networks training. Call it a good (or bad) luck of my students that since then, I <strong>enforced</strong> them to do backpropagation by hand.</p>

<p>But, we need to caution ourselves against <em>milking</em> the (manual) backpropagation too much and ultimately have to find some elegant computational solutions. While, we are lucky to live (<em>grown up</em>) amongst libraries like PyTorch, JAX, etc., unfortunately, most of the early ML researchers/scientists (for example, <a href="https://leon.bottou.org/publications/pdf/online-1998.pdf">Bottou, 1998 for Stochastic Gradient Descent</a>) didn’t have this luxury and had to go through a slow, laborious (and not guaranteed to be error-free) process of calculating the <em>analytical</em> derivatives.</p>

<h3 id="using-programs">Using Programs</h3>

<p>Programming-based solutions obviously save some effort and time, but they have their own share of errors. We can categorize them into three paradigms:</p>

<ol>
  <li><strong>Symbolic</strong> differentiation</li>
  <li><strong>Numeric</strong> differentiation</li>
  <li><strong>Auto</strong> differentiation</li>
</ol>

<p>Both Symbolic and Numeric differentiation methods are prone to errors, like:</p>

<ul>
  <li>Calculating higher-derivatives is tricky</li>
  <li>Numeric differentiation’s use of discretization results in loss of accuracy</li>
  <li>Symbolic differentiation leads to inefficient code. Actually, Symbolic differentiation is just a way of computing a derivative in a human-readable form, not evaluating it.</li>
  <li>Both are slow at calculating the partial derivatives (a key feature of <strong>gradient-based optimization algorithms</strong>)</li>
</ul>

<h2 id="automatic-differentiation">Automatic Differentiation</h2>

<p>Automatic differentiation addresses all the issues above and is a key feature of the modern ML/DL libraries. Actually, the last (sweeping) statement fails to highlight the fact that merely a decade ago, Autodiff was something alien to the ML community $-$ it may sound bizarre today, but even this <a href="https://justindomke.wordpress.com/2009/02/17/automatic-differentiation-the-most-criminally-underused-tool-in-the-potential-machine-learning-toolbox/">less than 15 years old post</a> was trying to clarify the fundamental differences between numeric and symbolic differentiation.</p>

<h3 id="chain-rule">Chain Rule</h3>

<p>Autodiff centers around the <strong>Chain rule</strong> concept $-$ the fundamental rule in Calculus $-$ to calculate derivatives of the composed functions.</p>

<p>For example,</p>

<p>\[y = 2x^2 +4 ; x = 3w\]</p>

<p>Obviously differentiating <em>y</em> with respect to <em>w</em> (i.e.$\frac{dy}{dw}$) is not possible directly. So it will be calculated indirectly using the Chain rule:</p>

<p>\[\frac {dy}{dw} = \frac {dy}{dx} \times \frac {dx}{dw} = 36w\]</p>

<p>There are a couple of ways to calculate the products using the chain rule $-$ depending on whether we go from inputs to outputs or the other way around.</p>

<h3 id="forward-accumulation">Forward Accumulation</h3>

<p>In forward accumulation, we fix the independent variable and compute gradients recursively (similar to how we calculated $\frac{dy}{dw}$ above).</p>

<p><img src="/public/images/2021-09-01-stochastic-autograd/chain-forward-greek.png" alt="Forward Accumulation (taken from Colah blog)" /></p>

<h3 id="reverse-accumulation">Reverse Accumulation</h3>

<p>Reverse accumulation is the other name for backpropagation. It is implemented in all the major frameworks like <strong>PyTorch</strong> or <strong>Tensorflow</strong> (JAX is even better at <strong>providing both options</strong> at the user’s disposal).</p>

<p><img src="/public/images/2021-09-01-stochastic-autograd/chain-backward-greek.png" alt="Reverse Accumulation (taken from Colah blog)" /></p>

<blockquote>
  <p>Whether to go for forward or reverse accumulation usually depends on the number of features, but reverse accumulation is the <em>de facto</em> method in deep learning.</p>
</blockquote>

<hr />

<h2 id="automatic-differentiation-in-practice">Automatic Differentiation in Practice</h2>

<p>So far, <strong>AutoGrad</strong> sounds like a great choice. Let’s explore it bit more from the Neural Networks point of view.</p>

<h3 id="jacobians">Jacobians</h3>

<p>If we recall, a Jacobian matrix is a counterpart of Gradient for vector-valued functions and defined as:</p>

<p>\[J = \begin{bmatrix} \dfrac{\partial \mathbf{f}}{\partial x_1} &amp; \cdots &amp; \dfrac{\partial \mathbf{f}}{\partial x_n} \end{bmatrix} = \begin{bmatrix} \nabla^\mathsf{T} f_1 \\ \vdots \\ \nabla^\mathsf{T} f_m \end{bmatrix} = \begin{bmatrix} \dfrac{\partial f_1}{\partial x_1} &amp; \cdots &amp; \dfrac{\partial f_1}{\partial x_n}\\ \vdots &amp; \ddots &amp; \vdots\\ \dfrac{\partial f_m}{\partial x_1} &amp; \cdots &amp; \dfrac{\partial f_m}{\partial x_n} \end{bmatrix}\]</p>

<p>Since, neural networks are trained in a vectorized way as well, it can be represented as: $f:\mathcal R^n\to\mathcal R^m$ and consequently, its <strong>differential</strong> (Jacobian) would be:</p>

<p>\[f^{‘}:\mathcal R^n \to \mathcal R^{m\times n}\]</p>

<h3 id="jacobian-calculation-as-computational-graph">Jacobian Calculation as Computational Graph</h3>

<p>Remember, we are using AutoGrad, so there must be a way to find out the <em>nightmarish</em> Jacobians (not <a href="https://en.wikipedia.org/wiki/Jacobin"><em>that Jacobins</em></a>). A neat approach is to use <strong>Linearized Computational Graph (LCG)</strong>, a DAG having vertices as the variables. Figure below (taken from the paper itself) illustrates it better:</p>

<p><img src="/public/images/2021-09-01-stochastic-autograd/paper-fig1.png" alt="LCG" /></p>

<p>Since, a Jacobian is calculated by sum over all paths in the graph, it’s calculation does take time (<em>did someone spell Dynamic Programming here?</em>) and can be done in either forward and reverse modes (for example, <strong><code class="language-plaintext highlighter-rouge">jacfwd()</code></strong> and <strong><code class="language-plaintext highlighter-rouge">jacrev()</code></strong> in JAX).</p>

<h3 id="first-order-optimization-methods">First-order Optimization Methods</h3>

<p>Right from the <a href="https://projecteuclid.org/journals/annals-of-mathematical-statistics/volume-22/issue-3/A-Stochastic-Approximation-Method/10.1214/aoms/1177729586.full">Robbins, Monro (1951)</a>, there has been a realization of stochastic optimization for better training of neural networks. It has been even <a href="https://arxiv.org/abs/1902.00247v2">proven already</a> that SGD can manage to escape the saddle points as well.</p>

<p>It leads to apparently simple, but every thought-provoking question: <em>Why spend resources on <strong>exact gradients</strong> when we’re going to use <strong>stochastic optimization?</strong></em></p>

<h2 id="here-comes-the-challenger">Here Comes the Challenger</h2>

<p>This excellent question was raised by the authors of the paper titled <strong><a href="https://openreview.net/pdf?id=xpx9zj7CUlY">Randomized Automatic Differentiation</a></strong> (ICLR Oral!!! $-$ not your everyday NeurIPS paper).</p>

<p>Well jokes apart, I came across this impressive paper while learning JAX and its AutoGrad. To be honest, I am still surprised by the authors being either unaware of JAX (found no mention of JAX, while Tensorflow and PyTorch found their honorary mentions) or being aware but just not caring to use it (which is an even bigger surprise).</p>

<p>Choice of implemention framework aside, the paper addresses one of the fundamental issues in ML: saving the resources by using inexact gradients (keeping them unbiased).</p>

<h3 id="implementation">Implementation</h3>

<p>Implementation is pretty simple and quoting here in author’s own words:</p>

<blockquote>
  <p>Our main idea is to consider an unbiased estimator $\mathcal J_θ [f]$ such that $EJ_θ [f] = \mathcal J_θ[f]$ which allows us to save memory required for reverse-mode AD. Our approach is to <strong>determine a sparse (but random) linearized computational graph during the forward pass such that reverse-mode AD applied on the sparse graph yields an unbiased estimate of the true gradient</strong>. Note that the original computational graph is used for the forward pass, and randomization is used to determine a LCG to use for the backward pass in place of the original computation graph. We may then decrease memory costs by storing the sparse LCG directly or storing intermediate variables required to compute the sparse LCG.</p>
</blockquote>

<h3 id="questions">Questions</h3>

<p>Since its not meant to be a review or any such thing, I would not go into details of the paper further. Though, there are some questions in my mind after reading it:</p>

<ul>
  <li>How will <em>determining the optimal path</em> in LCG affect the overall performance? How long does it take on average?</li>
  <li>Why only MNIST and CIFAR? Did authors try other datasets? How well can it scale?</li>
  <li>Do we have to perform the exhaustive search for hyperparameters in the future as well or that $10^{-4}$ will do more or less?</li>
  <li>Will a better PRNG (like threefry counter used in JAX) result in better results?</li>
  <li>Can we compare the performance with the existing attempts (like Sun <em>et al.</em> or Gomez <em>et al.</em>)?</li>
</ul>

<p>(<em>This blog is bit incomplete due to time constraints</em>)</p>

<h2 id="akcnowledgments">Akcnowledgments</h2>
<p>We are deeply grateful to Cristopher Olah for his kind permission to use the images from his awesome blog.</p>

<h2 id="references">References</h2>

<ol>
  <li>
    <p><a href="https://colah.github.io/posts/2015-08-Backprop/">Calculus on Computational Graphs: Backpropagation</a> $-$ a recommended read (both images on forward and reverse accumulation are taken from the blog)</p>
  </li>
  <li>
    <p><a href="https://jax.readthedocs.io/en/latest/notebooks/autodiff_cookbook.html">JAX - AutoDiff Cookbook</a></p>
  </li>
  <li>
    <p><a href="https://arxiv.org/pdf/1502.05767v4.pdf">Automatic Differentiation in Machine Learning: a Survey</a></p>
  </li>
</ol>
 -->
    <ul>
  <li><a href="#background">Background</a>
    <ul>
      <li><a href="#manual-calculations">Manual Calculations</a></li>
      <li><a href="#using-programs">Using Programs</a></li>
    </ul>
  </li>
  <li><a href="#automatic-differentiation">Automatic Differentiation</a>
    <ul>
      <li><a href="#chain-rule">Chain Rule</a></li>
      <li><a href="#forward-accumulation">Forward Accumulation</a></li>
      <li><a href="#reverse-accumulation">Reverse Accumulation</a></li>
    </ul>
  </li>
  <li><a href="#automatic-differentiation-in-practice">Automatic Differentiation in Practice</a>
    <ul>
      <li><a href="#jacobians">Jacobians</a></li>
      <li><a href="#jacobian-calculation-as-computational-graph">Jacobian Calculation as Computational Graph</a></li>
      <li><a href="#first-order-optimization-methods">First-order Optimization Methods</a></li>
    </ul>
  </li>
  <li><a href="#here-comes-the-challenger">Here Comes the Challenger</a>
    <ul>
      <li><a href="#implementation">Implementation</a></li>
      <li><a href="#questions">Questions</a></li>
    </ul>
  </li>
  <li><a href="#akcnowledgments">Akcnowledgments</a></li>
  <li><a href="#references">References</a></li>
</ul>

    <hr>
  </div>
  
</div>

<div class="pagination">
  
  <span class="pagination-item older">Older</span>
  
  
  <span class="pagination-item newer">Newer</span>
  
</div>

      </div>
    </div>

    <label for="sidebar-checkbox" class="sidebar-toggle"></label>

    <script src='/public/js/script.js'></script>
  </body>
</html>
