Add multi-dimensional noise terms
Multi-dimensional stagewise-independent random variables can be created by forming the Cartesian product of the random variables.
A simple example
If the sample space and probabilities are given as vectors for each marginal distribution, do:
julia> model = SDDP.LinearPolicyGraph(
stages = 3,
lower_bound = 0,
optimizer = HiGHS.Optimizer,
) do subproblem, t
@variable(subproblem, x, SDDP.State, initial_value = 0.0)
Ω = [(value = v, coefficient = c) for v in [1, 2] for c in [3, 4, 5]]
P = [v * c for v in [0.5, 0.5] for c in [0.3, 0.5, 0.2]]
SDDP.parameterize(subproblem, Ω, P) do ω
JuMP.fix(x.out, ω.value)
@stageobjective(subproblem, ω.coefficient * x.out)
end
end;
julia> for s in SDDP.simulate(model, 1)[1]
println("ω is: ", s[:noise_term])
end
ω is: (value = 1, coefficient = 4)
ω is: (value = 1, coefficient = 3)
ω is: (value = 2, coefficient = 4)
Using Distributions.jl
For sampling multidimensional random variates, it can be useful to use the Product
type from Distributions.jl.
Finite discrete distributions
There are several ways to go about this. If the sample space is finite, and small enough that it makes sense to enumerate each element, we can use Base.product
and Distributions.support
to generate the entire sample space Ω
from each of the marginal distributions.
We can then evaluate the density function of the product distribution on each element of this space to get the vector of corresponding probabilities, P
.
julia> import Distributions
julia> distributions = [
Distributions.Binomial(10, 0.5),
Distributions.Bernoulli(0.5),
Distributions.truncated(Distributions.Poisson(5), 2, 8)
];
julia> supports = Distributions.support.(distributions);
julia> Ω = vec([collect(ω) for ω in Base.product(supports...)]);
julia> P = [Distributions.pdf(Distributions.Product(distributions), ω) for ω in Ω];
julia> model = SDDP.LinearPolicyGraph(
stages = 3,
lower_bound = 0,
optimizer = HiGHS.Optimizer,
) do subproblem, t
@variable(subproblem, x, SDDP.State, initial_value = 0.0)
SDDP.parameterize(subproblem, Ω, P) do ω
JuMP.fix(x.out, ω[1])
@stageobjective(subproblem, ω[2] * x.out + ω[3])
end
end;
julia> for s in SDDP.simulate(model, 1)[1]
println("ω is: ", s[:noise_term])
end
ω is: [10, 0, 3]
ω is: [0, 1, 6]
ω is: [6, 0, 5]
Sampling
For sample spaces that are too large to explicitly represent, we can instead approximate the distribution by a sample of N
points. Now Ω
is a sample from the full sample space, and P
is the uniform distribution over those points. Points with higher density in the full sample space will appear more frequently in Ω
.
julia> import Distributions
julia> distributions = Distributions.Product([
Distributions.Binomial(100, 0.5),
Distributions.Geometric(1 / 20),
Distributions.Poisson(20),
]);
julia> N = 100;
julia> Ω = [rand(distributions) for _ in 1:N];
julia> P = fill(1 / N, N);
julia> model = SDDP.LinearPolicyGraph(
stages = 3,
lower_bound = 0,
optimizer = HiGHS.Optimizer,
) do subproblem, t
@variable(subproblem, x, SDDP.State, initial_value = 0.0)
SDDP.parameterize(subproblem, Ω, P) do ω
JuMP.fix(x.out, ω[1])
@stageobjective(subproblem, ω[2] * x.out + ω[3])
end
end;
julia> for s in SDDP.simulate(model, 1)[1]
println("ω is: ", s[:noise_term])
end
ω is: [54, 38, 19]
ω is: [43, 3, 13]
ω is: [43, 4, 17]