Code Adam Optimization Algorithm From Scratch

Gradient descent is an optimization algorithm that follows the negative gradient of an objective function in order to locate the minimum of the function.

A limitation of gradient descent is that a single step size (learning rate) is used for all input variables. Extensions to gradient descent like AdaGrad and RMSProp update the algorithm to use a separate step size for each input variable but may result in a step size that rapidly decreases to very small values.

The Adaptive Movement Estimation algorithm, or Adam for short, is an extension to gradient descent and a natural successor to techniques like AdaGrad and RMSProp that automatically adapts a learning rate for each input variable for the objective function and further smooths the search process by using an exponentially decreasing moving average of the gradient to make updates to variables.

In this tutorial, you will discover how to develop gradient descent with Adam optimization algorithm from scratch.

After completing this tutorial, you will know:

  • Gradient descent is an optimization algorithm that uses the gradient of the objective function to navigate the search space.
  • Gradient descent can be updated to use an automatically adaptive step size for each input variable using a decaying average of partial derivatives, called Adam.
  • How to implement the Adam optimization algorithm from scratch and apply it to an objective function and evaluate the results.

Kick-start your project with my new book Optimization for Machine Learning, including step-by-step tutorials and the Python source code files for all examples.

Let’s get started.

Gradient Descent Optimization With Adam From Scratch

Gradient Descent Optimization With Adam From Scratch
Photo by Don Graham, some rights reserved.

Tutorial Overview

This tutorial is divided into three parts; they are:

  1. Gradient Descent
  2. Adam Optimization Algorithm
  3. Gradient Descent With Adam
    1. Two-Dimensional Test Problem
    2. Gradient Descent Optimization With Adam
    3. Visualization of Adam

Gradient Descent

Gradient descent is an optimization algorithm.

It is technically referred to as a first-order optimization algorithm as it explicitly makes use of the first-order derivative of the target objective function.

  • First-order methods rely on gradient information to help direct the search for a minimum …

— Page 69, Algorithms for Optimization, 2019.

The first-order derivative, or simply the “derivative,” is the rate of change or slope of the target function at a specific point, e.g. for a specific input.

If the target function takes multiple input variables, it is referred to as a multivariate function and the input variables can be thought of as a vector. In turn, the derivative of a multivariate target function may also be taken as a vector and is referred to generally as the gradient.

  • Gradient: First-order derivative for a multivariate objective function.

The derivative or the gradient points in the direction of the steepest ascent of the target function for a specific input.

Gradient descent refers to a minimization optimization algorithm that follows the negative of the gradient downhill of the target function to locate the minimum of the function.

The gradient descent algorithm requires a target function that is being optimized and the derivative function for the objective function. The target function f() returns a score for a given set of inputs, and the derivative function f'() gives the derivative of the target function for a given set of inputs.

The gradient descent algorithm requires a starting point (x) in the problem, such as a randomly selected point in the input space.

The derivative is then calculated and a step is taken in the input space that is expected to result in a downhill movement in the target function, assuming we are minimizing the target function.

A downhill movement is made by first calculating how far to move in the input space, calculated as the step size (called alpha or the learning rate) multiplied by the gradient. This is then subtracted from the current point, ensuring we move against the gradient, or down the target function.

  • x(t) = x(t-1) – step_size * f'(x(t-1))

The steeper the objective function at a given point, the larger the magnitude of the gradient and, in turn, the larger the step taken in the search space. The size of the step taken is scaled using a step size hyperparameter.

  • Step Size (alpha): Hyperparameter that controls how far to move in the search space against the gradient each iteration of the algorithm.

If the step size is too small, the movement in the search space will be small and the search will take a long time. If the step size is too large, the search may bounce around the search space and skip over the optima.

Now that we are familiar with the gradient descent optimization algorithm, let’s take a look at the Adam algorithm.

Want to Get Started With Optimization Algorithms?

Take my free 7-day email crash course now (with sample code).

Click to sign-up and also get a free PDF Ebook version of the course.

Adam Optimization Algorithm

Adaptive Movement Estimation algorithm, or Adam for short, is an extension to the gradient descent optimization algorithm.

The algorithm was described in the 2014 paper by Diederik Kingma and Jimmy Lei Ba titled “Adam: A Method for Stochastic Optimization.”

