Enhancements to Optimizer in Analytica 4.0
(Back to What's new in Analytica 4.0?)
Selecting an external engine
The functions LpDefine, QpDefine, and NlpDefine now have an optional engine parameter, which can be used to specify an external add-on engine such as Knitro, Mosek, Xpress, etc. For example:
NlpDefine( ... ; Engine : "KNITRO" )
would use the Knitro engine to solve the non-linear optimization problem. In order to use this feature, the Knitro engine must be properly installed and a license for it acquired. These configurations are described in Using a Solver Add-on Engine.
If the Engine parameter is not explicitly specifyed, LpDefine, QpDefine or NlpDefine will use the internal default solver engine that is deemed most appropriate for the problem. To determine which engine has been selected for a given problem previously defined, use:
SolverInfo( "engine", lp )
New Functions
- SolverInfo allows you to retrieve the components of a previously-defined LP/QP/NLP, or to obtain information about a solver engine. The function is described at SolverInfo.
Generalized Access to Solver Engine Parameters
Each solver engine in general has its own unique set of parameters that can be used to control its search. For example, the parameters used by Knitro only partially overlap the parameters used by the built-in evolutionary algorithm. To be able to take full advantage of Knitro, you need to be able to access all its parameters. If a third-party engine becomes available after Analytica 4.0 is released, it may have additional parameters.
The solver engine parameters have previously been exposed to the Analytica end-user as optional parameters to LpDefine, QpDefine and NlpDefine. Since these definitions are fixed, it doesn't provide a fully flexible mechanism for accessing and setting the solver parameters. For this reason, a new method for specifying solver control settings has been added to LpDefine, QpDefine and NlpDefine.
To change a single setting, you identify the parameter and its setting using optional parameters, e.g.:
NlpDefine( ..., parameter: "PopulationSize", setting: 50 )
Note: LpDefine, QpDefine and NlpDefine all provide optional parameter and setting arguments in the same fashion.
To set multiple settings, you must supply one or two arrays. There are two methods for doing this. The first is to specify one array for the parameter argument, another for the setting argument, such that the two arrays have the same dimensionality. For example:
Index SettingNum := [1,2,3]; Variable ParamArr := Table(SettingNum)("MultiStart","TopoSearch","PopulationSize") Variable SettingArr := Table(SettingNum)(1,1,10) Variable The_Nlp := NlpDefine( ..., parameter: ParamArr, setting: SettingArr )
With this usage, ParamArr may have any number of indexes in common, they need not be one-dimensional. All settings (at any coordinate) are used. A special case of this places the parameter names in the index:
Index ParamNames := ["MultiStart","TopoSearch","PopulationSize"] Variable SettingArr := Table(ParamNames)(1,1,10) Variable The_Nlp := NlpDefine( ..., parameter: ParamNames, setting: SettingArr )
This arrangement does not work if ParamNames is a self-indexed table (or any other self-indexed array) where the parameter names reside in the self-index values. This is because the parameter value, not index value, is passed in.
When the setting parameter is guaranteed to be 1-dimensional, and the parameter names are in the index labels, the parameters argument can be omitted entirely. In the above example, The_Nlp could equally be defined using
NlpDefine( ..., setting: SettingArr )
The one caveat with this is that an error will result if SettingArr is not exactly one-dimensional after array abstraction. If the SettingArr contains any indexes that are iterated over by array abstraction (because of other parameters), those dimensions will be sliced from SettingArr. Therefore, a two or three dimensional SettingArr can be provided, as long as multiple instances of optimization problems are being defined via array abstraction. It is best to ensure that your ParamNames dimension does not appear as a dimension of other parameters, otherwise it would also be abstracted over and you'd end up with a zero-dimensional array after array abstraction (hence an error).
Often the most convenient method for specifying search parameters is to set up a self-indexed table, where the index labels are the parameter names and the cell values are the settings. This is an instance of the previous technique, in which the settings parameter is specified, but the parameter argument is omitted. E.g.:
IndexVals of Parms := ["MultiStart","TopoSearch","PopulationSize"] Definition of Parms := Table(Self)(1,1,10) Variable The_NLP := NlpDefine( ..., Setting:Parms )
Again, notice that the Parameter argument is not explicitly specified.
To view the current search control settings used in a particular LP, QP, or NLP, the function:
SolverInfo( "Setting", lp )
returns a 1-D array, in which a local index ".Parameter" contains the parameter name, and the array value contains the current setting. In general, the local index may contain a different set of parameters than those provided in the Parameter argument to LpDefine. There may be additional ones since in general only a subset will be specified explicitly, and if parameters that don't apply to the current engine appear in the Parameter argument to LpDefine, then those won't appear in the result from SolverInfo.
To access the default parameter settings for a given solver engine, use:
SolverInfo( Engine: engname, Item: "Setting" )
The largest and smallest allowable values for each setting can also be obtained using:
SolverInfo( Engine: engname, Item: ["Setting","Min Setting","Max Setting"] )
Descriptions of what parameter settings do is found in the Frontline Premium Solver for Excel manual.
FindIIS
Two parameters have been added to Function FindIIS, which now has the declaration:
FindIIS( lp : LP_TYPE ; newlp : optional boolean = false ; constraintsOnly : optional boolean )
where:
- lp : an LP instance previously defined using LpDefine. FindIIS cannot be used with QPs or NLPs.
- newlp: When true, a new LP instance is returned. When false, a subset of the constraints is returned.
- constraintsOnly: When true, the lower and upper variable bounds are not relaxed.
FindIIS can now return a new LP instance, having fewer constraints or variable bounds so as to ensure the problem has a feasible solution. When an LP is returned, you access the modified variable bounds or constraint set using:
SolverInfo( iis, "lb" ) SolverInfo( iis, "ub" ) SolverInfo( iis, "Constraints" )
When newlp is false, the constraintsOnly is implied to be true, and the function returns a subset of the Constraints index. FindIIS(lp,false) is equivalent to:
SolverInfo( FindIIS(lp,true,true), "Constraints" )
The previous version of FindIIS returned only a subset of the Constraints index, which provided no method in which reduced variable bounds could be returned. Therefore, the previous version did not relax variable bounds.
When newlp is true, constraintsOnly defaults to false. Note: FindIIS(lp,true) should always be able to find a feasible problem, while Find(lp,false) does not have this guarantee (i.e., when the variable bounds are not consistent).
Multiple File Formats
LpWrite, LpRead, and LpWriteIIS now all contain an optional format parameter, and can read/write to three different file formats:
LpWrite( lp : LP_Type, filename : text ; format : optional text = "LP" ) LpWriteIIS( lp : Lp_type, filename : text ; format : optional text = "LP" ) LpRead( filename : text ; Vars,Constraint : optional IndexType ; format : optional text = "LP" )
Possible file formats are: "LP", "MPS", and "LPFML".
Quadratic Constraints
Previously, a QP could have a quadratic objective function, but constraints had to be linear. QpDefine now permits quadratic constraints. The built-in SOCP engine in Frontline 7.0 can handle quadratic constraints, and the external MOSEK engine add-on can handle these at an improved performance level.
To define quadratic constraints, the quadratic component is specified in the lhsQ parameter, indexed by Vars, Vars2 and Constraints. The linear part of the constraint is defined in the lhs parameter (as before) indexed by Vars and Constraints. For example, consider the following problem:
Minimize x^2 + y^2 - 2*x Subject to: x^2 + 4*x*y + 6*y <= 36 y^2 + 4*x - 2*y <= 0 where x >= 0, y >= 0
The optimum for this problem is at x=.1286, y=0.3031, objective=-0.1488.
We define indexes as follows:
Index Vars := ['x','y'] Index Vars2 := CopyIndex(Vars) Index Constraints := ['c1','c2']
The objective is given by:
Variable c := Table(Vars)(-2,0)
Vars | ||
---|---|---|
Vars2 | x | y |
x | 1 | 0 |
y | 0 | 1 |
The left-hand side is given by:
Vars | ||
---|---|---|
Constraints | x | y |
c1 | 0 | 6 |
c2 | 4 | -2 |
Constraints | ||||
---|---|---|---|---|
c1 | c2 | |||
Vars | Vars | |||
Vars2 | x | y | x | y |
x | 1 | 4 | 0 | 0 |
y | 4 | 0 | 0 | 1 |
Variable Rhs := Table(Constraints)(36,0)
The problem is then formulated as:
QpDefine( Vars, Vars2, Constraints, c:c, q:Q, lhs:Lhs, rhs:Rhs, sense:'<', lb:0, maximize:false )
Grouped Integer Variables
The optimizer now supports a new type of integer constraint -- group exclusive.
A group of N integer variables assigns an integer 1..N to each of the N variables such that they are constrained to have unique values. For example, if a group contains {x1, x2, x3}, then x1=1, x2=2, x3=3 is acceptible, as is x1=3, x2=1, x3=2. But x1=2, x2=3, x3=2 is not acceptible, since the values are not unique.
There can be multiple groups. For example, consider this set of constraints:
Group 1: {x1,x2,x3} Group 2: {x4,x5} Constraints: x1 - x4 = 0 x3 - x5 = 0 x1 + 2*x5 = 4
This system has a single feasible solution: {x1=2, x2=3, x3=1, x4=2, x5=1}.
To specify that a variable belongs to a group, its ctype is set to 'G'. If there is more than one group, then the parameter
group : optional numeric[Vars]
must be specified to LpDefine, QpDefine, NlpDefine. For example:
Variable VarGroups := Table(Vars)(1,1,1,2,2) Variable ub := Table(Vars)(3,3,3,2,2) Variable lp := LpDefine( Vars, ..., ctype: 'G', group: VarGroups, lb:1, ub:ub )
The constraints are specified the usual fashion and thus are omitted in the above for conciseness. Here we have two variable groups, numbered 1 and 2, with membership of each variable in each group specified by the group parameter.
Trace Files for NLP optimization
When a non-linear optimization performs poorly, it is often necessary to look at the path the search took to diagnose the reasons. To make this easier, Ana:NlpDefine has a new parameter, TraceFile. Tracefile accepts a filename (full path or relative to Ana:CurrentDataDirectory). When specified, Analytica records the independent decision vector and each explicitly computed item at each search step (e.g., objective, LHS, gradient, jacobian, objHessian, lhsHessian).
After an optimization has been run, the tab-separated file can be viewed in a text editor.
Enable comment auto-refresher