Gradient descent with Python

Introduction

Gradient descent method is a way to find a local minimum of a function. The way it works is we start with an initial guess of the solution and we take the gradient of the function at that point. We step the solution in the negative direction of the gradient and we repeat the process. The algorithm will eventually converge where the gradient is zero (which correspond to a local minimum). Its brother, the gradient ascent, finds the local maximum nearer the current solution by stepping it towards the positive direction of the gradient. They are both first-order algorithms because they take only the first derivative of the function.

Let's say we are trying to find the solution to the minimum of some function J(X). Given some initial value X0 for X, we can change its value in many directions (proportional to the dimension of X: with only one dimension, we can make it higher or lower). To figure out what is the best direction to minimize J, we take the gradient J of it (the derivative along every dimension of X). Intuitively, the gradient will give the slope of the curve at that X and its direction will point to an increase in the function. So we change X in the opposite direction to lower the function value: Xk+1=XkαJ(Xk) The α>0 is a small number that forces the algorithm to make small jumps. That keeps the algorithm stable and its optimal value depends on the function. Given stable conditions (a certain choice of α), it is guaranteed that J(Xk+1)J(Xk).

There is a chronical problem to the gradient descent. For functions that have valleys (in the case of descent) or saddle points (in the case of ascent), the gradient descent/ascent algorithm zig-zags, because the gradient is nearly orthogonal to the direction of the local minimum in these regions. It is like being inside a round tube and trying to stay in the lower part of the tube. In case we are not, the gradient tells us we should go almost perpendicular to the longitudinal direction of the tube. If the local minimum is at the end of the tube, it will take a long time to reach it because we keep jumping between the sides of the tube (zig-zag). The Rosenbrock function is used to test this difficult problem.