Adam is designed to accelerate the optimization process, e.g. decrease the number of function evaluations required to reach the optima, or to improve the capability of the optimization algorithm, e.g. result in a better final result.

This is achieved by calculating a step size for each input parameter that is being optimized. Importantly, each step size is automatically adapted throughput the search process based on the gradients (partial derivatives) encountered for each variable.

We propose Adam, a method for efficient stochastic optimization that only requires first-order gradients with little memory requirement. The method computes individual adaptive learning rates for different parameters from estimates of first and second moments of the gradients; the name Adam is derived from adaptive moment estimation

Adam: A Method for Stochastic Optimization

This involves maintaining a first and second moment of the gradient, e.g. an exponentially decaying mean gradient (first moment) and variance (second moment) for each input variable.

The moving averages themselves are estimates of the 1st moment (the mean) and the 2nd raw moment (the uncentered variance) of the gradient.

Adam: A Method for Stochastic Optimization

Let’s step through each element of the algorithm.

First, we must maintain a moment vector and exponentially weighted infinity norm for each parameter being optimized as part of the search, referred to as m and v (really the Greek letter nu) respectively. They are initialized to 0.0 at the start of the search.

  • m = 0
  • v = 0

The algorithm is executed iteratively over time t starting at t=1, and each iteration involves calculating a new set of parameter values x, e.g. going from x(t-1) to x(t).

It is perhaps easy to understand the algorithm if we focus on updating one parameter, which generalizes to updating all parameters via vector operations.

First, the gradient (partial derivatives) are calculated for the current time step.

  • g(t) = f'(x(t-1))

Next, the first moment is updated using the gradient and a hyperparameter beta1.

  • m(t) = beta1 * m(t-1) + (1 – beta1) * g(t)

Then the second moment is updated using the squared gradient and a hyperparameter beta2.

  • v(t) = beta2 * v(t-1) + (1 – beta2) * g(t)^2

The first and second moments are biased because they are initialized with zero values.

… these moving averages are initialized as (vectors of) 0’s, leading to moment estimates that are biased towards zero, especially during the initial timesteps, and especially when the decay rates are small (i.e. the betas are close to 1). The good news is that this initialization bias can be easily counteracted, resulting in bias-corrected estimates …

Adam: A Method for Stochastic Optimization

Next the first and second moments are bias-corrected, starring with the first moment:

  • mhat(t) = m(t) / (1 – beta1(t))

And then the second moment:

  • vhat(t) = v(t) / (1 – beta2(t))

Note, beta1(t) and beta2(t) refer to the beta1 and beta2 hyperparameters that are decayed on a schedule over the iterations of the algorithm. A static decay schedule can be used, although the paper recommend the following:

  • beta1(t) = beta1^t
  • beta2(t) = beta2^t

Finally, we can calculate the value for the parameter for this iteration.

  • x(t) = x(t-1) – alpha * mhat(t) / (sqrt(vhat(t)) + eps)

Where alpha is the step size hyperparameter, eps is a small value (epsilon) such as 1e-8 that ensures we do not encounter a divide by zero error, and sqrt() is the square root function.

Note, a more efficient reordering of the update rule listed in the paper can be used:

  • alpha(t) = alpha * sqrt(1 – beta2(t)) / (1 – beta1(t))
  • x(t) = x(t-1) – alpha(t) * m(t) / (sqrt(v(t)) + eps)

To review, there are three hyperparameters for the algorithm, they are:

  • alpha: Initial step size (learning rate), a typical value is 0.001.
  • beta1: Decay factor for first momentum, a typical value is 0.9.
  • beta2: Decay factor for infinity norm, a typical value is 0.999.

And that’s it.

For full derivation of the Adam algorithm in the context of the Adam algorithm, I recommend reading the paper.

Next, let’s look at how we might implement the algorithm from scratch in Python.

Gradient Descent With Adam

In this section, we will explore how to implement the gradient descent optimization algorithm with Adam.

Two-Dimensional Test Problem

First, let’s define an optimization function.

We will use a simple two-dimensional function that squares the input of each dimension and define the range of valid inputs from -1.0 to 1.0.

The objective() function below implements this function

We can create a three-dimensional plot of the dataset to get a feeling for the curvature of the response surface.

The complete example of plotting the objective function is listed below.

Running the example creates a three-dimensional surface plot of the objective function.

We can see the familiar bowl shape with the global minima at f(0, 0) = 0.

