DotProductComp#
DotProductComp
performs a dot product between two compatible inputs. It may be vectorized to provide the result at one or more points simultaneously.
DotProductComp Options#
The default vec_size
is 1, providing the dot product of \(a\) and \(b\) at a single
point. The lengths of \(a\) and \(b\) are provided by option length
.
Other options for DotProductComp allow the user to rename the input variables \(a\) and \(b\) and the output \(c\), as well as specifying their units.
Option | Default | Acceptable Values | Acceptable Types | Description |
---|---|---|---|---|
a_name | a | N/A | ['str'] | The variable name for input vector a. |
a_units | N/A | N/A | ['str'] | The units for vector a. |
always_opt | False | [True, False] | ['bool'] | If True, force nonlinear operations on this component to be included in the optimization loop even if this component is not relevant to the design variables and responses. |
b_name | b | N/A | ['str'] | The variable name for input vector b. |
b_units | N/A | N/A | ['str'] | The units for vector b. |
c_name | c | N/A | ['str'] | The variable name for output vector c. |
c_units | N/A | N/A | ['str'] | The units for vector c. |
derivs_method | N/A | ['jax', 'cs', 'fd', None] | N/A | The method to use for computing derivatives |
distributed | False | [True, False] | ['bool'] | If True, set all variables in this component as distributed across multiple processes |
length | 3 | N/A | ['int'] | The length of vectors a and b |
run_root_only | False | [True, False] | ['bool'] | If True, call compute, compute_partials, linearize, apply_linear, apply_nonlinear, and compute_jacvec_product only on rank 0 and broadcast the results to the other ranks. |
use_jit | True | [True, False] | ['bool'] | If True, attempt to use jit on compute_primal, assuming jax or some other AD package is active. |
vec_size | 1 | N/A | ['int'] | The number of points at which the dot product is computed |
DotProductComp Constructor#
The call signature for the DotProductComp
constructor is:
- DotProductComp.__init__(**kwargs)[source]
Initialize the Dot Product component.
DotProductComp Usage#
There are often situations when numerous products need to be computed, essentially in parallel.
You can reduce the number of components required by having one DotProductComp
perform multiple operations.
This is also convenient when the different operations have common inputs.
The add_product
method is used to create additional products after instantiation.
- DotProductComp.add_product(c_name, a_name='a', b_name='b', c_units=None, a_units=None, b_units=None, vec_size=1, length=3)[source]
Add a new output product to the dot product component.
- Parameters:
- c_namestr
The name of the vector product output.
- a_namestr
The name of the first vector input.
- b_namestr
The name of the second input.
- c_unitsstr or None
The units of the output.
- a_unitsstr or None
The units of input a.
- b_unitsstr or None
The units of input b.
- vec_sizeint
The number of points at which the dot vector product should be computed simultaneously. The shape of the output is (vec_size,).
- lengthint
The length of the vectors a and b. Their shapes are (vec_size, length).
DotProductComp Example#
In the following example DotProductComp is used to compute instantaneous power as the
dot product of force and velocity at 100 points simultaneously. Note the use of
a_name
, b_name
, and c_name
to assign names to the inputs and outputs.
Units are assigned using a_units
, b_units
, and c_units
.
Note that no internal checks are performed to ensure that c_units
are consistent
with a_units
and b_units
.
import numpy as np
import openmdao.api as om
n = 24
p = om.Problem()
dp_comp = om.DotProductComp(vec_size=n, length=3, a_name='F', b_name='v', c_name='P',
a_units='N', b_units='m/s', c_units='W')
p.model.add_subsystem(name='dot_prod_comp', subsys=dp_comp,
promotes_inputs=[('F', 'force'), ('v', 'vel')])
p.setup()
p.set_val('force', np.random.rand(n, 3))
p.set_val('vel', np.random.rand(n, 3))
p.run_model()
# Verify the results against numpy.dot in a for loop.
expected = []
for i in range(n):
a_i = p.get_val('force')[i, :]
b_i = p.get_val('vel')[i, :]
expected.append(np.dot(a_i, b_i))
actual_i = p.get_val('dot_prod_comp.P')[i]
rel_error = np.abs(expected[i] - actual_i)/actual_i
assert rel_error < 1e-9, f"Relative error: {rel_error}"
print(p.get_val('dot_prod_comp.P', units='kW'))
[0.00058514 0.00088843 0.00085348 0.00055612 0.00096191 0.00156695
0.0002154 0.00068691 0.0004566 0.00057905 0.00096445 0.00013484
0.00046878 0.00067396 0.00064807 0.00091011 0.00063833 0.00126274
0.00062597 0.00014261 0.00073965 0.0001417 0.00093248 0.00059342]
DotProductComp Example with Multiple Products#
When defining multiple products:
An input name in one call to
add_product
may not be an output name in another call, and vice-versa.The units and shape of variables used across multiple products must be the same in each one.
n = 24
p = om.Problem()
dp_comp = om.DotProductComp(vec_size=n, length=3,
a_name='F', b_name='d', c_name='W',
a_units='N', b_units='m', c_units='J')
dp_comp.add_product(vec_size=n, length=3,
a_name='F', b_name='v', c_name='P',
a_units='N', b_units='m/s', c_units='W')
p.model.add_subsystem(name='dot_prod_comp', subsys=dp_comp,
promotes_inputs=[('F', 'force'), ('d', 'disp'), ('v', 'vel')])
p.setup()
p.set_val('force', np.random.rand(n, 3))
p.set_val('disp', np.random.rand(n, 3))
p.set_val('vel', np.random.rand(n, 3))
p.run_model()
# Verify the results against numpy.dot in a for loop.
expected_P = []
expected_W = []
for i in range(n):
a_i = p.get_val('force')[i, :]
b_i = p.get_val('disp')[i, :]
expected_W.append(np.dot(a_i, b_i))
actual_i = p.get_val('dot_prod_comp.W')[i]
rel_error = np.abs(actual_i - expected_W[i])/actual_i
assert rel_error < 1e-9, f"Relative error: {rel_error}"
b_i = p.get_val('vel')[i, :]
expected_P.append(np.dot(a_i, b_i))
actual_i = p.get_val('dot_prod_comp.P')[i]
rel_error = np.abs(expected_P[i] - actual_i)/actual_i
assert rel_error < 1e-9, f"Relative error: {rel_error}"
print(p.get_val('dot_prod_comp.W', units='kJ'))
[0.00019549 0.0006515 0.00036298 0.00075048 0.00050097 0.00057572
0.00067904 0.00044299 0.00062443 0.00075764 0.00032 0.00044788
0.0003864 0.00019364 0.00033729 0.00032381 0.00071293 0.00118303
0.00026586 0.00041336 0.00075064 0.00076006 0.00046978 0.00051975]
print(p.get_val('dot_prod_comp.P', units='kW'))
[5.49947949e-04 2.31686114e-04 8.20938197e-04 1.76556698e-04
7.26216598e-04 1.05879379e-03 7.94067493e-04 9.55123831e-04
1.06381281e-03 6.69362014e-04 6.76984116e-04 6.13069290e-04
8.65772541e-04 2.82387559e-04 9.35980113e-05 9.79503719e-04
5.78707677e-04 1.07809340e-03 6.16461528e-04 3.19780148e-04
7.26846109e-04 1.11978679e-03 1.12883857e-03 5.92924130e-04]