plot_sparsity (generic function with 1 method)

Block-Coordinate Frank-Wolfe and Block-Vectors

In this example, we demonstrate the usage of the FrankWolfe.block_coordinate_frank_wolfe and FrankWolfe.BlockVector. We consider the problem of minimizing the squared Euclidean distance between two sets. We compare different update orders and different update steps.

Import and setup

We first import the necessary packages and include the code for plotting the results.

using FrankWolfe
using LinearAlgebra

include("plot_utils.jl")
plot_sparsity (generic function with 1 method)

Next, we define the objective function and its gradient. The iterates x are instances of the FrankWolfe.BlockVector type. The different blocks of the vector can be accessed via the blocks field.

f(x) = dot(x.blocks[1] - x.blocks[2], x.blocks[1] - x.blocks[2])

function grad!(storage, x)
    @. storage.blocks = [x.blocks[1] - x.blocks[2], x.blocks[2] - x.blocks[1]]
end
grad! (generic function with 1 method)

In our example we consider the probability simplex and an L-infinity norm ball as the feasible sets.

n = 100
lmo1 = FrankWolfe.ScaledBoundLInfNormBall(-ones(n), zeros(n))
lmo2 = FrankWolfe.ProbabilitySimplexOracle(1.0)
prod_lmo = FrankWolfe.ProductLMO((lmo1, lmo2))
FrankWolfe.ProductLMO{2, Tuple{FrankWolfe.ScaledBoundLInfNormBall{Float64, 1, Vector{Float64}, Vector{Float64}}, FrankWolfe.ProbabilitySimplexOracle{Float64}}}((FrankWolfe.ScaledBoundLInfNormBall{Float64, 1, Vector{Float64}, Vector{Float64}}([-1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0  …  -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0, -1.0], [0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0  …  0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0]), FrankWolfe.ProbabilitySimplexOracle{Float64}(1.0)))

We initialize the starting point x0 as a FrankWolfe.BlockVector with two blocks. The two other arguments are the block sizes and the overall number of entries.

x0 = FrankWolfe.BlockVector([-ones(n), [i == 1 ? 1 : 0 for i in 1:n]], [(n,), (n,)], 2 * n);

Running block-coordinate Frank-Wolfe with different update-orders

In a first step, we compare different update orders. There are three different update orders implemented, FrankWolfe.FullUpdate, CyclicUpdate and Stochasticupdate. For creating a custome FrankWolfe.BlockCoordinateUpdateOrder, one needs to implement the function select_update_indices.

struct CustomOrder <: FrankWolfe.BlockCoordinateUpdateOrder end

function FrankWolfe.select_update_indices(::CustomOrder, state::FrankWolfe.CallbackState, dual_gaps)
    return [rand() < 1 / n ? 1 : 2 for _ in 1:length(state.lmo.lmos)]
end

We run the block-coordinate Frank-Wolfe method with the different update orders and store the trajectories.

trajectories = []

for order in [
    FrankWolfe.FullUpdate(),
    FrankWolfe.CyclicUpdate(),
    FrankWolfe.StochasticUpdate(),
    CustomOrder(),
]

    _, _, _, _, traj_data = FrankWolfe.block_coordinate_frank_wolfe(
        f,
        grad!,
        prod_lmo,
        x0;
        verbose=true,
        trajectory=true,
        update_order=order,
    )
    push!(trajectories, traj_data)