Three-Dimensional Plot of the Test Objective Function

Three-Dimensional Plot of the Test Objective Function

We can also create a two-dimensional plot of the function. This will be helpful later when we want to plot the progress of the search.

The example below creates a contour plot of the objective function.

Running the example creates a two-dimensional contour plot of the objective function.

We can see the bowl shape compressed to contours shown with a color gradient. We will use this plot to plot the specific points explored during the progress of the search.

Two-Dimensional Contour Plot of the Test Objective Function

Two-Dimensional Contour Plot of the Test Objective Function

Now that we have a test objective function, let’s look at how we might implement the Adam optimization algorithm.

Gradient Descent Optimization With Adam

We can apply the gradient descent with Adam to the test problem.

First, we need a function that calculates the derivative for this function.

  • f(x) = x^2
  • f'(x) = x * 2

The derivative of x^2 is x * 2 in each dimension. The derivative() function implements this below.

Next, we can implement gradient descent optimization.

First, we can select a random point in the bounds of the problem as a starting point for the search.

This assumes we have an array that defines the bounds of the search with one row for each dimension and the first column defines the minimum and the second column defines the maximum of the dimension.

Next, we need to initialize the first and second moments to zero.

We then run a fixed number of iterations of the algorithm defined by the “n_iter” hyperparameter.

The first step is to calculate the gradient for the current solution using the derivative() function.

The first step is to calculate the derivative for the current set of parameters.

Next, we need to perform the Adam update calculations. We will perform these calculations one variable at a time using an imperative programming style for readability.

In practice, I recommend using NumPy vector operations for efficiency.

First, we need to calculate the moment.

Then the second moment.

Then the bias correction for the first and second moments.

Then finally the updated variable value.

This is then repeated for each parameter that is being optimized.

At the end of the iteration we can evaluate the new parameter values and report the performance of the search.

We can tie all of this together into a function named adam() that takes the names of the objective and derivative functions as well as the algorithm hyperparameters, and returns the best solution found at the end of the search and its evaluation.

This complete function is listed below.

Note: we have intentionally used lists and imperative coding style instead of vectorized operations for readability. Feel free to adapt the implementation to a vectorized implementation with NumPy arrays for better performance.

We can then define our hyperparameters and call the adam() function to optimize our test objective function.

In this case, we will use 60 iterations of the algorithm with an initial steps size of 0.02 and beta1 and beta2 values of 0.8 and 0.999 respectively. These hyperparameter values were found after a little trial and error.

Tying all of this together, the complete example of gradient descent optimization with Adam is listed below.

Running the example applies the Adam optimization algorithm to our test problem and reports the performance of the search for each iteration of the algorithm.

Note: Your results may vary given the stochastic nature of the algorithm or evaluation procedure, or differences in numerical precision. Consider running the example a few times and compare the average outcome.

In this case, we can see that a near-optimal solution was found after perhaps 53 iterations of the search, with input values near 0.0 and 0.0, evaluating to 0.0.

Visualization of Adam

We can plot the progress of the Adam search on a contour plot of the domain.

This can provide an intuition for the progress of the search over the iterations of the algorithm.

We must update the adam() function to maintain a list of all solutions found during the search, then return this list at the end of the search.

The updated version of the function with these changes is listed below.

We can then execute the search as before, and this time retrieve the list of solutions instead of the best final solution.

We can then create a contour plot of the objective function, as before.

Finally, we can plot each solution found during the search as a white dot connected by a line.

Tying this all together, the complete example of performing the Adam optimization on the test problem and plotting the results on a contour plot is listed below.

Running the example performs the search as before, except in this case, a contour plot of the objective function is created.

In this case, we can see that a white dot is shown for each solution found during the search, starting above the optima and progressively getting closer to the optima at the center of the plot.

Contour Plot of the Test Objective Function With Adam Search Results Shown

Contour Plot of the Test Objective Function With Adam Search Results Shown

Further Reading

This section provides more resources on the topic if you are looking to go deeper.

Papers

Books

APIs

Articles

Summary

In this tutorial, you discovered how to develop gradient descent with Adam optimization algorithm from scratch.

Specifically, you learned:

  • Gradient descent is an optimization algorithm that uses the gradient of the objective function to navigate the search space.
  • Gradient descent can be updated to use an automatically adaptive step size for each input variable using a decaying average of partial derivatives, called Adam.
  • How to implement the Adam optimization algorithm from scratch and apply it to an objective function and evaluate the results.