Exercise [A "good" example]
We want to find the minimum value of the following function: J(x1,x2)=(x124x1+4)(x12+4x1+2)

  • Calculate the minimum values of this function.
    We calculate the gradient vector of J. It yields: J(x1,x2)=[4x1320x1+80] Consequently, the possible minimum values are 2, 21 or 21.
    We then calculate the hessian matrix: 2J(x1,x2)=[12x1220000] The hessian matrix is only positive definite for x1=2 or x1=21 whatever the value of x2.
  • Draw the level sets of the function. For this, you can use the following Python code:

    
    import numpy as np
    import matplotlib.pyplot as plt
    
    """ Level sets"""
    def g(x1,x2):
        res = (x1**2-4*x1+4)*(x1**2+4*x1+2)
        return res
        
    # level set drawing of the function
    X1,X2 = np.meshgrid(np.linspace(-4,4,401),np.linspace(-3,2.5,401))
    Z = g(X1,X2)
    graphe = plt.contour(X1,X2,Z,120)
    plt.show()
    

  • Define the J and GradJ functions that allows to calculate the gradient of J and the Hess which returns the hessian matrix of J.

    
    import numpy as np
    import matplotlib.pyplot as plt
    
    """ The J, GradJ and Hess functions"""
    def J(X):
        res = (X[0]**2-4*X[0]+4)*(X[0]**2+4*X[0]+2)
        return res
    
    def GradJ(X):
        res = np.array([4*X[0]**3-20*X[0]+8.0, 0.0])
        return res
    
    def Hess(X):
        res = np.array([[#TO DO,#TO DO],[#TO DO,#TO DO]])
        return res
    

  • Gradient Descent with a fixed α coefficient:
    Let X0=(x1,x2)=(1,1) be the initial value. Calculate the Xk values by using the following algorithm:
    k=0
    While J(Xk)>ϵ
    Δ=α.J(Xk)
    Xk+1=Xk+Δ
    k=k+1
    return Xk

    Tune the value of α.

    
    import numpy as np
    from numpy import linalg as LA
    import matplotlib.pyplot as plt
    
    """ The Gradient Descent function"""
    def Gradient(X0,alpha,eps,niter):
        k=0
        while(#TO DO):
            #TO DO
        return x,y
    
    X0=[-1.0,1.0]
    alpha = 0.01
    eps = 0.001
    niter = 150
    x0,y0 = Gradient(X0,lambda,eps,niter)
    plt.plot(x0,y0,'r', label=r"Gradient Method with a fixed $\alpha$", linestyle='--')
    plt.legend(loc='upper left', bbox_to_anchor=(1.1, 0.95),fancybox=True, shadow=True)
    plt.show()
    
  • Draw the Xk values of this algorithm in the level sets graph.


  •     
    """ The Gradient Descent function"""
    def Gradient(X0,alpha,eps,niter):
        k=0
        LX=[X[0]]
        LY=[X[1]]
        while((LA.norm(GradJ(X))>eps and k< niter):
            X=X-np.dot(alpha,GradJ(X))
            k=k+1
            LX.append(X[0])
            LX.append(X[1])
        print("\n iteration number of the fixed step gradient descent method: ",k)
        print("Optimal value x= ",X[0]," y= ",X[1])
        return LX,LY
    
  • Confirm that the value found by the Algorithm is a local minimum value.

  • Gradient Descent with momentum:
    Let X0=(1,1) be the initial value. Calculate the Xk values by using the following momentum Gradient Descent algorithm:
    Δ=0, k=0
    While J(Xk)>ϵ
    Δ=α.J(Xk)+μ.Δ
    Xk+1=Xk+Δ
    k=k+1
    return Xk

    
    """ The Momentum Gradient Descent function"""
    def MGradient(X0,alpha,beta,eps,niter):
        k=0
        while(#TO DO):
            #TO DO
        return x,y
    
    X0=[-1.0,1.0]
    alpha = 0.01
    mu = 0.01
    eps = 0.001
    niter = 150
    x1,y1 = MGradient(X0,alpha,mu,eps,niter)
    plt.plot(x1,y1,'r', label=r"Momentum Gradient Method")
    plt.legend(loc='upper left', bbox_to_anchor=(1.1, 0.95),fancybox=True, shadow=True)
    plt.show()
    


    The μ[0,1[ value has been to be tuned.


  •     
    """ The Momentum Gradient Descent function"""
    def MGradient(X0,lambda,mu,eps,niter):
        k=0
        LX=[X[0]]
        LY=[X[1]]
        Delta=[0.0 for i in range(len(X))]
        while((LA.norm(GradJ(X))>eps and k< niter):
            Delta=-np.dot(alpha,GradJ(X))+np.dot(beta,Delta)
            X=np.add(X,Delta)
            k=k+1
            LX.append(X[0])
            LX.append(X[1])
        print("\n iteration number of the gradient descent with momentum: ",k)
        print("Optimal value x= ",X[0]," y= ",X[1])
        return LX,LY
    
  • Draw the Xk values of this algorithm in the level sets graph.

  • Confirm that the value found by the Algorithm is a local minimum value.

We obtain:

  • 7 iterations of the Gradient Descent with α=0.02 give:
    x= -2.4141942582083296 and y= 1.0
  • 12 iterations of the Momentum Gradient with α=0.02 and μ=0.06 give:
    x= -2.4142172250611096 and y= 1.0

Exercise [The Rosenbrock function]
We want to find the minimum value of the following function: J(x1,x2)=(x11)2+10(x12x2)2
  • Calculate the minimum value of this function.
  • Draw the level sets of this function.

  • Apply the previous Gradient Descent methods

  • Apply the Newton-Raphson Algorithm to this function.

    
    
    """ The Newton-Raphson function"""
    def Newton(X0,eps,niter):
        k=0
        x=[X[0]]
        y=[X[1]]
        while(LA.norm(GradJ(X))>eps and k< niter):
            delta=#TO DO
            X=X-delta
            k=k+1
            x.append(X[0])
            y.append(X[1])
        return x,y
    
    X0=[-1.0,1.0]
    eps = 0.001
    niter = 150
    x3,y3 = Newton(X0,eps,niter)
    plt.plot(x3,y3,'r', label=r"Newton-Raphson Method")
    plt.legend(loc='upper left', bbox_to_anchor=(1.1, 0.95),fancybox=True, shadow=True)
    plt.show()
    

    
    import numpy as np
    
    """ GradJ and Hess functions"""
    
    def GradJ(X):
        res = np.array([2*(X[0]-1)+40*X[0]*(X[0]**2-X[1]),-20*(X[0]**2-X[1])])
        return res
    
    def Hess(X):
        res = np.array([[2 + 80*X[0]**2 + 40*(X[0]**2 - X[1]),-40*X[0]],[-40*X[0], 20]])
        return res

    """ The Newton-Raphson function"""
        delta=LA.solve(Hess(X),GradJ(X))
    

We obtain:

  • 150 iterations of the Gradient Descent with α=0.02 give:
    x= 0.8467914701456682 y= 0.7102982431003074
  • 150 iterations of the Momentum Gradient with α=0.02 and μ=0.06 give:
    x= 0.8610575985931694 y= 0.73534414819584
  • 2 iterations of the Newton Raphson method give:
    x= 0.9999999999999982 y= 0.9999999999999964

Estimating the step-size

A wrong step size α may not reach convergence, so a careful selection of the step size is important. Too large it will diverge, too small it will take a long time to converge. One option is to choose a fixed step size that will assure convergence wherever you start gradient descent. Another option is to choose a different step size at each iteration (adaptive step size).

Any differentiable function has a maximum derivative value, i.e., the maximum of the derivatives at all points. If this maximum is not infinite, this value is known as the Lipschitz constant and the function is Lipschitz continuous. f(x)f(y)xyL(f), for any x,y This constant is important because it says that, given a certain function, any derivative will have a smaller value than the Lipschitz constant. The same can be said for the gradient of the function: if the maximum second derivative is finite, the function is Lipschitz continuous gradient and that value is the Lipschitz constant of f. f(x)f(y)xyL(f), for any x,y For the f(x)=x2 example, the derivative is df(x)dx=2x and therefore the function is not Lipschitz continuous. But the second derivative is d2f(x)dx2=2, and the function is Lipschitz continuous gradient with Lipschitz constant of f=2.

Each gradient descent can be viewed as a minimization of the function: xk+1=argminx(f(xk)+(xxk)Tf(xk)+12αxxk2) If we differentiate the equation with respect to x, we get: 0=f(xk)+1α(xxk) x=xkαf(xk) It can be shown that for any α1L(f): f(x)f(xk)+(xxk)Tf(xk)+12αxxk2 i.e., for any x, f(x) will always be equal or lower than the function that the gradient descent minimizes, if α1L(f).
Under this view, the result we get from minimizing the right hand side is an upper bound to the real value of the function at x. In our good example above, L(f)=2, then α0.5 for good convergence. In fact, for this simple case, α=0.5 is a perfect value for the step size: smaller will converge slower, larger will exceed the optimum point at each iteration (although it still converges up to α=1).

Adaptive step size

There are methods, known as line search, that make an estimate of what the step size should be at a given iteration. After calculating the gradient, these methods choose a step size by minimizing a function of the step size itself.
Each method defines its own function, based on some assumptions. Exact methods accurately minimize this function, while inexact methods make an approximation that just improves on the last iteration. The following are just a few examples.

Cauchy (in 1847)

One of the most obvious choices of α is to choose the one that minimizes the objective function: αk=argminαf(xkαf(xk)) This approach is conveniently called the steepest descent method. Although it seems to the best choice, it converges only linearly (error 1/k) and is very sensitive to ill-conditioning of problems.

Barzilai and Borwein

An approach proposed in 1988 is to find the step size that minimizes: αk=argminαΔxαΔg(x)2 where Δx=xkxk1 and Δg(x)=f(xk)f(xk1).
This is an approximation to the secant equation, used in quasi-Newton methods. The solution to this problem is easily solved differentiating with respect to α: 0=Δg(x)T(ΔxαΔg(x)) α=Δg(x)TΔxΔg(x)TΔg(x) This approach works really well, even for large dimensional problems.

Exercise [Adaptative Steps]

We want to minimize the following problem: J(x)=12xTAxbTx where A=(200000100000200001)b=(1111) Note that 2(f)=A.
  1. Show that the steepest-descent (or Cauchy) step is given by αk=gkTgkgkTAgk where gk=(J(xk)). Hence define the OptimalGradient function in this case.
  2. Define the Barzilai and Borwein Gradient Descent function BorweinGradient
  3. Compare these two methods.

Descent Lemma

In Proposition A.24 from Bertsekas 1999 we have the following result:

Let f:RnR be continuously differentiable, and let x and y be two vectors in Rn. Suppose that f(x+ty)f(x)Lty,t[0,1] where L is some scalar. Then f(x+y)f(x)+yTf(x)+L2y22
f(x+y)f(x)=f(x+ty)|t=1f(x+ty)|t=0 The two terms can be seen as the limits of an integral: f(x+ty)|t=1f(x+ty)|t=0=01df(x+ty)dtdt=01yTf(x+ty)dt Let's add positive and negative versions of the same term: 01yTf(x+ty)dt=01yTf(x+ty)yTf(x)dt+01yTf(x)dt and take the norm of the first term, which guarantees the result is equal or greater than the original: f(x+y)f(x)01yT(f(x+ty)f(x))2dt+01yTf(x)dt Now we make use of the Hölder's inequality |xTy|xy: f(x+y)f(x)y201f(x+ty)f(x)2dt+yTf(x)01dt Now we make use of the condition in the lemma to simplify the inequality: f(x+y)f(x)y2L01ty2dt+yTf(x)=L2y22+yTf(x).

Optimization with constraint equalities

Exercise [Equality-constrained problem]

We want to minimize the following problem: minx1,x2RJ(X)=x1x2 such that g(x1,x2)=x12+x221=0
  1. By using the Lagrange framework, calculate the four possible extremal (or stationnary points) values of this problem. Show that the Lagrange function is convex.
  2. Plot the contours of J and the contour line of the constraint g.
  3. Apply the Gradient function on the Lagrange function to find the global minimum of the problem.

References

  1. D. P. Bertsekas (1999), Nonlinear Programming.
  2. J. Barzilai, J.M. Borwein (1988), Two-Point Step Size Gradient Methods, IMA Journal of Numerical Analysis (8), pp. 141-148.
  3. A. Cauchy (1847), Méthode générale pour la résolution des systèmes d'équations simultanées, Comptes Rendus Académie des Sciences Paris 25, pp. 536-538.