end

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.FullUpdate() UPDATE_STEP: DataType[FrankWolfe.FrankWolfeStep, FrankWolfe.FrankWolfeStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
    FW          1000   1.006522e-02   9.790473e-03   2.747434e-04   3.311670e-01   3.019625e+03
    FW          2000   1.002905e-02   9.883695e-03   1.453591e-04   3.534226e-01   5.658947e+03
    FW          3000   1.001690e-02   9.923899e-03   9.300514e-05   3.755876e-01   7.987485e+03
    FW          4000   1.001088e-02   9.940501e-03   7.037823e-05   3.976240e-01   1.005976e+04
    FW          5000   1.000730e-02   9.952154e-03   5.515099e-05   4.196753e-01   1.191397e+04
    FW          6000   1.000507e-02   9.960902e-03   4.416633e-05   4.415930e-01   1.358717e+04
    FW          7000   1.000360e-02   9.967424e-03   3.617261e-05   5.253043e-01   1.332561e+04
    FW          8000   1.000260e-02   9.972504e-03   3.009367e-05   5.481584e-01   1.459432e+04
    FW          9000   1.000190e-02   9.976620e-03   2.528359e-05   5.722612e-01   1.572708e+04
    FW         10000   1.000141e-02   9.979993e-03   2.141696e-05   5.944546e-01   1.682214e+04
  Last         10001   1.000141e-02   9.979981e-03   2.142870e-05   6.958566e-01   1.437221e+04
-------------------------------------------------------------------------------------------------

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.CyclicUpdate(-1) UPDATE_STEP: DataType[FrankWolfe.FrankWolfeStep, FrankWolfe.FrankWolfeStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
    FW          1000   1.006522e-02   9.790473e-03   2.747434e-04   1.219966e-01   8.196947e+03
    FW          2000   1.002905e-02   9.883695e-03   1.453591e-04   2.066274e-01   9.679259e+03
    FW          3000   1.001690e-02   9.923899e-03   9.300514e-05   2.374823e-01   1.263252e+04
    FW          4000   1.001088e-02   9.940501e-03   7.037823e-05   2.660156e-01   1.503671e+04
    FW          5000   1.000730e-02   9.952154e-03   5.515099e-05   2.945144e-01   1.697710e+04
    FW          6000   1.000507e-02   9.960902e-03   4.416633e-05   3.229600e-01   1.857815e+04
    FW          7000   1.000360e-02   9.967424e-03   3.617261e-05   3.512757e-01   1.992737e+04
    FW          8000   1.000260e-02   9.972504e-03   3.009367e-05   3.795654e-01   2.107674e+04
    FW          9000   1.000190e-02   9.976620e-03   2.528359e-05   4.078186e-01   2.206864e+04
    FW         10000   1.000141e-02   9.979993e-03   2.141696e-05   4.978910e-01   2.008472e+04
  Last         10001   1.000141e-02   9.979981e-03   2.142870e-05   4.982775e-01   2.007115e+04
-------------------------------------------------------------------------------------------------

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.StochasticUpdate(-1) UPDATE_STEP: DataType[FrankWolfe.FrankWolfeStep, FrankWolfe.FrankWolfeStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
    FW          1000   1.006649e-02   9.797679e-03   2.688115e-04   6.084894e-02   1.643414e+04
    FW          2000   1.003282e-02   9.887544e-03   1.452767e-04   9.769974e-02   2.047088e+04
    FW          3000   1.001920e-02   9.919860e-03   9.934003e-05   1.263833e-01   2.373731e+04
    FW          4000   1.001224e-02   9.935454e-03   7.678946e-05   2.088665e-01   1.915099e+04
    FW          5000   1.000812e-02   9.949492e-03   5.863302e-05   2.400557e-01   2.082850e+04
    FW          6000   1.000561e-02   9.958403e-03   4.720724e-05   2.686234e-01   2.233610e+04
    FW          7000   1.000400e-02   9.964922e-03   3.907771e-05   2.967139e-01   2.359175e+04
    FW          8000   1.000288e-02   9.971244e-03   3.163293e-05   3.248982e-01   2.462310e+04
    FW          9000   1.000209e-02   9.975610e-03   2.647559e-05   3.537457e-01   2.544200e+04
    FW         10000   1.000153e-02   9.979405e-03   2.212997e-05   3.820177e-01   2.617680e+04
  Last         10001   1.000153e-02   9.979008e-03   2.252590e-05   3.823232e-01   2.615849e+04
-------------------------------------------------------------------------------------------------

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: Main.CustomOrder() UPDATE_STEP: DataType[FrankWolfe.FrankWolfeStep, FrankWolfe.FrankWolfeStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   1.024074e+02           -Inf            Inf   0.000000e+00            Inf
    FW          1000   1.003847e-02   9.875995e-03   1.624779e-04   7.284469e-02   1.372784e+04
    FW          2000   1.001380e-02   9.931922e-03   8.188221e-05   1.112896e-01   1.797114e+04
    FW          3000   1.000624e-02   9.955789e-03   5.044741e-05   1.496087e-01   2.005230e+04
    FW          4000   1.000315e-02   9.969360e-03   3.378772e-05   1.877481e-01   2.130514e+04
    FW          5000   1.000169e-02   9.978459e-03   2.323352e-05   2.844057e-01   1.758052e+04
    FW          6000   1.000095e-02   9.983540e-03   1.740806e-05   3.231213e-01   1.856888e+04
    FW          7000   1.000055e-02   9.987713e-03   1.283319e-05   3.617661e-01   1.934952e+04
    FW          8000   1.000032e-02   9.990836e-03   9.486923e-06   4.001505e-01   1.999248e+04
    FW          9000   1.000019e-02   9.992954e-03   7.238509e-06   4.389514e-01   2.050341e+04
    FW         10000   1.000012e-02   9.994356e-03   5.760023e-06   5.377919e-01   1.859455e+04
  Last         10001   1.000012e-02   9.994357e-03   5.759399e-06   5.381898e-01   1.858266e+04
-------------------------------------------------------------------------------------------------

Plotting the results

labels = ["Full update", "Cyclic order", "Stochstic order", "Custom order"]
plot_trajectories(trajectories, labels, xscalelog=true)

Running BCFW with different update methods

As a second step, we compare different update steps. We consider the FrankWolfe.BPCGStep and the FrankWolfe.FrankWolfeStep. One can either pass a tuple of FrankWolfe.UpdateStep to define for each block the update procedure or pass a single update step so that each block uses the same procedure.

trajectories = []

for us in [(FrankWolfe.BPCGStep(), FrankWolfe.FrankWolfeStep()), (FrankWolfe.FrankWolfeStep(), FrankWolfe.BPCGStep()), FrankWolfe.BPCGStep(), FrankWolfe.FrankWolfeStep()]

    _, _, _, _, traj_data = FrankWolfe.block_coordinate_frank_wolfe(
        f,
        grad!,
        prod_lmo,
        x0;
        verbose=true,
        trajectory=true,
        update_step=us,
    )
    push!(trajectories, traj_data)
end

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.CyclicUpdate(-1) UPDATE_STEP: DataType[FrankWolfe.BPCGStep, FrankWolfe.FrankWolfeStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
    FW          1000   1.006522e-02   9.790473e-03   2.747434e-04   4.971753e-01   2.011363e+03
    FW          2000   1.002905e-02   9.883695e-03   1.453591e-04   5.346716e-01   3.740614e+03
    FW          3000   1.001690e-02   9.923899e-03   9.300514e-05   5.718449e-01   5.246178e+03
    FW          4000   1.001088e-02   9.940501e-03   7.037823e-05   6.651349e-01   6.013817e+03
    FW          5000   1.000730e-02   9.952154e-03   5.515099e-05   7.037838e-01   7.104454e+03
    FW          6000   1.000507e-02   9.960902e-03   4.416633e-05   7.409595e-01   8.097608e+03
    FW          7000   1.000360e-02   9.967424e-03   3.617261e-05   7.781269e-01   8.995962e+03
    FW          8000   1.000260e-02   9.972504e-03   3.009367e-05   8.149748e-01   9.816254e+03
    FW          9000   1.000190e-02   9.976620e-03   2.528359e-05   8.522364e-01   1.056045e+04
    FW         10000   1.000141e-02   9.979993e-03   2.141696e-05   9.523644e-01   1.050018e+04
  Last         10001   1.000141e-02   9.979981e-03   2.142870e-05   9.551900e-01   1.047017e+04
-------------------------------------------------------------------------------------------------

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.CyclicUpdate(-1) UPDATE_STEP: DataType[FrankWolfe.FrankWolfeStep, FrankWolfe.BPCGStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
  Last           474   1.000000e-02   9.999910e-03   8.999181e-08   1.992544e-01   2.378868e+03
-------------------------------------------------------------------------------------------------

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.CyclicUpdate(-1) UPDATE_STEP: DataType[FrankWolfe.BPCGStep, FrankWolfe.BPCGStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
  Last           474   1.000000e-02   9.999910e-03   8.999181e-08   2.702789e-02   1.753744e+04
-------------------------------------------------------------------------------------------------

Block coordinate Frank-Wolfe (BCFW).
MEMORY_MODE: FrankWolfe.InplaceEmphasis() STEPSIZE: DataType[FrankWolfe.Adaptive{Float64, Int64}, FrankWolfe.Adaptive{Float64, Int64}] EPSILON: 1.0e-7 MAXITERATION: 10000 TYPE: Float64
MOMENTUM: nothing GRADIENTTYPE: FrankWolfe.BlockVector{Float64, Vector{Float64}, Tuple{Int64}} UPDATE_ORDER: FrankWolfe.CyclicUpdate(-1) UPDATE_STEP: DataType[FrankWolfe.FrankWolfeStep, FrankWolfe.FrankWolfeStep]
[ Info: In memory_mode memory iterates are written back into x0!

-------------------------------------------------------------------------------------------------
  Type     Iteration         Primal           Dual       Dual Gap           Time         It/sec
-------------------------------------------------------------------------------------------------
     I             1   5.732358e-01  -1.014268e+02   1.020000e+02   0.000000e+00            Inf
    FW          1000   1.006522e-02   9.790473e-03   2.747434e-04   3.057318e-02   3.270841e+04
    FW          2000   1.002905e-02   9.883695e-03   1.453591e-04   1.160158e-01   1.723903e+04
    FW          3000   1.001690e-02   9.923899e-03   9.300514e-05   1.446041e-01   2.074630e+04
    FW          4000   1.001088e-02   9.940501e-03   7.037823e-05   1.730041e-01   2.312083e+04
    FW          5000   1.000730e-02   9.952154e-03   5.515099e-05   2.012618e-01   2.484327e+04
    FW          6000   1.000507e-02   9.960902e-03   4.416633e-05   2.294786e-01   2.614623e+04
    FW          7000   1.000360e-02   9.967424e-03   3.617261e-05   2.578751e-01   2.714492e+04
    FW          8000   1.000260e-02   9.972504e-03   3.009367e-05   2.860709e-01   2.796509e+04
    FW          9000   1.000190e-02   9.976620e-03   2.528359e-05   3.755640e-01   2.396396e+04
    FW         10000   1.000141e-02   9.979993e-03   2.141696e-05   4.043384e-01   2.473176e+04
  Last         10001   1.000141e-02   9.979981e-03   2.142870e-05   4.046997e-01   2.471215e+04
-------------------------------------------------------------------------------------------------

Plotting the results

labels = ["BPCG FW", "FW BPCG", "BPCG", "FW"]
plot_trajectories(trajectories, labels, xscalelog=true)

This page was generated using Literate.jl.