Do you have any questions?
Ask your questions in the comments below and I will do my best to answer.

Get a Handle on Modern Optimization Algorithms!

Optimization for Maching Learning

Develop Your Understanding of Optimization

...with just a few lines of python code

Discover how in my new Ebook:
Optimization for Machine Learning

It provides self-study tutorials with full working code on:
Gradient Descent, Genetic Algorithms, Hill Climbing, Curve Fitting, RMSProp, Adam, and much more...

Bring Modern Optimization Algorithms to
Your Machine Learning Projects


See What's Inside

42 Responses to Code Adam Optimization Algorithm From Scratch

  1. Avatar
    Rick Qiu January 13, 2021 at 2:10 pm #

    It is even better if you could use a difficult objective function to test the Adam implementation.

    • Avatar
      Jason Brownlee January 14, 2021 at 6:11 am #

      Great suggestion!

      I wanted to focus on the algorithm in this case.

  2. Avatar
    M. Shobana January 14, 2021 at 4:09 am #

    Is it possible to implement this optimization technique in antennna design

  3. Avatar
    Cindrella January 14, 2021 at 10:18 pm #

    If it’s possible to use emotion classification

    • Avatar
      Jason Brownlee January 15, 2021 at 5:57 am #

      No, you would use a neural network for that task. Adam could be used to train your neural network.

  4. Avatar
    auraham January 15, 2021 at 5:16 am #

    Great article, Jason. I wonder if you discuss about this and other optimization algorithms in any of your books.

    • Avatar
      Jason Brownlee January 15, 2021 at 5:59 am #

      Thanks!

      I hope to write a book on the topic of optimization soon.

  5. Avatar
    Dhanya January 15, 2021 at 5:06 pm #

    Thank you so much..

  6. Avatar
    Elie January 16, 2021 at 2:04 am #

    Hi Jason, the following equation is wrong.

    x(t) = x(t-1) – alpha * mhat(t) / sqrt(vhat(t)) + eps

    Due to the precedence of division over addition, it should read:

    x(t) = x(t-1) – alpha * mhat(t) / (sqrt(vhat(t)) + eps)

    • Avatar
      Jason Brownlee January 16, 2021 at 6:56 am #

      Thanks, agreed!

      Lucky it was just the comment – the code was correct.

  7. Avatar
    AndyLucny January 22, 2021 at 1:07 am #

    I would recommend the following fix:
    # sample input range uniformly at 0.1 increments
    xaxis = arange(bounds[0,0], bounds[0,1]+0.1, 0.1)
    yaxis = arange(bounds[1,0], bounds[1,1]+0.1, 0.1)
    # equalize axis
    pyplot.axis(‘equal’)

  8. Avatar
    Colin Jenkins March 18, 2021 at 9:23 am #

    Hi Jason,

    Do you have an example of Adam with mini-batching. For example would you calculate the 2 moment vectors for each sample and then average them (like the gradients) before the end-of-batch weight update. Or maybe treat each batch as one iteration WRT t. Or something else…?

    • Avatar
      Jason Brownlee March 19, 2021 at 6:11 am #

      I don’t think so.

      Off the cuff, I believe you sum/average the terms across multiple samples.

  9. Avatar
    Novato March 29, 2021 at 8:44 am #

    Hey Jason,

    Thanks for the nice article. I was wondering how you would proceed when there is no analytical solution for the derivative of the objective function? As is the case for many neural networks. How does Adam unfold in this case?

    Thanks again.

    • Avatar
      Jason Brownlee March 30, 2021 at 5:53 am #

      GD requires a gradient.

      If no such gradient is available, you can use a different algorithm that does not require a gradient, so-called direct search algorithms (nelder mead and friends), or even stochastic algorithms (simulated annealing, genetic algorithms and friends).

  10. Avatar
    vipul April 23, 2021 at 7:45 pm #

    How to use adam optimization to design an FIR filter using objective function-

    J2 = ∑ abs [abs (| Η (ω) | -1) -δp] + ∑ [abs (-δs)]

    in Matlab,plz help or give some references

  11. Avatar
    Nishant Gaurav August 7, 2021 at 11:19 am #

    Hey Jason,
    Which book of yours has all the optimization algorithms explained as above?

    • Avatar
      Jason Brownlee August 8, 2021 at 5:07 am #

      There’s no optimization book yet, hopefully soon.

  12. Avatar
    Eitan Mizrahi October 17, 2021 at 7:29 am #

    I can use ADAM optimization for adjusting weights.
    Each weight in in iteration is adjusted.

    There may be an error that is the different between the expected output and the calculated output (which must use an input and a weight for calculating).

    I must know the error, because I need to adjust properly (in gradient I adjust by the error delta).
    I don’t understand how the error delta is corrected for ADAM optimization algorithm. Maybe this is missed by the example.

    • Adrian Tam
      Adrian Tam October 20, 2021 at 9:02 am #

      If I understand correctly, the error correction you mentioned is g(t) in the formula.

  13. Avatar
    Marc January 14, 2022 at 9:37 am #

    I think it’s not \beta (t) but \beta ^t in \hat{m}

    • Avatar
      James Carmichael February 20, 2022 at 12:58 pm #

      Thank you for the feedback Marc! You are correct.

  14. Avatar
    Marc January 14, 2022 at 9:37 am #

    I think it’s not \beta (t) but \beta ^t in \hat{m}.

    • Avatar
      James Carmichael January 15, 2022 at 11:31 am #

      Thank you for the feedback Marc!

  15. Avatar
    Ashutosh gupta May 4, 2022 at 4:50 pm #

    James, A little correction perhaps. Adam stands for Adaptive Moment Estimation (refer the paper), where the moment refers to the statistical moments and not the movement of candidate point in our search space. Although semantically Movement makes equivalent sense 😉

    • Avatar
      James Carmichael May 5, 2022 at 6:26 am #

      Thank you for the feedback Ashutosh! You are correct!

  16. Avatar
    jake July 14, 2022 at 11:49 pm #

    I’m sure if you could learn the mathematics of machine learning you could also have learned basic Latex mathematics notation to have made this and any other blog post on your site many times more readable.

    It’s pretty disappointed, not to mention confusing to read “beta1” and “beta2” on a blog post about machine learning…

    • Avatar
      James Carmichael July 15, 2022 at 8:38 am #

      Thank you for the feedback Jake!

  17. Avatar
    mb-bahreini January 16, 2023 at 8:04 am #

    Hello, thanks for your very good article.
    I need the capture 5 of your book ,” gradient descend method”, for using and referencein my article.
    thank you very much.

    • Avatar
      James Carmichael January 16, 2023 at 8:18 am #

      Hi mb-bahreini…You are very welcome!

      If you use my code or material in your own project, please reference the source, including:

      The Name of the author, e.g. “Jason Brownlee”.
      The Title of the tutorial or book.
      The Name of the website, e.g. “Machine Learning Mastery”.
      The URL of the tutorial or book.
      The Date you accessed or copied the code.
      For example:

      Jason Brownlee, Machine Learning Algorithms in Python, Machine Learning Mastery, Available from https://machinelearningmastery.com/machine-learning-with-python/, accessed April 15th, 2018.

      Also, if your work is public, contact me, I’d love to see it out of general interest.

  18. Avatar
    John Barnes February 2, 2023 at 8:28 am #

    Very informative James, thanks. But I am trying to apply the method to a minimize a discrete function instead of a continuous function that you can differentiate. Is there a way to do that?

  19. Avatar
    Kevin February 16, 2023 at 4:45 am #

    I believe Adam stands for Adaptive Moments (not movements). The paper states: “the name Adam
    is derived from adaptive moment estimation.” (https://arxiv.org/pdf/1412.6980.pdf)

    • Avatar
      James Carmichael February 16, 2023 at 8:30 am #

      Thank you for your feedback Kevin!

  20. Avatar
    luxiyang March 9, 2023 at 2:39 pm #

    score = objective(x[0], x[1]),What do x[0]and x[1] represent respectively

    • Avatar
      James Carmichael March 10, 2023 at 8:05 am #

      Hi luxiyang…They represent the value of x and y passed to the function:

      # objective function def objective(x, y): return x**2.0 + y**2.0

  21. Avatar
    Beginner May 18, 2023 at 3:33 am #

    Would you please add indentations to the code?
    The page is otherwise great but the lack of indentations makes it difficult to read the code.

    • Avatar
      James Carmichael May 18, 2023 at 5:55 am #

      Thank you Beginner for your suggestions and feedback!

Leave a Reply