from BPTK_Py import Model
from BPTK_Py.bptk import bptk
= bptk()
testbptk = Model(starttime = 0.0, stoptime= 15.0, dt= 1.0, name="TestModel") model
Creating multidimensional SD Models
Multidimensional SD Models
This document illustrates how vector- or matrix-valued SD Models can be defined.
We start with some boilerplate to get a BPTK project up and running:
This is already enough to define arrayed components.
How to define arrayed components
There are two options for arrayed components: - Vectors (one dimensional arrays) - Matrices (two dimensional arrays)
Moreover, both types of arrays - Vectors and Matrices - can be setup: - using numerical indices - using string-valued indices (named arrays)
Lets have a look at some examples:
## Defining a Vector (with numerical indices)
# Define an sd dsl element
= model.converter("vector1")
vector1
# Create a vector of length 2 with different values
2, [2.0, 3.0])
vector1.setup_vector(
# Create a vector of length 2 with identical values
2, 3.0) vector1.setup_vector(
## Defining a named Vector (with string-valued indices)
# Define a sd dsl element
= model.converter("vector2")
vector2
# Create a named vector of length 2 using string-valued indices
"value1": 4.0, "value2": 5.0}) vector2.setup_named_vector({
As can be seen, we need two parameters for setting up a Vector using numerical indices. Moreover, there is one optional parameter.
Parameter | Tye | Meaning |
---|---|---|
size | Integer | Defines the length of the Vector |
values | List of Float/Integer | Defines the values of the Vector elements |
set_stack_equation | Boolean | (optional) If the element is a stock, the initial value is set (False) or the equation is set (True). Default is False. |
And we need one parameter (+ one optional parameter) for setting up a Vector using string indices:
Parameter | Tye | Meaning |
---|---|---|
Values | Dictionary | Defines the string-values indices and their values |
set_stack_equation | Boolean | (optional) If the element is a stock, the initial value is set (False) or the equation is set (True). Default is False. |
For matrices, we can proceed completely similar.
## Defining a Matrix (with numerical indices)
# Define a sd dsl element
= model.converter("matrix1")
matrix1
# Create a matrix of size 2x2 with different values
2,2], [[2.0, 3.0],
matrix1.setup_matrix([4.0, 5.0]]) [
## Defining a named Matrix (with string-valued indices)
# Define a sd dsl element
= model.converter("matrix2")
matrix2
# Create a named vector of lenght 2 using string-valued indices
"value1": {"value11": 2.0, "value12": 3.0},
matrix2.setup_named_matrix({"value2": {"value21": 4.0, "value22": 5.0}})
As can be seen, we need two parameters (+ one optional parameter) for setting up a Matrix using numerical indices:
Parameter | Tye | Meaning |
---|---|---|
size | List (tuple) of Integer | Defines the size of the Matrix |
values | List of Lists of Float/Integer | Defines the values of the Vector elements |
set_stack_equation | Boolean | (optional) If the element is a stock, the initial value is set (False) or the equation is set (True). Default is False. |
And we need one parameter (+ one optional parameter) for setting up a Matrix using string-valued indices:
Parameter | Tye | Meaning |
---|---|---|
Values | Dictionary | Defines the string-values indices and their values |
set_stack_equation | Boolean | (optional) If the element is a stock, the initial value is set (False) or the equation is set (True). Default is False. |
Math-Operations for arrayed Components
The standard Operations (+, -, *, %) can be used for arrayed Components. Moreover some array-specific Operations are provided.
Standard Operations
It is possible to use the standard Operations (+, -, *, %) for:
Operand 1 | Operand 2 |
---|---|
Arrayed Element | Arrayed Element |
Arrayed Element | Float/Integer |
Float/Integer | Arrayed Element |
⚠️ If both operands are arrayed elements, they must have the same numerical/string-valued indices.
That means it is not possible to have operand 1 = vector with numerical indices and operand 2 = vector with string-valued indicies, even if they have the same size.
It is also not possible to have operand 1 = vector and operand 2 = matrix or vice versa.
Other standard Operations (**, //, %, …) are not supported yet.
Standard Operations are always performed element-wise. Lets have a look at some examples:
Addition ( )
#Add not-named vectors
= model.converter("vectorAdd1")
vectorAdd1 2, [1.1, 2.2])
vectorAdd1.setup_vector(
= model.converter("vectorAdd2")
vectorAdd2 2, [3.1, 4.2])
vectorAdd2.setup_vector(
= model.converter("addResult")
addResult = vectorAdd1 + vectorAdd2
addResult.equation
print("[ " + str(addResult[0](0)) + " , " + str(addResult[1](0)) + " ]")
[ 4.2 , 6.4 ]
#Add not-named vector and numerical element
= vectorAdd2 + 1.0
addResult.equation
print("[ " + str(addResult[0](0)) + " , " + str(addResult[1](0)) + " ]")
[ 4.1 , 5.2 ]
Subtraction ( )
#Subtract not-named matrices
= model.converter("matrixMinus1")
matrixMinus1 2,2], [[1.1, 2.2],
matrixMinus1.setup_matrix([3.3, 4.4]])
[
= model.converter("matrixMinus2")
matrixMinus2 2,2], [[5.5, 7.7],
matrixMinus2.setup_matrix([3.3, 14.4]])
[
= model.converter("minusResult")
minusResult = matrixMinus1 - matrixMinus2 minusResult.equation
print("[ " + "[" + str(minusResult[0][0](1)) + " , " + str(minusResult[0][1](1)) + "]")
print(" " + "[" + str(minusResult[1][0](1)) + " , " + str(minusResult[1][1](1)) + "]" + " ]")
[ [-4.4 , -5.5]
[0.0 , -10.0] ]
#Subtract not-named matrix and numerical element
= matrixMinus2 - 1.0 minusResult.equation
print("[ " + "[" + str(minusResult[0][0](1)) + " , " + str(minusResult[0][1](1)) + "]")
print(" " + "[" + str(minusResult[1][0](1)) + " , " + str(minusResult[1][1](1)) + "]" + " ]")
[ [4.5 , 6.7]
[2.3 , 13.4] ]
Multiplication ( )
#Multiply named vectors
= model.converter("vectorTimes1")
vectorTimes1 "value1": 4.0, "value2": 5.0})
vectorTimes1.setup_named_vector({
= model.converter("vectorTimes2")
vectorTimes2 "value1": 6.0, "value2": 7.0})
vectorTimes2.setup_named_vector({
= model.converter("timesResult")
timesResult = vectorTimes1 * vectorTimes2
timesResult.equation
print("[ " + str(timesResult["value1"](0)) + " , " + str(timesResult["value2"](0)) + " ]")
[ 24.0 , 35.0 ]
#Multiplay named vector and numerical element
= vectorTimes2 * 3.0 timesResult.equation
print("[ " + str(timesResult["value1"](0)) + " , " + str(timesResult["value2"](0)) + " ]")
[ 18.0 , 21.0 ]
The case “- arrayed element” is a special case since it is interpreted as “(-1)
##Works after merge
#Multiplay named vector and numerical element
#timesResult = model.converter("timesResult")
#timesResult.equation = -vectorTimes2
#print("[ " + str(timesResult["value1"](0)) + " , " + str(timesResult["value2"](0)) + " ]")
Division ( )
#Divide not-named matrices
= model.converter("matrixDivide1")
matrixDivide1 2,2], [[2.0, 4.0],
matrixDivide1.setup_matrix([8.0, 16.0]])
[
= model.converter("matrixDivide2")
matrixDivide2 2,2], [[2.0, 1.0],
matrixDivide2.setup_matrix([0.5, 0.25]])
[
= model.converter("divideResult")
divideResult = matrixDivide1 / matrixDivide2 divideResult.equation
print("[ " + "[" + str(divideResult[0][0](1)) + " , " + str(divideResult[0][1](1)) + "]")
print(" " + "[" + str(divideResult[1][0](1)) + " , " + str(divideResult[1][1](1)) + "]" + " ]")
[ [1.0 , 4.0]
[16.0 , 64.0] ]
#Divide not-named matrix and numerical element
= matrixDivide2 / 5.0 divideResult.equation
print("[ " + "[" + str(divideResult[0][0](1)) + " , " + str(divideResult[0][1](1)) + "]")
print(" " + "[" + str(divideResult[1][0](1)) + " , " + str(divideResult[1][1](1)) + "]" + " ]")
[ [0.4 , 0.2]
[0.1 , 0.05] ]
Array-specific Operations
Array Sum
Calculates the element-wise sum of an array.
#Calculate the element-wise sum of a named-vector
= model.converter("vectorSum")
vectorSum "value1": 1.0, "value2": 2.0, "value3": 3.0})
vectorSum.setup_named_vector({
= model.converter("sumResult")
sumResult = vectorSum.arr_sum()
sumResult.equation
print(sumResult(1))
6.0
Array Product
Calculates the element-wise product of an array.
#Calculate the element-wise product of a not-named-matrix
= model.converter("matrixProd")
matrixProd 2,3],[[2.0, 3.0, 4.0],
matrixProd.setup_matrix([5.0, 6.0, 7.0]])
[
= model.converter("prodResult")
prodResult = matrixProd.arr_prod()
prodResult.equation
print(prodResult(1))
5040.0
Array Rank
Calculates the
For
#Calculate the highest elements of a not-named vector
= model.converter("vectorRank")
vectorRank 5, [-2.0, -0.1, 3.1, 5.2, 11.1])
vectorRank.setup_vector(
= model.converter("rankResult")
rankResult = vectorRank.arr_rank(1) rankResult.equation
print(rankResult(1))
11.1
= vectorRank.arr_rank(4) rankResult.equation
print(rankResult(1))
-0.1
= vectorRank.arr_rank(-1) rankResult.equation
print(rankResult(1))
-2.0
Array Mean
Calculates the element-wise mean of an array.
#Calculate the element-wise mean of a named matrix
= model.converter("matrixMean")
matrixMean "value1": {"value11": 2.0, "value12": 4.0},
matrixMean.setup_named_matrix({"value2": {"value21": 6.0, "value22": 8.0},
"value3": {"value31": 10.0, "value32": 12.0}})
= model.converter("meanResult")
meanResult = matrixMean.arr_mean() meanResult.equation
print(meanResult(1))
7.0
Array Median
Calculates the element-wise median of an array.
#Calculate the median of a not-named vector
= model.converter("vectorMedian1")
vectorMedian1 5, [-2.0, -0.1, 3.1, 5.2, 11.1])
vectorMedian1.setup_vector(= model.converter("vectorMedian2")
vectorMedian2 4, [-2.0, -0.1, 3.1, 5.2])
vectorMedian2.setup_vector(
= model.converter("medianResult")
medianResult = vectorMedian1.arr_median()
medianResult.equation
print(medianResult(1))
3.1
= vectorMedian2.arr_median() medianResult.equation
print(medianResult(1)) #Median of vectorMedian2 is (-0.1 + 3.1)/2 = 1.5
1.5
Array Standdarddeviation
Calculates the element-wise standard deviation of an array.
#Calculate the standard deviation of a not-named matrix
= model.converter("matrixStddev")
matrixStddev 2,2], [[1.0, 3.0],
matrixStddev.setup_matrix([3.0, 1.0]])
[
= model.converter("stddevResult")
stddevResult = matrixStddev.arr_stddev()
stddevResult.equation
print(stddevResult(1))
1.0
Array Size
Calculates the size of an array.
For a vector, the length will be returned. For a matrix, the size of the highest level will be returned (for example 2 for a
#Calculate the size of a not-named vector
= model.converter("vectorSize")
vectorSize 6, [1.0, 1.0, 1.0, 1.0, 1.0, 1.0])
vectorSize.setup_vector(
#Calculate the size of a not-named matrix
= model.converter("matrixSize2")
matrixSize1 2,3], [[1.0, 1.0, 1.0],
matrixSize1.setup_matrix([1.0, 1.0, 1.0]])
[= model.converter("matrixSize3")
matrixSize2 4,3], [[1.0, 1.0, 1.0],
matrixSize2.setup_matrix([1.0, 1.0, 1.0],
[1.0, 1.0, 1.0],
[1.0, 1.0, 1.0]])
[
= model.converter("sizeResult")
sizeResult = vectorSize.arr_size() sizeResult.equation
print(sizeResult(1))
6
= matrixSize1.arr_size() sizeResult.equation
print(sizeResult(1))
2
= matrixSize2.arr_size() sizeResult.equation
print(sizeResult(1))
4
Array Dot
The Dot function provides the classical vector/matrix-multiplication logic. That means, the following can be calculated:
Factor 1 | Factor 2 | Result |
---|---|---|
Vector of size |
Constant | Vector of size |
Constant | Vector of size |
Vector of size |
Matrix of size |
Constant | Matrix of size |
Constant | Matrix of size |
Matrix of size |
Vector of size |
Vector of size |
Value (Scalar Product) |
Vector of size |
Matrix of size |
Vector of size |
Matrix of size |
Vector of size |
Vector of size |
Matrix of size |
Matrix of size |
Matrix of size |
❗ Using the Dot function for an array and a constant yields the same result as using the
If the dimensions of the arrays to which the dot function is applied do not allow for a valid array multiplication, an exception is thrown.
⚠️ The Dot function is currently supported for not-named arrays only!
Lets have a look at some examples:
#Calculate vector * constant & constant * vector
= model.converter("constant")
constant = 2.0
constant.equation
= model.converter("vectorDot1")
vectorDot1 3,[1.0, 2.0, 3.0])
vectorDot1.setup_vector(= model.converter("vectorDot2")
vectorDot2 3,[4.0, 5.0, 6.0])
vectorDot2.setup_vector(
= model.converter("dotResult1")
dotResult1
= vectorDot1.dot(constant) dotResult1.equation
print("[" + str(dotResult1[0](1)) + " , " + str(dotResult1[1](1)) + " , " + str(dotResult1[2](1)) + "]")
[2.0 , 4.0 , 6.0]
= constant.dot(vectorDot2) dotResult1.equation
print("[" + str(dotResult1[0](1)) + " , " + str(dotResult1[1](1)) + " , " + str(dotResult1[2](1)) + "]")
[8.0 , 10.0 , 12.0]
#Calculate matrix * constant & constant * matrix
= model.converter("constant")
constant = 2.0
constant.equation
= model.converter("matrixDot1")
matrixDot1 3,2],[[1.0, 2.0],
matrixDot1.setup_matrix([3.0, 4.0],
[5.0, 6.0]])
[= model.converter("matrixDot2")
matrixDot2 2,3],[[-1.0, -2.0, -3.0],
matrixDot2.setup_matrix([-4.0, -5.0, -6.0]])
[
= model.converter("dotResult2")
dotResult2
= matrixDot1.dot(constant) dotResult2.equation
print("[ " + "[" + str(dotResult2[0][0](1)) + " , " + str(dotResult2[0][1](1)) + "]")
print(" " + "[" + str(dotResult2[1][0](1)) + " , " + str(dotResult2[1][1](1)) + "]")
print(" " + "[" + str(dotResult2[2][0](1)) + " , " + str(dotResult2[2][1](1)) + "]" + " ]")
[ [2.0 , 4.0]
[6.0 , 8.0]
[10.0 , 12.0] ]
= constant.dot(matrixDot2) dotResult2.equation
print("[ " + "[" + str(dotResult2[0][0](1)) + " , " + str(dotResult2[0][1](1)) + " , "
+ str(dotResult2[0][2](1)) + "]")
print(" " + "[" + str(dotResult2[1][0](1)) + " , " + str(dotResult2[1][1](1)) + " , "
+ str(dotResult2[1][2](1)) + "]" + " ]")
[ [-2.0 , -4.0 , -6.0]
[-8.0 , -10.0 , -12.0] ]
#Calculate vector * vector
= model.converter("dotResult3")
dotResult3
= vectorDot1.dot(vectorDot2) dotResult3.equation
print(dotResult3(1)) #1*4 + 2*5 + 3*6 = 32
32.0
#Calculate vector * matrix & matrix * vector
= model.converter("dotResult4")
dotResult4
= vectorDot1.dot(matrixDot1) dotResult4.equation
print("[" + str(dotResult4[0](1)) + " , " + str(dotResult4[1](1)) + "]")
[22.0 , 28.0]
= matrixDot2.dot(vectorDot2) dotResult4.equation
print("[" + str(dotResult4[0](1)) + " , " + str(dotResult4[1](1)) + "]")
[-32.0 , -77.0]
#Calculate matrix * matrix
= model.converter("matrixDot3")
matrixDot3 2,2],[[1.0, 2.0],
matrixDot3.setup_matrix([3.0, 4.0]])
[= model.converter("matrixDot4")
matrixDot4 2,2],[[-1.0, -2.0],
matrixDot4.setup_matrix([-4.0, -5.0]])
[
= model.converter("dotResult5")
dotResult5 = matrixDot3.dot(matrixDot4) dotResult5.equation
print("[ " + "[" + str(dotResult5[0][0](1)) + " , " + str(dotResult5[0][1](1)) + "]")
print(" " + "[" + str(dotResult5[1][0](1)) + " , " + str(dotResult5[1][1](1)) + "]" + " ]")
[ [-9.0 , -12.0]
[-19.0 , -26.0] ]
Plotting arrayed Components
Similar to one-dimensional SD DSL elements, we can also plot these elements. Lets have a look:
= model.converter("vector3")
vector3 "value1": 6.0, "value2": 7.0})
vector3.setup_named_vector({ vector3.plot()
As can be seen, both elements of the Vector are plotted.
If you want to plot the values of the Matrix, you need to specify the first index.
= model.converter("matrix3")
matrix3 "value1": {"value11": 6.0, "value12": 7.0},
matrix3.setup_named_matrix({"value2": {"value21": 8.0, "value22": 9.0}})
"value1"].plot() matrix3[
A simple Example
Lets have a look on a concrete example, how a multidimensional SD Model can look like.
Consider an investment depot with two accounts: - bank account - depot account
Both accounts will have different deposit rates and different interest rates each year. We want to investigate the value development of the bank account, the depot account and the whole investment depot.
Lets set up the model:
= model.stock("account")
account "bank" : 0.0, "depot": 0.0})
account.setup_named_vector({
#define the initial values of the accounts
= model.constant("accountInitialValues")
accountInitialValues "bank" : 1000.0, "depot": 500.0})
accountInitialValues.setup_named_vector({
"bank"].initial_value = accountInitialValues["bank"]
account["depot"].initial_value = accountInitialValues["depot"]
account[##Does not work yet:
#account.initial_value = accountInitialValues
#define the interest rates
= model.constant("interestRate")
interestRate "bank" : 0.02, "depot": 0.1})
interestRate.setup_named_vector({
#define the deposit rates
= model.constant("depositRate")
depositRate "bank" : 200.0, "depot": 100.0})
depositRate.setup_named_vector({
#define the flows
= model.flow("deposit")
deposit = depositRate * 1
deposit.equation
= model.flow("interest")
interest = account * interestRate
interest.equation
#set the equation for the stock value
= deposit + interest
account.equation
#finally define a converter for the total value of the account
= model.converter("totalValue")
totalValue = account.arr_sum() totalValue.equation
As always we define a scenario manager and scenarios:
testbptk.register_model(model)= {
scenario_manager "sm": {
"model": model,
"base_constants": {
"interestRate[bank]": 0.02,
"interestRate[depot]": 0.1,
"depositRate[bank]": 200,
"depositRate[depot]": 100,
"accountInitialValues[bank]": 1000.0,
"accountInitialValues[depot]": 500.0
},
}
}
testbptk.register_scenario_manager(scenario_manager)
testbptk.register_scenarios(="sm",
scenario_manager=
scenarios
{"base":{},
"scenarioHighDepotInterestRate":{
"constants": {
"interestRate[depot]": 0.2
}
},"scenarioHighDepotDepositRate":{
"constants": {
"depositRate[depot]": 250.0
}
},"scenarioHighDepotInitialValue":{
"constants": {
"accountInitialValues[depot]": 750.0
}
}
} )
And plot the results:
testbptk.plot_scenarios(=["base"],
scenarios="sm",
scenario_managers=["account[bank]", "account[depot]", "totalValue"],
equations={}) series_names
As always we can compare different scenarios with each other by plotting them simultaneously:
testbptk.plot_scenarios(=["base",
scenarios"scenarioHighDepotInterestRate",
"scenarioHighDepotDepositRate",
"scenarioHighDepotInitialValue"],
="sm",
scenario_managers=["totalValue"],
equations={}) series_names