VirtualPTDF
Contrary to the traditional PTDF
matrix, the VirtualPTDF
is a stucture contatining rows of the original matrix, related to specific system branches. The different rows of the PTDF
matrix are cached in the VirtualPTDF
structure as they are evaluated. This allows to keep just the portion of the original matrix which is of interest to the user, avoiding the unecessary computation of the whole matrix.
Refer to the different arguments of the VirtualPTDF
methods by looking at the "Public API Reference" page.
How the VirtualPTDF
works
The VirtualPTDF
is a structure containing everything needed to compute any row of the PTDF matrix and store it. To do so, the VirtualPTDF
must store the BA matrix (coming from the BA_Matrix
struct) and the inverse of the ABA matrix (coming from ABA_MAtrix
struct). In particular, KLU
is used to get the LU factorization matrices of the ABA matrix and these ones are stored, avoid the inversion.
Once the VirtualPTDF
is initialized, each row of the PTDF matrix can be evaluated separately. The algorithmic procedure is the following:
- Define the
VirtualPTDF
structure - Call any element of the matrix to define and store the relative row as well as showing the selected element
Regarding point 2, if the row has been stored previosly then the desired element is just loaded from the cache and shown.
The flowchart below shows how the VirtualPTDF
is structured and how it works. Examples will be presented in the following sections.
Initialize VirtualPTDF
and compute/access row/element
As for the PTDF
matrix, at first the System
data must be loaded. The "RTS-GMLC" systems is considered as example:
julia> using PowerNetworkMatrices
julia> using PowerSystemCaseBuilder
julia> const PNM = PowerNetworkMatrices;
julia> const PSB = PowerSystemCaseBuilder;
julia> sys = PSB.build_system(PSB.PSISystems, "RTS_GMLC_DA_sys");
[ Info: Loaded time series from storage file existing=/home/runner/.julia/packages/PowerSystemCaseBuilder/dJGb8/data/serialized_system/e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855/RTS_GMLC_DA_sys_time_series_storage.h5 new=/tmp/jl_9hCTa1 compression=InfrastructureSystems.CompressionSettings(false, InfrastructureSystems.CompressionTypesModule.CompressionTypes.DEFLATE = 1, 3, true)
At this point the VirtualPTDF
is initialized with the following simple command:
julia> v_ptdf = VirtualPTDF(sys);
Now, an element of the matrix can be computed by calling the branch name and bus number:
julia> el_C31_2_105 = v_ptdf["C31-2", 105]
-0.005119964728130656
Alternatively, the number of the branch and bus (corresponding to the number of the PTDF row and column) can be used. In this case the row and column numbers are mapped by the dictonaries contained in the lookup
field.
julia> row_number = v_ptdf.lookup[1]["C31-2"]
112
julia> col_number = v_ptdf.lookup[2][105]
5
julia> el_C31_2_105_bis = v_ptdf[row_number, col_number]
-0.005119964728130656
NOTE: this example was made for the sake of completeness and considering the actual branch name and bus number is reccomended.
As previosly mentioned, in order to evaluate a single element of the VirtualPTDF
, the entire row related to the selected branch must be considered. For this reason it is cached in the VirtualPTDF
structure for later calls. This is evident by looking at the following example:
julia> sys_2k = PSB.build_system(PSB.PSYTestSystems, "tamu_ACTIVSg2000_sys");
julia> v_ptdf_2k = VirtualPTDF(sys_2k); # evaluate PTDF row related to branch "ODESSA 2 0 -1001-ODESSA 3 0 -1064-i_1"
julia> @time v_ptdf_2k["ODESSA 2 0 -1001-ODESSA 3 0 -1064-i_1", 8155] # call same element after the row has been stored
0.000107 seconds (21 allocations: 47.703 KiB) 5.0022477285046985e-5
julia> @time v_ptdf_2k["ODESSA 2 0 -1001-ODESSA 3 0 -1064-i_1", 8155]
0.000011 seconds (1 allocation: 16 bytes) 5.0022477285046985e-5
VirtualPTDF
with distributed slack bus
As for the PTDF
matrix, here too each row can be evaluated considering distibuted slack buses. A vector of type Vector{Float64}
is defined, specifying the weights for each bus of the system.
julia> # smaller system for the next examples sys_2 = PSB.build_system(PSB.PSITestSystems, "c_sys5"); # consider equal distribution accross each bus for this example
[ Info: Loaded time series from storage file existing=/home/runner/.julia/packages/PowerSystemCaseBuilder/dJGb8/data/serialized_system/614e094ea985a55912fc1321256a49b755f9fc451c0f264f24d6d04af20e84d7/c_sys5_time_series_storage.h5 new=/tmp/jl_NnrDvW compression=InfrastructureSystems.CompressionSettings(false, InfrastructureSystems.CompressionTypesModule.CompressionTypes.DEFLATE = 1, 3, true)
julia> buscount = length(PNM.get_buses(sys_2));
julia> dist_slack = 1 / buscount * ones(buscount);
julia> dist_slack_array = dist_slack / sum(dist_slack);
Now initialize the VirtualPTDF
by defining the dist_slack
field with the vector of weights previosly computed:
julia> v_ptdf_distr = VirtualPTDF(sys_2, dist_slack=dist_slack_array);
[ Info: Distributed bus
julia> v_ptdf_orig = VirtualPTDF(sys_2);
Now check the difference with the same row related to the branch "1"
evaluated without considering distributed slack bus.
julia> row_distr = [v_ptdf_distr["1", j] for j in v_ptdf_distr.axes[2]]
5-element Vector{Float64}: 0.28820251124689455 -0.3816088095078225 -0.2547035520042974 0.09428590613039686 0.25382394413482845
julia> row_original = [v_ptdf_orig["1", j] for j in v_ptdf_orig.axes[2]]
5-element Vector{Float64}: 0.19391660511649766 -0.47589471563821933 -0.34898945813469423 0.0 0.15953803800443161
"Sparse" VirtualPTDF
Sparsification of each row can be achieved in the same fashion as for the PTDF
matrix, by removing those elements whose absolute values is below a certain threshold.
As for the example show for the PTDF
matrix, here to a very high values of 0.2 is considered for the tol
field. Again, this value is considered just for the sake of this example.
julia> v_ptdf_dense = VirtualPTDF(sys_2);
julia> v_ptdf_sparse = VirtualPTDF(sys_2, tol=0.2);
Let's now evaluate the same row as before and compare the results:
julia> original_row = [v_ptdf_dense["1", j] for j in v_ptdf_dense.axes[2]]
5-element Vector{Float64}: 0.19391660511649766 -0.47589471563821933 -0.34898945813469423 0.0 0.15953803800443161
julia> sparse_row = [v_ptdf_sparse["1", j] for j in v_ptdf_sparse.axes[2]]
5-element Vector{Float64}: 0.0 -0.47589471563821933 -0.34898945813469423 0.0 0.0