BalanceComp#
BalanceComp
is a specialized implementation of ImplicitComponent that is intended to provide a simple way to implement most implicit equations without the need to define your own residuals.
BalanceComp Options#
Option | Default | Acceptable Values | Acceptable Types | Description |
---|---|---|---|---|
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. |
assembled_jac_type | N/A | ['csc', 'csr', 'dense', None] | N/A | Linear solver(s) in this group or implicit component, if using an assembled jacobian, will use this type. |
default_shape | (1,) | N/A | ['tuple'] | Default shape for variables that do not set val to a non-scalar value or set shape, shape_by_conn, copy_shape, or compute_shape. Default is (1,). |
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 |
guess_func | N/A | N/A | ['function'] | A callable function in the form f(inputs, outputs, residuals) that can provide an initial "guess" value of the state variable(s) based on the inputs, outputs and residuals. |
run_root_only | False | [True, False] | ['bool'] | If True, call compute, compute_partials, linearize, apply_linear, apply_nonlinear, solve_linear, solve_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 capable of jitting is active. |
BalanceComp Constructor#
The call signature for the BalanceComp
constructor is:
- BalanceComp.__init__()[source]
Initialize a BalanceComp, optionally creating a new implicit state variable.
The BalanceComp allows for the creation of one or more implicit state variables, and computes the residuals for those variables based on the following equation.
\[\mathcal{R}_{name} = \frac{f_{mult}(x,...) \times f_{lhs}(x,...) - f_{rhs}(x,...)}{f_{norm}(f_{rhs}(x,...))}\]Where \(f_{lhs}\) represents the left-hand-side of the equation, \(f_{rhs}\) represents the right-hand-side, and \(f_{mult}\) is an optional multiplier on the left hand side. At least one of these quantities should be a function of the associated state variable. If use_mult is True the default value of the multiplier is 1. The optional normalization function \(f_{norm}(f_{rhs}(x,...))\) is computed as:
\[\begin{split}f_{norm}(f_{rhs}(x,...)) = \begin{cases} \left| f_{rhs} \right|, & \text{if normalize and } \left| f_{rhs} \right| \geq 2 \\ 0.25 f_{rhs}^2 + 1, & \text{if normalize and } \left| f_{rhs} \right| < 2 \\ 1, & \text{if not normalize} \end{cases}\end{split}\]New state variables, and their associated residuals are created by calling add_balance. As an example, solving the equation \(x**2 = 2\) implicitly can be be accomplished as follows:
prob = Problem() bal = BalanceComp() bal.add_balance('x', val=1.0) exec_comp = ExecComp('y=x**2') prob.model.add_subsystem(name='exec', subsys=exec_comp) prob.model.add_subsystem(name='balance', subsys=bal) prob.model.connect('balance.x', 'exec.x') prob.model.connect('exec.y', 'balance.lhs:x') prob.model.linear_solver = DirectSolver() prob.model.nonlinear_solver = NewtonSolver(solve_subsystems=False) prob.setup() prob.set_val('exec.x', 2) prob.run_model()The arguments to add_balance can be provided on initialization to provide a balance with a one state/residual without the need to call add_balance:
prob = Problem() bal = BalanceComp('x', val=1.0) exec_comp = ExecComp('y=x**2') prob.model.add_subsystem(name='exec', subsys=exec_comp) prob.model.add_subsystem(name='balance', subsys=bal) prob.model.connect('balance.x', 'exec.x') prob.model.connect('exec.y', 'balance.lhs:x') prob.model.linear_solver = DirectSolver() prob.model.nonlinear_solver = NewtonSolver(solve_subsystems=False) prob.setup() prob.set_val('exec.x', 2) prob.run_model()
Using the BalanceComp#
BalanceComp
allows you to add one or more state variables and its associated
implicit equations. For each balance
added to the component it
solves the following equation:
The optional normalization function \(f_{norm}(f_{rhs})\) is computed as:
The following inputs and outputs are associated with each implicit state.
Name |
I/O |
Description |
---|---|---|
{name} |
output |
implicit state variable |
lhs:{name} |
input |
left-hand side of equation to be balanced |
rhs:{name} |
input |
right-hand side of equation to be balanced |
mult:{name} |
input |
left-hand side multiplier of equation to be balanced |
The default value for the rhs:{name}
input can be set to via the
rhs_val
argument (see arguments below). If the rhs value is fixed (e.g. 0),
then the input can be left unconnected. The lhs:{name}
input must always have
something connected to it and should be dependent upon the value of the implicit state variable.
The multiplier is optional and will default to 1.0 if not connected.
BalanceComp
supports vectorized implicit states. Simply provide a default
value or shape when adding the balance that reflects the correct shape.
You can provide the arguments to create a balance when instantiating a BalanceComp
or you can use the add_balance
method to create one or more state variables after
instantiation. The constructor accepts all the same arguments as the add_balance
method:
- BalanceComp.add_balance(name, eq_units=None, lhs_name=None, rhs_name=None, rhs_val=0.0, use_mult=False, mult_name=None, mult_val=1.0, normalize=True, val=None, lhs_kwargs=None, rhs_kwargs=None, mult_kwargs=None, **kwargs)[source]
Add a new state variable and associated equation to be balanced.
This will create new inputs lhs:name, rhs:name, and mult:name that will define the left and right sides of the equation to be balanced, and a multiplier for the left-hand-side.
- Parameters:
- namestr
The name of the state variable to be created.
- eq_unitsstr or None
Units for the left-hand-side and right-hand-side of the equation to be balanced.
- lhs_namestr or None
Optional name for the LHS variable associated with the implicit state variable. If None, the default will be used: ‘lhs:{name}’.
- rhs_namestr or None
Optional name for the RHS variable associated with the implicit state variable. If None, the default will be used: ‘rhs:{name}’.
- rhs_valint, float, or np.array
Default value for the RHS. Must be compatible with the shape (optionally) given by the val or shape option in kwargs.
- use_multbool
Specifies whether the LHS multiplier is to be used. If True, then an additional input mult_name is created, with the default value given by mult_val, that multiplies lhs. Default is False.
- mult_namestr or None
Optional name for the LHS multiplier variable associated with the implicit state variable. If None, the default will be used: ‘mult:{name}’.
- mult_valint, float, or np.array
Default value for the LHS multiplier. Must be compatible with the shape (optionally) given by the val or shape option in kwargs.
- normalizebool
Specifies whether or not the resulting residual should be normalized by a quadratic function of the RHS.
- valfloat, int, or np.ndarray
Set initial value for the state.
- lhs_kwargsdict
Keyword arguments to be passed to the add_input call for the left-hand-side input of the equation (see add_input method).
- rhs_kwargsdict
Keyword arguments to be passed to the add_input call for the right-hand-side input of the equation (see add_input method).
- mult_kwargsdict
Keyword arguments to be passed to the add_input call for the multiplier input of the equation, if used (see add_input method).
- **kwargsdict
Additional arguments to be passed for the creation of the implicit state variable. (see add_output method).
Note that the kwargs
arguments can include any of the keyword arguments normally available
when creating an output variable with the
add_output
method of a Component.
Example: Scalar Root Finding#
The following example uses a BalanceComp to implicitly solve the equation:
Here, our LHS is connected to a computed value for \(x^2\), the multiplier is 2, and the RHS is 4. The expected solution is \(x=\sqrt{2}\). We initialize \(x\) with a value of 1 so that it finds the positive root.
import openmdao.api as om
prob = om.Problem()
bal = om.BalanceComp()
bal.add_balance('x', use_mult=True)
exec_comp = om.ExecComp('y=x**2', x={'val': 1}, y={'val': 1})
prob.model.add_subsystem(name='exec', subsys=exec_comp)
prob.model.add_subsystem(name='balance', subsys=bal)
prob.model.connect('balance.x', 'exec.x')
prob.model.connect('exec.y', 'balance.lhs:x')
prob.model.linear_solver = om.DirectSolver(assemble_jac=True)
prob.model.nonlinear_solver = om.NewtonSolver(solve_subsystems=False, maxiter=100, iprint=0)
prob.setup()
prob.set_val('balance.rhs:x', 4)
prob.set_val('balance.mult:x', 2.)
# A reasonable initial guess to find the positive root.
prob['balance.x'] = 1.0
prob.run_model()
print(prob.get_val('balance.x'))
[1.41421356]
Alternatively, we could simplify the code by using the mult_val
argument.
prob = om.Problem()
bal = om.BalanceComp()
bal.add_balance('x', use_mult=True, mult_val=2.0)
exec_comp = om.ExecComp('y=x**2', x={'val': 1}, y={'val': 1})
prob.model.add_subsystem(name='exec', subsys=exec_comp)
prob.model.add_subsystem(name='balance', subsys=bal)
prob.model.connect('balance.x', 'exec.x')
prob.model.connect('exec.y', 'balance.lhs:x')
prob.model.linear_solver = om.DirectSolver(assemble_jac=True)
prob.model.nonlinear_solver = om.NewtonSolver(solve_subsystems=False, maxiter=100, iprint=0)
prob.setup()
prob.set_val('balance.rhs:x', 4)
# A reasonable initial guess to find the positive root.
prob.set_val('balance.x', 1.0)
prob.run_model()
print(prob.get_val('balance.x'))
[1.41421356]
Example: Vectorized Root Finding#
The following example uses a BalanceComp to implicitly solve the equation:
for various values of \(b\), and \(c\). Here, our LHS is connected to a computed value of the linear equation. The multiplier is one and the RHS is zero (the defaults), and thus they need not be connected.
n = 100
prob = om.Problem()
exec_comp = om.ExecComp('y=b*x+c',
b={'val': np.random.uniform(0.01, 100, size=n)},
c={'val': np.random.rand(n)},
x={'val': np.zeros(n)},
y={'val': np.ones(n)})
prob.model.add_subsystem(name='exec', subsys=exec_comp)
prob.model.add_subsystem(name='balance', subsys=om.BalanceComp('x', val=np.ones(n)))
prob.model.connect('balance.x', 'exec.x')
prob.model.connect('exec.y', 'balance.lhs:x')
prob.model.linear_solver = om.DirectSolver(assemble_jac=True)
prob.model.nonlinear_solver = om.NewtonSolver(solve_subsystems=False, maxiter=100, iprint=0)
prob.setup()
prob.set_val('balance.x', np.random.rand(n))
prob.run_model()
b = prob.get_val('exec.b')
c = prob.get_val('exec.c')
print(prob.get_val('balance.x'))
[-2.80153726e-02 -8.27687322e-02 -2.00458070e-02 -6.90864839e-02
-1.28542063e-03 -3.68351118e-02 -1.23758298e-03 -5.07795953e-02
-2.07438150e-03 -1.55385949e-02 -8.84027891e-04 -5.51846305e-02
-1.27798251e-02 -1.92698884e-03 -1.10567316e-02 -1.05144702e-02
-6.81598910e-03 -7.30295770e-03 -1.86598867e-04 -1.20104213e-02
-2.21997114e-02 -1.33593466e-04 -1.05382070e-02 -1.54023964e-03
-1.14950038e-02 -3.65284113e-03 -8.50470083e-03 -8.85133809e-04
-1.20808261e-02 -2.10905777e-03 -7.80883109e-03 -6.93474437e-03
-1.86214119e-03 -2.18162297e-03 -4.92200737e-03 -5.51629778e-02
-7.27944930e-03 -1.54585807e+00 -1.02952662e-02 -6.48204852e-02
-4.72316059e-01 -1.17661646e-03 -1.13383895e-02 -3.22629474e-02
-5.06605782e-02 -2.27060483e-02 -9.45521879e-03 -1.78097200e-02
-6.85684863e-03 -8.07820237e-03 -8.92280580e-04 -6.52451209e-03
-2.36421388e-01 -1.33506620e-05 -3.10638782e-03 -3.02659546e-02
-4.30062532e-03 -5.69366479e-03 -8.84633605e-03 -6.27673307e-03
-1.18896811e-02 -2.69587116e-03 -1.70506864e-02 -4.26364490e-03
-2.50110605e-03 -1.48499221e-02 -7.08204327e-03 -3.08906337e-03
-5.10490084e-03 -4.05625860e-02 -3.74233435e-02 -9.44060386e-02
-1.78904653e-03 -3.32679371e-03 -6.00990182e-03 -9.76476454e-03
-1.91806314e-02 -2.67014706e-02 -9.94313056e-02 -1.41971231e-02
-9.49824692e-03 -5.15733095e-03 -1.33598145e-02 -9.30054498e-05
-1.15827288e-02 -4.95975987e-03 -5.56534365e-02 -4.25918185e-03
-1.00403303e-02 -8.17865520e-03 -1.72406723e-02 -4.59193526e-02
-1.03177142e-04 -1.71314172e-03 -5.57785561e-02 -2.13795696e-02
-9.80095880e-02 -1.45780183e+01 -6.00311834e-03 -1.49833585e-02]
print(-c/b) # expected
[-2.80153726e-02 -8.27687322e-02 -2.00458070e-02 -6.90864839e-02
-1.28542063e-03 -3.68351118e-02 -1.23758298e-03 -5.07795953e-02
-2.07438150e-03 -1.55385949e-02 -8.84027891e-04 -5.51846305e-02
-1.27798251e-02 -1.92698884e-03 -1.10567316e-02 -1.05144702e-02
-6.81598910e-03 -7.30295770e-03 -1.86598867e-04 -1.20104213e-02
-2.21997114e-02 -1.33593466e-04 -1.05382070e-02 -1.54023964e-03
-1.14950038e-02 -3.65284113e-03 -8.50470083e-03 -8.85133809e-04
-1.20808261e-02 -2.10905777e-03 -7.80883109e-03 -6.93474437e-03
-1.86214119e-03 -2.18162297e-03 -4.92200737e-03 -5.51629778e-02
-7.27944930e-03 -1.54585807e+00 -1.02952662e-02 -6.48204852e-02
-4.72316059e-01 -1.17661646e-03 -1.13383895e-02 -3.22629474e-02
-5.06605782e-02 -2.27060483e-02 -9.45521879e-03 -1.78097200e-02
-6.85684863e-03 -8.07820237e-03 -8.92280580e-04 -6.52451209e-03
-2.36421388e-01 -1.33506620e-05 -3.10638782e-03 -3.02659546e-02
-4.30062532e-03 -5.69366479e-03 -8.84633605e-03 -6.27673307e-03
-1.18896811e-02 -2.69587116e-03 -1.70506864e-02 -4.26364490e-03
-2.50110605e-03 -1.48499221e-02 -7.08204327e-03 -3.08906337e-03
-5.10490084e-03 -4.05625860e-02 -3.74233435e-02 -9.44060386e-02
-1.78904653e-03 -3.32679371e-03 -6.00990182e-03 -9.76476454e-03
-1.91806314e-02 -2.67014706e-02 -9.94313056e-02 -1.41971231e-02
-9.49824692e-03 -5.15733095e-03 -1.33598145e-02 -9.30054498e-05
-1.15827288e-02 -4.95975987e-03 -5.56534365e-02 -4.25918185e-03
-1.00403303e-02 -8.17865520e-03 -1.72406723e-02 -4.59193526e-02
-1.03177142e-04 -1.71314172e-03 -5.57785561e-02 -2.13795696e-02
-9.80095880e-02 -1.45780183e+01 -6.00311834e-03 -1.49833585e-02]
Example: Providing an Initial Guess for a State Variable#
BalanceComp
has a guess_func
option that can be used to supply an initial guess
value for the state variables. This option provides the same functionality as the
guess_nonlinear
method of ImplicitComponent.
The Kepler example script shows how guess_func
can be used.
prob = om.Problem()
bal = om.BalanceComp()
bal.add_balance(name='E', val=0.0, units='rad', eq_units='rad', rhs_name='M')
# Use M (mean anomaly) as the initial guess for E (eccentric anomaly)
def guess_function(inputs, outputs, residuals):
if np.abs(residuals['E']) > 1.0E-2:
outputs['E'] = inputs['M']
bal.options['guess_func'] = guess_function
# ExecComp used to compute the LHS of Kepler's equation.
lhs_comp = om.ExecComp('lhs=E - ecc * sin(E)',
lhs={'val': 0.0, 'units': 'rad'},
E={'val': 0.0, 'units': 'rad'},
ecc={'val': 0.0})
prob.model.add_subsystem(name='balance', subsys=bal,
promotes_inputs=['M'],
promotes_outputs=['E'])
prob.model.set_input_defaults('M', 85.0, units='deg')
prob.model.add_subsystem(name='lhs_comp', subsys=lhs_comp,
promotes_inputs=['E', 'ecc'])
# Explicit connections
prob.model.connect('lhs_comp.lhs', 'balance.lhs:E')
# Set up solvers
prob.model.linear_solver = om.DirectSolver()
prob.model.nonlinear_solver = om.NewtonSolver(solve_subsystems=False, maxiter=100, iprint=2)
prob.setup()
prob.set_val('ecc', 0.6)
prob.run_model()
print(np.degrees(prob.get_val('E')))
NL: Newton 0 ; 1.30402513 1
NL: Newton 1 ; 0.117134648 0.0898254536
NL: Newton 2 ; 0.00208780933 0.00160104992
NL: Newton 3 ; 7.36738647e-07 5.64972738e-07
NL: Newton 4 ; 9.19264664e-14 7.04943981e-14
NL: Newton Converged
[115.91942563]