Lingo Users Manual
Lingo Users Manual
Lingo Users Manual
The LINGO software and it’s related documentation are copyrighted. You may not copy the LINGO
software or related documentation except in the manner authorized in the related documentation or
with the written permission of LINDO Systems Inc.
TRADEMARKS
LINGO is a trademark, and LINDO is a registered trademark, of LINDO Systems Inc. Other product
and company names mentioned herein are the property of their respective owners.
DISCLAIMER
LINDO Systems, Inc. warrants that on the date of receipt of your payment, the disk enclosed in the
disk envelope contains an accurate reproduction of the LINGO software and that the copy of the
related documentation is accurately reproduced. Due to the inherent complexity of computer programs
and computer models, the LINGO software may not be completely free of errors. You are advised to
verify your answers before basing decisions on them. NEITHER LINDO SYSTEMS, INC. NOR
ANYONE ELSE ASSOCIATED IN THE CREATION, PRODUCTION, OR DISTRIBUTION OF
THE LINGO SOFTWARE MAKES ANY OTHER EXPRESSED WARRANTIES REGARDING
THE DISKS OR DOCUMENTATION AND MAKES NO WARRANTIES AT ALL, EITHER
EXPRESSED OR IMPLIED, REGARDING THE LINGO SOFTWARE, INCLUDING THE
IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE,
OR OTHERWISE. Further, LINDO Systems, Inc. reserves the right to revise this software and related
documentation and make changes to the content hereof without obligation to notify any person of such
revisions or changes.
Published by
iii
iv CONTENTS
5. Help Menu .................................................................................................................................209
6 Command-Line Commands.........................................................................................................215
The Commands In Brief ................................................................................................................215
The Commands In Depth ..............................................................................................................217
7 LINGO’s Operators and Functions..............................................................................................285
Standard Operators.......................................................................................................................285
Mathematical Functions ................................................................................................................289
Financial Functions .......................................................................................................................290
Probability Functions .....................................................................................................................290
Variable Domain Functions ...........................................................................................................293
Set Handling Functions .................................................................................................................293
Set Looping Functions...................................................................................................................295
Interface Functions........................................................................................................................296
Report Functions ...........................................................................................................................298
Miscellaneous Functions ...............................................................................................................309
8 Interfacing with External Files.....................................................................................................313
Cut and Paste Transfers ...............................................................................................................313
Text File Interface Functions .........................................................................................................315
LINGO Command Scripts..............................................................................................................323
Specifying Files in the Command-line ...........................................................................................326
Redirecting Input and Output ........................................................................................................328
Managing LINGO Files..................................................................................................................328
9 Interfacing With Spreadsheets....................................................................................................331
Importing Data from Spreadsheets................................................................................................331
Exporting Solutions to Spreadsheets ............................................................................................336
OLE Automation Links from Excel.................................................................................................344
Embedding LINGO Models in Excel..............................................................................................348
Embedding Excel Sheets in LINGO ..............................................................................................354
Summary.......................................................................................................................................358
10 Interfacing with Databases ..........................................................................................................359
ODBC Data Sources .....................................................................................................................360
Importing Data from Databases with @ODBC ..............................................................................367
Importing Data with ODBC in a PERT Model ................................................................................369
Exporting Data with @ODBC ........................................................................................................371
Exporting Data with ODBC in a PERT Model................................................................................374
11 Interfacing with Other Applications............................................................................................379
The LINGO Dynamic Link Library..................................................................................................379
User Defined Functions.................................................................................................................414
12 Developing More Advanced Models ...........................................................................................419
Production Management Models...................................................................................................420
Logistics Models............................................................................................................................434
CONTENTS v
Financial Models ...........................................................................................................................441
Queuing Models ............................................................................................................................457
Marketing Models ..........................................................................................................................465
13 Programming LINGO....................................................................................................................473
Programming Features..................................................................................................................473
Programming Example: Binary Search .........................................................................................487
Programming Example: Markowitz Efficient Frontier.....................................................................490
Programming Example: Cutting Stock...........................................................................................497
Summary.......................................................................................................................................503
14 On Mathematical Modeling ..........................................................................................................505
Solvers Used Internally by LINGO.................................................................................................505
Type of Constraints .......................................................................................................................506
Local Optima vs. Global Optima....................................................................................................508
Smooth vs. Nonsmooth Functions ................................................................................................513
Guidelines for Nonlinear Modeling ................................................................................................514
Appendix A: Additional Examples of LINGO Modeling..................................................................517
Appendix B: Error Messages ...........................................................................................................603
Appendix C: Bibliography and Suggested Reading.......................................................................635
Index ....................................................................................................................................................637
vi CONTENTS
Preface
The new features in LINGO 10.0 include the following:
Scripting Capability:
The syntax of the calc section has been extended to include full scripting
capabilities. This allows you to solve one or more models in a loop, with model
solutions being fed to subsequent models as input. Some of the control
structures and functions added to support scripting include:
♦ The SUBMODEL statement allows you to define one or more
submodels within a main model. Submodels may be solved
using the @SOLVE command.
♦ @IFC/@ELSE statements for conditional IF/THEN/ELSE
branching.
♦ @WHILE statement for executing compound statements in a
loop.
♦ Refer to sample models LOOPPORT , LOOPCUT and
LOOPTSP for examples of using scripting loops to solve,
respectively, cutting stock, Markowitz portfolio and traveling
salesman models.
vii
viii PREFACE
♦ Improved preprocessing to derive implied variable bounds and
types
We hope you enjoy this new release of LINGO. Many of the new features in this release are due to
suggestions from our users. If there are any features you'd like to see in the next release of LINGO,
please let us know. You can reach us at:
LINDO Systems Inc.
1415 N. Dayton St.
Chicago, Illinois 60622
(312) 988-7422
[email protected]
http://www.lindo.com
March 2006
1 Getting Started with
LINGO
What is LINGO?
LINGO is a simple tool for utilizing the power of linear and nonlinear optimization to formulate large
problems concisely, solve them, and analyze the solution. Optimization helps you find the answer that
yields the best result; attains the highest profit, output, or happiness; or achieves the lowest cost, waste,
or discomfort. Often these problems involve making the most efficient use of your resources—
including money, time, machinery, staff, inventory, and more. Optimization problems are often
classified as linear or nonlinear, depending on whether the relationships in the problem are linear with
respect to the variables.
If you are a new user, it is recommended you go through the first seven chapters to familiarize yourself
with LINGO. Then, you may want to see Chapter 14, On Mathematical Modeling, for more
information on the difference between linear and nonlinear models and how to develop large models. It
may also be helpful to view some sample models in Chapter 12, Developing More Advanced Models,
or Appendix A, Additional Examples of LINGO Modeling, to see if a particular template example is
similar to a problem you have. For users of previous versions of LINGO, the new features are
summarized in the Preface at the beginning of the manual.
Installing LINGO
This section discusses how to install LINGO on the Windows platform. To install LINGO on
platforms other than Windows, refer to the installation instructions included with your software.
Installing the LINGO software is straightforward. To setup LINGO for Windows, place your CD in the
appropriate drive and run the installation program SETUP contained in the LINGO folder. The LINGO
installation program will open and guide you through the steps required to install LINGO on your hard
drive.
Note: If there is a previous version of LINGO installed on your machine, then you may need to
uninstall it before you can install the new copy of LINGO. To uninstall the existing copy of
LINGO, click on the Windows Start button, select the Settings command, select Control
Panel, then double click on the Add or Remove Programs icon. You should then be able to
select LINGO and have the old version removed from your system.
1
2 CHAPTER 1
Most copies of LINGO come with their licenses preinstalled. However, some versions of LINGO
require you to input a license key. If your version of LINGO requires a license key, you will be
presented with the following dialog box when you start LINGO:
Your license key may have been included in an email sent to you when you ordered your software.
The license key is a string of letters, symbols and numbers, separated into groups of four by hyphens
(e.g., r82m-XCW2-dZu?-%72S-fD?S-Wp@). Carefully enter the license key into the edit field,
including hyphens. License keys are case sensitive, so you must be sure to preserve the case of the
individual letters when entering your key. Click the OK button and, assuming the key was entered
correctly, LINGO will then start. In the future, you will be able to run LINGO directly without
entering the key.
Note: If you received your license key by email, then you have the option of cutting-and-pasting it
into the license key dialog box. Cut the key from the email that contains it with the Ctrl+C
key, then select the key field in LINGO dialog box and paste the key with the Ctrl+V key.
If you don’t have a key, you can choose to run LINGO in demo mode by clicking the Demo button. In
demo mode, LINGO has all the functionality of a standard version of LINGO with the one exception
that the maximum problem size is restricted. Demo licenses expire after 30 days.
GETTING STARTED 3
The outer window, labeled LINGO, is the main frame window. All other windows will be contained
within this window. The top of the frame window also contains all the command menus and the
command toolbar. See Chapter 5, Windows Commands, for details on the toolbar and menu
commands. The lower edge of the main frame window contains a status bar that provides various
pieces of information regarding LINGO's current state. Both the toolbar and the status bar can be
suppressed through the use of the LINGO|Options command.
The smaller child window labeled LINGO Model − LINGO1 is a new, blank model window. In the
next section, we will be entering a sample model directly into this window.
4 CHAPTER 1
Note: Each mathematical expression in LINGO is terminated with a semicolon. These semicolons
are required. Your model will not solve without them. For more information on the syntax of
LINGO, see below.
Next, we must input our constraints on line capacity and labor supply. The number of Standard and
Turbo computers produced must be constrained to the production line limits of 100 and 120,
respectively. Do this by entering the following two constraints just below the objective function:
STANDARD <= 100;
TURBO <= 120;
In words, the first constraint says the number of Standard computers produced daily (STANDARD)
must be less-than-or-equal-to (<=) the production line capacity of 100. Likewise, the second constraint
says the number of Turbo computers produced daily (TURBO) must be less-than-or-equal-to (<=) its
line capacity of 120.
Note: Since most computers do not have less-than-or-equal-to keys (≤), LINGO has adopted the
convention of using the two character symbol <= to denote ≤. As an alternative, you may
simply enter < to signify less-than-or-equal-to. In a similar manner, >= or > are used to
signify greater-than-or-equal-to (≥).
The final constraint on the amount of labor used can be expressed as:
STANDARD + 2 * TURBO <= 160;
Specifically, the total number of labor hours used (STANDARD + 2 * TURBO) must be
less-than-or-equal-to (<=) the amount of labor hours available of 160.
6 CHAPTER 1
After entering the above and entering comments to improve the readability of the model, your model
window should look like this:
LINGO lets you know there is a syntax error in your model, lists the line of the model it is in, and
points to the place in the line where it occurred. For more information on error codes, see Appendix B,
Error Messages.
8 CHAPTER 1
Solver Status Window
If there are no formulation errors during the compilation phase, LINGO will invoke the appropriate
internal solver to begin searching for the optimal solution to your model. When the solver starts, it
displays a solver status window on your screen resembling the following:
The solver status window is useful for monitoring the progress of the solver and the dimensions of
your model. The various fields are described in more detail below.
The solver status window also provides you with an Interrupt Solver button. Interrupting the solver
causes LINGO to halt the solver on the next iteration. In most cases, LINGO will be able to restore and
report the best solution found so far. The one exception is in the case of linear programming models
(i.e., linear models without integer variables). If a linear programming model is interrupted, the
solution returned will be meaningless and should be ignored. This should not be a problem because
linear programs generally solve quickly, thus minimizing the need to interrupt.
Note: You must be careful how you interpret solutions after interrupting the solver. These solutions
1) will definitely not be optimal, 2) may not be feasible to all the constraints, and 3) are
worthless if the model is a linear program.
GETTING STARTED 9
Next to the Interrupt Solver button is another button labeled Close. Hitting the Close button will close
the solver status window. This window can be reopened at any time by selecting the Window|Status
Window command.
At the bottom of the solver status window, you will find an Update Interval field. LINGO will update
the solver status window every n seconds, where n is the value contained in the Update Interval field.
You may set this interval to any value you desire. However, setting it to 0 will result in longer solution
times—LINGO will spend more time updating the solver status window than solving your model. On
larger models, LINGO may not always be able to update the solver status window on a regular
interval. So, don't be concerned if you sometimes must wait longer than the indicated interval.
Variables box
The Variables box shows the total number of variables in the model. The Variables box also displays
the number of the total variables that are nonlinear. A variable is considered to be nonlinear if it enters
into any nonlinear relationship in any constraint in the model. For instance, the constraint:
X + Y = 100;
would be considered linear because the graph of this function would be a straight line. On the other
hand, the nonlinear function:
X * Y = 100;
is quadratic and has a curved line as its graph. If we were to solve a model containing this particular
nonlinear constraint, the nonlinear variable count would be at least 2 to represent the fact that the two
variables X and Y appear nonlinearly in this constraint.
As another example, consider the constraint:
X * X + Y = 100;
In this case, X appears nonlinearly while Y appears as a linear variable. This constraint would not cause
Y to be counted as one of the nonlinear variables. See Chapter 14, On Mathematical Modeling, for
more information on the difference between linear and nonlinear equations.
The Variables box in the solver status window also gives you a count of the total number of integer
variables in the model. In general, the more nonlinear and integer variables your model has, the more
difficult it will be to solve to optimality in a reasonable amount of time. Pure linear models without
integer variables will tend to solve the fastest. For more details on the use of integer variables, refer to
Chapter 3, Using Variable Domain Functions.
The variable counts do not include any variables LINGO determines are fixed in value. For instance,
consider the following constraints:
X = 1;
X + Y = 3;
From the first constraint, LINGO determines X is fixed at the value of 1. Using this information in
constraint 2, LINGO determines Y is fixed at a value of 2. X and Y will then be substituted out of the
model and they will not contribute to the total variable count.
10 CHAPTER 1
Constraints box
The Constraints box shows the total constraints in the expanded model and the number of these
constraints that are nonlinear. A constraint is considered nonlinear if one or more variables appear
nonlinearly in the constraint.
LINGO searches your model for fixed constraints. A constraint is considered fixed if all the variables
in the constraint are fixed. Fixed constraints are substituted out of the model and do not add to the total
constraint count.
Nonzeroes box
The Nonzeros box shows the total nonzero coefficients in the model and the number of these that
appear on nonlinear variables. In a given constraint, only a small subset of the total variables typically
appears. The implied coefficient on all the non-appearing variables is zero, while the coefficients on
the variables that do appear will be nonzero. Thus, you can view the total nonzero coefficient count as
a tally of the total number of times variables appear in all the constraints. The nonlinear nonzero
coefficient count can be viewed as the number of times variables appear nonlinearly in all the
constraints.
State Field
When LINGO begins solving your model, the initial state of the current solution will be
"Undetermined". This is because the solver has not yet had a chance to generate a solution to your
model.
12 CHAPTER 1
Once the solver begins iterating, the state will progress to "Infeasible". In the infeasible state, LINGO
has generated tentative solutions, but none that satisfy all the constraints in the model.
Assuming a feasible solution exists, the solver will then progress to the "Feasible" state. In the feasible
state, LINGO has found a solution that satisfies all the constraints in your model, but the solver is not
yet satisfied it has found the best solution to your model.
Once the solver can no longer find better solutions to your model, it will terminate in either the
"Global Optimum" or "Local Optimum" state. If your model does not have any nonlinear constraints,
then any locally optimal solution will also be a global optimum. Thus, all optimized linear models will
terminate in the global optimum state. If, on the other hand, your model has one or more nonlinear
constraints, then any locally optimal solution may not be the best solution to your model. There may
be another "peak" that is better than the current one, but the solver's local search procedure is unable to
"see" the better peak. Thus, on nonlinear models, LINGO can terminate only in the local optimum
state. LINGO may, in fact, have a globally optimal solution, but, given the nature of nonlinear
problems, LINGO is unable to claim it as such. Given this fact, it is always preferred to formulate a
model using only linear constraints whenever possible. For more details on the concept of global vs.
local optimal points, refer to On Mathematical Modeling.
Note: LINGO’s optional global solver may be used to find globally optimal solutions to nonlinear
models. For more information on the global solver, refer to the Nonlinear Solver Tab help
topic.
If a model terminates in the "Unbounded" state, it means LINGO can improve the objective function
without bound. In real life, this would correspond to a situation where you can generate infinite profits.
Because such a situation is rare, if not impossible, you have probably omitted or misspecified some
constraints in your model.
Finally, the "Interrupted" state will occur when you prematurely interrupt LINGO's solver before it has
found the final solution to your model. The mechanics of interrupting the solver are discussed in more
detail above.
Objective Field
The Objective field gives the objective value for the current solution. If your model does not have an
objective function, then "N/A" will appear in this field.
Infeasibility Field
The Infeasibility field lists the amount that all the constraints in the model are violated by. Keep in
mind that this figure does not track the amount of any violations on variable bounds. Thus, it is
possible for the Infeasibility field to be zero while the current solution is infeasible due to violated
variable bounds. The LINGO solver may also internally scale a model such that the units of the
Infeasibility field no longer correspond to the unscaled version of the model. To determine whether
LINGO has found a feasible solution, you should refer to the State field discussed above.
Iterations Field
The Iterations field displays a count of the number of iterations completed thus far by LINGO's solver.
The fundamental operation performed by LINGO's solver is called iteration. Iteration involves finding
a variable, currently at a zero value, which would be attractive to introduce into the solution at a
GETTING STARTED 13
nonzero value. This variable is then introduced into the solution at successively larger values until
either a constraint is about to be driven infeasible or another variable is driven to zero. At this point,
the iteration process begins anew. In general, as a model becomes larger, it will require more iterations
to solve and each iteration will require more time to complete.
Steps Field
The information displayed in the Steps field depends on the particular solver that is running. The table
below explains:
Solver Steps Field Interpretation
Branch-and-Bound Number of branches in the branch-
and-bound tree.
Global Number of subproblem boxes
generated.
Multistart Number of solver restarts.
Active Field
This field pertains to the branch –and –bound and global solvers. It lists the number of open
subproblems remaining to be evaluated. The solver must run until this valve goes to zero.
This solution tells us CompuQuick should build 100 Standards and 30 Turbos each day to give them a
total daily profit of $14,500. Refer to the Examining the Solution section below for additional details
on the various fields in this report.
GETTING STARTED 15
The colon character (:) at the bottom of the screen in LINGO’s prompt for input. When you see the
colon prompt, LINGO is expecting a command. When you see the question mark prompt, you have
already initiated a command and LINGO is asking you to supply additional information related to this
command such as a number or a name. If you wish to “back out” of a command you have already
started, you may enter a blank line in response to the question mark prompt and LINGO will return
you to the command level colon prompt. All available commands are listed in Chapter 6,
Command-line Commands.
16 CHAPTER 1
The END command tells LINGO you are finished inputting the model. Once you enter the END
command and return to the colon prompt, the model is in memory and ready to be solved.
This solution tells us that CompuQuick should build 100 Standards and 30 Turbos each day to give
them a total daily profit of $14,500. Refer to the Examining the Solution section below for additional
details on the various fields in this report.
GETTING STARTED 17
To save your model to disk, issue the SAVE command followed by the name of a file to store your
model under. For example, the command:
SAVE MYFILE.LNG
saves a copy of the current model to the file titled MYFILE.LNG. The model may be retrieved for use
later with the TAKE command.
Please refer to Chapter 6, Command-line Commands, for more detailed information on these and other
commands.
Reduced Cost
In a LINGO solution report, you’ll find a reduced cost figure for each variable. There are two valid,
equivalent interpretations of a reduced cost.
First, you may interpret a variable’s reduced cost as the amount that the objective coefficient of the
variable would have to improve before it would become profitable to give the variable in question a
positive value in the optimal solution. For example, if a variable had a reduced cost of 10, the objective
coefficient of that variable would have to increase by 10 units in a maximization problem and/or
decrease by 10 units in a minimization problem for the variable to become an attractive alternative to
enter into the solution. A variable in the optimal solution, as in the case of STANDARD or TURBO,
automatically has a reduced cost of zero.
18 CHAPTER 1
Second, the reduced cost of a variable may be interpreted as the amount of penalty you would have to
pay to introduce one unit of that variable into the solution. Again, if you have a variable with a reduced
cost of 10, you would have to pay a penalty of 10 units to introduce the variable into the solution. In
other words, the objective value would fall by 10 units in a maximization model or increase by 10
units in a minimization model.
Reduced costs are valid only over a range of values for the variable in questions. For more information
on determining the valid range of a reduced cost, see the LINGO|Range command in Chapter 5,
Windows Commands.
Slack or Surplus
The Slack or Surplus column in a LINGO solution report tells you how close you are to satisfying a
constraint as an equality. This quantity, on less-than-or-equal-to (≤) constraints, is generally referred to
as slack. On greater-than-or-equal-to (≥) constraints, this quantity is called a surplus.
If a constraint is exactly satisfied as an equality, the slack or surplus value will be zero. If a constraint
is violated, as in an infeasible solution, the slack or surplus value will be negative. Knowing this can
help you find the violated constraints in an infeasible model—a model for which there doesn't exist a
set of variable values that simultaneously satisfies all constraints. Nonbinding constraints, will have
positive, nonzero values in this column.
In our CompuQuick example, note that row 3 (TURBO <= 120) has a slack of 90. Because the optimal
value of TURBO is 30, this row is 90 units from being satisfied as an equality.
Dual Price
The LINGO solution report also gives a dual price figure for each constraint. You can interpret the
dual price as the amount that the objective would improve as the right-hand side, or constant term, of
the constraint is increased by one unit. For example, in the CompuQuick solution, the dual price of 75
on row 4 means adding one more unit of labor would cause the objective to improve by 75, to a value
of 14,575.
Notice that “improve” is a relative term. In a maximization problem, improve means the objective
value would increase. However, in a minimization problem, the objective value would decrease if you
were to increase the right-hand side of a constraint with a positive dual price.
Dual prices are sometimes called shadow prices, because they tell you how much you should be
willing to pay for additional units of a resource. Based on our analysis, CompuQuick should be willing
to pay up to 75 dollars for each additional unit of labor.
As with reduced costs, dual prices are valid only over a range of values. Refer to the LINGO|Range
command in Chapter 5, Windows Commands, for more information on determining the valid range of a
dual price.
GETTING STARTED 19
V1 V2 V3 V4 V5 V6 V7 V8
For brevity, we included only 9 of the 48 terms in the objective. As one can see, entering such a
lengthy formula would be tedious and prone to errors. Extrapolate to the more realistic case where
vendors could number in the thousands, and it becomes apparent that scalar based modeling is
problematic at best.
If you are familiar with mathematical notation, you could express this long equation in a much more
compact manner as follows:
Minimize Σij COSTij • VOLUMEij
In a similar manner, LINGO’s modeling language allows you to express the objective function in a
form that is short, easy to type, and easy to understand. The equivalent LINGO statement is:
MIN = @SUM(LINKS(I,J): COST(I,J) * VOLUME(I,J));
In words, this says to minimize the sum of the shipping COST per widget times the VOLUME of
widgets shipped for all LINKS between the warehouses and vendors. The following table compares the
mathematical notation to the LINGO syntax for our objective function:
Math Notation LINGO Syntax
Minimize MIN =
Σij @SUM(LINKS(I, J):
COSTij COST(I, J)
• *
VOLUMEij VOLUME(I,J));
22 CHAPTER 1
The Constraints
With the objective function in place, the next step is to formulate the constraints. There are two sets of
constraints in this model. The first set guarantees that each vendor receives the number of widgets
required. We will refer to this first set of constraints as being the demand constraints. The second set of
constraints, called the capacity constraints, ensures no warehouse ships out more widgets than it has on
hand.
Starting with the demand constraint for Vendor 1, we need to sum up the shipments from all the
warehouses to Vendor 1 and set them equal to Vendor 1’s demand of 35 widgets. Thus, if we were
using scalar-based notation, we would need to construct the following:
VOLUME_1_1 + VOLUME_2_1 + VOLUME_3_1 +
VOLUME_4_1 + VOLUME_5_1 + VOLUME_6_1 = 35;
You would then need to type seven additional demand constraints, in a similar form, to cover all eight
vendors. Again, as one can see, this would be a tedious and error prone process. However, as with our
objective function, we can use LINGO’s set based modeling language to simplify our task.
Using mathematical notation, all eight demand constraints can be expressed in the single statement:
Σi VOLUMEij = DEMANDj , for all j in VENDORS
The corresponding LINGO modeling statement appears as follows:
@FOR(VENDORS(J):
@SUM(WAREHOUSES(I): VOLUME(I, J)) =
DEMAND(J));
This LINGO statement replaces all eight demand constraints. In words, this says for all VENDORS, the
sum of the VOLUME shipped from each of the WAREHOUSES to that vendor must equal the
corresponding DEMAND of the vendor. Notice how closely this statement resembles the mathematical
notation above as the following table shows:
Math Notation LINGO Syntax
For all j in VENDORS @FOR(VENDORS(J):
Σi @SUM(WAREHOUSES(I):
VOLUMEij VOLUME(I, J))
= =
DEMANDj DEMAND(J));
Now, we will move on to constructing the capacity constraints. In standard mathematical notation, the
six capacity constraints would be expressed as:
Σj VOLUMEij <= CAPi , for all i in WAREHOUSES
The equivalent LINGO statement for all capacity constraints would be:
@FOR(WAREHOUSES(I):
@SUM(VENDORS(J): VOLUME(I, J))<=
CAPACITY(I));
GETTING STARTED 23
In words, this says, for each member of the set WAREHOUSES, the sum of the VOLUME shipped to
each of the VENDORS from that warehouse must be less-than-or-equal-to the CAPACITY of the
warehouse.
Putting together everything we’ve done so far yields the following complete LINGO model:
MODEL:
MIN = @SUM(LINKS(I, J):
COST(I, J) * VOLUME(I, J));
@FOR(VENDORS(J):
@SUM(WAREHOUSES(I): VOLUME(I, J)) =
DEMAND(J));
@FOR(WAREHOUSES(I):
@SUM(VENDORS(J): VOLUME(I, J)) <=
CAPACITY(I));
END
Model: WIDGETS
However, we still need to define sets of objects used in the model (vendors, warehouses and shipping
arcs) as well as the data. We will do this is two additional model sections called the sets section and
the data section.
The second line says that the set WAREHOUSES has an attribute called CAPACITY. The following
line declares the vendor set and that it has an attribute called DEMAND.
24 CHAPTER 1
The final set, titled LINKS, represents the links in the shipping network. Each link has a COST and a
VOLUME attribute associated with it. The syntax used to define this set differs from the previous two.
By specifying:
LINKS( WAREHOUSES, VENDORS)
we are telling LINGO that the LINKS set is derived from the WAREHOUSES and VENDORS sets. In
this case, LINGO generates each ordered (warehouse, vendor) pair. Each of these 48 ordered pairs
becomes a member in the LINKS set. To help clarify this, we list selected members from the LINKS set
in the following table.
Member Index Shipping Arc
1 WH1ÆV1
2 WH1ÆV2
3 WH1ÆV3
… …
47 WH6ÆV7
48 WH6ÆV8
A nice feature of LINGO is that it will automatically generate the members of the LINKS set based on
the members of the WAREHOUSES and VENDORS sets, thereby saving us considerable work.
!attribute values;
CAPACITY = 60 55 51 43 41 52;
DEMAND = 35 37 22 32 41 32 43 38;
COST = 6 2 6 7 4 2 5 9
4 9 5 3 8 5 8 2
5 2 1 9 7 4 3 3
7 6 7 3 9 2 7 1
2 3 9 5 7 2 6 5
5 5 2 2 8 1 4 3;
ENDDATA
The data section begins with the keyword DATA: on a line by itself and ends with ENDDATA on a line
by itself.
Next, we input the list of warehouses and vendors. Had we preferred, we could have also used the
following shorthand notation to the same end:
!set members;
WAREHOUSES = WH1..WH6;
VENDORS = V1..V8;
GETTING STARTED 25
LINGO interprets the double-dots to mean that it should internally generate the six warehouses and
eight vendors.
Both the CAPACITY attribute of the set WAREHOUSES and DEMAND attribute of the set VENDORS
are initialized in a straightforward manner. The COST attribute of the two-dimensional set LINKS is a
little bit trickier, however. When LINGO is initializing a multidimensional array in a data section, it
increments the outer index the fastest. Thus, in this particular example, COST( WH1, V1) is initialized
first, followed by COST( WH1, V2) through COST (WH1, V8). Then, the next one to be initialized with
be COST(WH2, V1), and so on.
In this particular example, we have isolated all the model’s data within a single data section. Given
that the data is the most likely feature to change from one run of a model to the next, isolating data, as
we have done here, makes modifications considerably easier. Contrast this to how difficult it would be
to track down and change the data in a large, scalar model where data is spread throughout all the
constraints of the model.
In order to facilitate data management further, LINGO has the ability to import data from external
sources. More specifically, a LINGO model can import data from external text files, establish real-time
OLE links to Excel, and/or create ODBC links to databases.
26 CHAPTER 1
Putting together the data section, the sets section, the objective, and the constraints, the completed
model is as follows:
MODEL:
! A 6 Warehouse 8 Vendor Transportation Problem;
SETS:
WAREHOUSES: CAPACITY;
VENDORS: DEMAND;
LINKS( WAREHOUSES, VENDORS): COST, VOLUME;
ENDSETS
! Here is the data;
DATA:
!set members;
WAREHOUSES = WH1 WH2 WH3 WH4 WH5 WH6;
VENDORS = V1 V2 V3 V4 V5 V6 V7 V8;
!attribute values;
CAPACITY = 60 55 51 43 41 52;
DEMAND = 35 37 22 32 41 32 43 38;
COST = 6 2 6 7 4 2 5 9
4 9 5 3 8 5 8 2
5 2 1 9 7 4 3 3
7 6 7 3 9 2 7 1
2 3 9 5 7 2 6 5
5 5 2 2 8 1 4 3;
ENDDATA
! The objective;
MIN = @SUM( LINKS( I, J):
COST( I, J) * VOLUME( I, J));
! The demand constraints;
@FOR( VENDORS( J):
@SUM( WAREHOUSES( I): VOLUME( I, J)) =
DEMAND( J));
! The capacity constraints;
@FOR( WAREHOUSES( I):
@SUM( VENDORS( J): VOLUME( I, J)) <=
CAPACITY( I));
END
Model: WIDGETS
Note that we have again added comments to improve the readability of the model. The model
is named WIDGETS, and can be found in the SAMPLES subdirectory off the main LINGO
directory.
Press down on the arrow button in the Attribute or Row Name field and select VOLUME from the list
of names in the drop-down box. To suppress the printing of variables with zero value, click on the
Nonzeros Only checkbox. Once you have done this, the dialog box should resemble:
28 CHAPTER 1
Now, click the OK button and you will be presented with the following report that contains the
nonzero VOLUME variables:
If you are running LINGO on a platform other than Windows, you can generate the same report by
issuing the NONZERO VOLUME command.
GETTING STARTED 29
Summary
This section has begun to demonstrate the virtues of LINGO’s set based modeling language. By
moving to a set based approach to modeling, you will find that your models become easier to build,
easier to understand, and easier to maintain. Set based modeling takes a little more work to become
comfortable with, but the benefits should substantially outweigh the extra effort involved in the
learning process. We will delve further into the concepts of set based modeling in the following
chapter, Using Sets.
!attribute values;
CAPACITY = 60 55 51 43 41 52;
DEMAND = 35 37 22 32 41 32 43 38;
COST = 6 2 6 7 4 2 5 9
4 9 5 3 8 5 8 2
5 2 1 9 7 4 3 3
7 6 7 3 9 2 7 1
2 3 9 5 7 2 6 5
5 5 2 2 8 1 4 3;
ENDDATA
! The objective;
[OBJECTIVE] MIN = @SUM( LINKS( I, J):
COST( I, J) * VOLUME( I, J));
! The demand constraints;
@FOR( VENDORS( J): [DEMAND_ROW]
@SUM( WAREHOUSES( I): VOLUME( I, J)) =
DEMAND( J));
! The capacity constraints;
@FOR( WAREHOUSES( I): [CAPACITY_ROW]
@SUM( VENDORS( J): VOLUME( I, J)) <=
CAPACITY( I));
END
WIDGETS with Constraint Names
The row section of the solution report is now considerably easier to interpret:
Row Slack or Surplus Dual Price
OBJECTIVE 664.0000 1.000000
DEMAND_ROW(V1) 0.0000000 -4.000000
DEMAND_ROW(V2) 0.0000000 -5.000000
DEMAND_ROW(V3) 0.0000000 -4.000000
DEMAND_ROW(V4) 0.0000000 -3.000000
DEMAND_ROW(V5) 0.0000000 -7.000000
DEMAND_ROW(V6) 0.0000000 -3.000000
DEMAND_ROW(V7) 0.0000000 -6.000000
DEMAND_ROW(V8) 0.0000000 -2.000000
CAPACITY_ROW(WH1) 0.0000000 3.000000
CAPACITY_ROW(WH2) 22.00000 0.000000
CAPACITY_ROW(WH3) 0.0000000 3.000000
CAPACITY_ROW(WH4) 0.0000000 1.000000
CAPACITY_ROW(WH5) 0.0000000 2.000000
CAPACITY_ROW(WH6) 0.0000000 2.000000
Row Report for WIDGETS with Constraint Names
GETTING STARTED 31
Note that each row now has a name rather than a simple index number. Furthermore, if the constraint
is generated over a set using the @FOR function, LINGO qualifies the constraint name by appending
the corresponding set member name in parentheses.
Model Title
You can insert a title for a model anywhere you would normally enter a constraint. If a title is
included, it will be printed at the top of solution reports. The title is also used as a default argument in
the @ODBC function (see Chapter 10, Interfacing with Databases).
The model’s title must begin with the keyword TITLE and end with a semicolon. All text between
TITLE and the semicolon will be taken as the title of the model.
In the following, we have added a title to the beginning of the WIDGETS model:
MODEL:
TITLE Widgets;
! A 6 Warehouse 8 Vendor Transportation Problem;
SETS:
WAREHOUSES: CAPACITY;
.
.
.
Excerpt from WIDGETS Model with a Title
Note that when we display the solution report, the title is now displayed along the top:
Model Title: Widgets
Note 2: In some versions of LINGO, the Nonlinear Variable limit will be 0 if you have not purchased
the nonlinear option for your copy of LINGO. Similarly, the global variable limit will be o if
the global solver option is not enabled.
Note 3: LINGO has two other implicit limits not given by the table above—memory and time. Large
models may require more memory to solve than is available on your system, or they may
require more time to solve than one would normally be willing to wait. So, when building
large models, be aware that just because your model falls within LINGO’s limits there is no
guarantee it will be solvable in a reasonable amount of time on a particular machine.
GETTING STARTED 33
35
36 CHAPTER 2
Types of Sets
LINGO recognizes two kinds of sets: primitive and derived.
A primitive set is a set composed only of objects that can’t be further reduced. In the Wireless Widgets
example (page 19), the WAREHOUSES set, which is composed of six warehouses, is a primitive set.
Likewise, the set composed of eight vendors is a primitive set.
A derived set is defined using one or more other sets. In other words, a derived set derives its members
from other preexisting sets. Again, using the Wireless Widgets example, the set composed of the links
between the six warehouses and eight vendors (LINKS) is a derived set. It derives its members from
the unique pairs of members of the WAREHOUSES and VENDORS sets. Although the LINKS set is
derived solely from primitive sets, it is also possible to build derived sets from other derived sets as
well. See the section below, Defining Derived Sets, for more information.
Note: The use of square brackets indicates an item is optional. In this particular case, a primitive
set’s attribute_list and member_list are both optional.
The setname is a name you choose to designate the set. It should be a descriptive name that is easy to
remember. The set name must conform to standard LINGO naming conventions. In other words, the
name must begin with an alphabetic character, which may be followed by up to 31 alphanumeric
characters or the underscore (_). LINGO does not distinguish between upper and lowercase characters
in names.
A member_list is a list of the members that constitute the set. If the set members are included in the set
definition, they may be listed either explicitly or implicitly. If set members are not included in the set
USING SETS 37
definition, then they may be defined subsequently in a data section of the model. For details on
defining set members in a data section, refer to Introduction to the Data Section.
When listing members explicitly, you enter a unique name for each member, optionally separated by
commas. As with set names, member names must also conform to standard naming conventions. In the
Wireless Widgets model, we used an explicit member list to define the WAREHOUSES set as follows:
WAREHOUSES / WH1 WH2 WH3 WH4 WH5 WH6/: CAPACITY;
When using implicit set member lists, you do not have to list a name for each set member. Use the
following syntax when using an implicit set member list:
setname / member1..memberN / [: attribute_list];
where member1 is the name of the first member in the set and memberN is the name of the last
member. LINGO automatically generates all the intermediate member names between member1 and
memberN. While this can be a very compact and convenient method for building a primitive set, there
is one catch in that only certain formats of names are accepted for the initial and terminal member
names. The following table details the available options:
Implicit Member List Example Set Members
Format
1..n 1..5 1, 2, 3, 4, 5
stringM..stringN TRUCKS3..TRUCKS204 TRUCKS3, TRUCKS4, …,
TRUCKS204
dayM..dayN MON..FRI MON, TUE, WED, THU,
FRI
monthM..monthN OCT..JAN OCT, NOV, DEC, JAN
monthYearM..monthYearN OCT2001..JAN2002 OCT2001, NOV2001,
DEC2001, JAN2002
When using the 1..n format, n may be any positive integer value, and the initial member must always
be a 1.
The stringM..stringN format allows you to use any string to start both the initial and terminal member
names as long as the string conforms to standard LINGO naming conventions. M and N must be
nonnegative and integer, such that M ≤ N.
The dayM..dayN format allows you to choose the initial and terminal member names for the names of
the days of the week. All names are abbreviated to three characters. Thus, the available options are:
Mon, Tue, Wed, Thu, Fri, Sat, and Sun.
The monthM..monthN format allows you to select from the months of the year, where all names are
abbreviated to three characters. The available options are: Jan, Feb, Mar, Apr, May, Jun, Jul, Aug, Sep,
Oct, Nov, and Dec.
The monthYearM..monthYearN option allows you to specify a month and a four digit year.
As further illustration, in the Wireless Widgets example, we could have also defined the
WAREHOUSES set as:
WAREHOUSES / 1..6/: CAPACITY;
38 CHAPTER 2
As an alternative, when using this 1..n form of implicit definition, you may also place the length of the
set in a data section, and then reference this length in a subsequent sets section as we do here:
DATA:
NUMBER_OF_WH = 6;
ENDDATA
SETS:
WAREHOUSES / 1..NUMBER_OF_WH/: CAPACITY;
ENDSETS
Set members may have one or more attributes specified in the attribute_list of the set definition. An
attribute is simply some property each member of the set displays. For instance, in the WAREHOUSES
set above, there is a single attribute titled CAPACITY, which is used to represent the shipping capacity
of the WAREHOUSES. Attribute names must follow standard naming conventions and be separated by
commas.
For illustration, suppose our warehouses had additional attributes related to the their location and the
number of loading docks. These additional attributes could be added to the attribute list of the set
declaration as follows:
WAREHOUSES / 1..6/: CAPACITY, LOCATION, DOCKS;
Summary
In summary, keep in mind that LINGO recognizes two types of sets—primitive and derived.
Primitive sets are the fundamental objects in a model and can’t be broken down into smaller
components. Primitive sets can be defined using either an explicit or implicit list. When using an
explicit list, you enter each member individually in the set member list. With an implicit list, you enter
the initial and terminal set members and LINGO generates all the intermediate members.
USING SETS 41
Derived sets, on the other hand, are created from other component sets. These component sets are
referred to as the parents of the derived set, and may be either primitive or derived. A derived set can
be either sparse or dense. Dense sets contain all combinations of the parent set members (sometimes
this is also referred to as the Cartesian product or cross of the parent sets). Sparse sets contain only a
subset of the cross of the parent sets and may be defined by two methodsexplicit listing or
membership filter. The explicit listing method involves listing the members of the sparse set. The
membership filter method allows you to specify the sparse set members compactly through the use of a
logical condition all members must satisfy. The relationships amongst the various set types are
illustrated in the graph below:
Sets
Derived Primitive
Explicit Implicit
Sparse Dense
List List
Explicit Membership
List Filter
We have two attributes, X and Y, defined on the set SET1. The three values of X are set to 1, 2, and 3,
while Y is set to 4, 5, and 6. We could have also used the following compound data statement to the
same end:
MODEL:
SETS:
SET1: X, Y;
ENDSETS
DATA:
SET1 X Y = A 1 4
B 2 5
C 3 6;
ENDDATA
END
An important fact to remember is that when LINGO reads a compound data statement's value list, it
assigns the first n values in the list to the first position of each of the n objects in the object list, the
second n values to the second position of each of the n objects, and so on. In other words, LINGO is
expecting the input data in column format rather than row format, which mirrors the flat file approach
used in relational databases.
This section has served to give you a brief introduction into the use of the data section. In Data and
Init Sections, you will learn more about the capabilities of the data section. You will learn data does
not have to actually reside in the data section as shown in examples here. In fact, your data section can
have OLE links to Excel, ODBC links to databases, and connections to text based data files.
Each vendor of the VENDORS set has a corresponding DEMAND. We could sum up the values of the
DEMAND attribute by adding the following expression after the ENDDATA statement:
TOTAL_DEMAND = @SUM(VENDORS(J): DEMAND(J));
LINGO evaluates the @SUM function by first initializing an internal accumulator to zero. LINGO then
begins looping over the members in the VENDORS set. The set index variable, J, is set to the first
member of VENDORS (i.e., V1) and DEMAND (V1) is then added to the accumulator. This process
continues until all DEMAND values have been added to the accumulator. The value of the sum is then
stored in the TOTAL_DEMAND variable.
Since all the attributes in our expression list (in this case, only DEMAND appears in the expression
list) are defined on the index set (VENDORS), we could have alternatively written our sum as:
TOTAL_DEMAND = @SUM(VENDORS: DEMAND);
In this case, we have dropped the superfluous index set list and the index on DEMAND. When an
expression uses this shorthand, we say the index list is implied. Implied index lists are not allowed
when attributes in the expression list have different parent sets.
Next, suppose we want to sum the first three elements of the attribute DEMAND. We can use a
conditional qualifier on the set index to accomplish this as follows:
DEMAND_3 = @SUM(VENDORS(J)|J #LE# 3: DEMAND(J));
The #LE# symbol is called a logical operator (see p. 287 for more details). This operator compares the
operand on the left (J) with the one on the right (3), and returns true if the left operand is
less-than-or-equal-to the one on the right. Otherwise, it returns false. Therefore, when LINGO
computes the sum this time, it plugs the set index variable, J, into the conditional qualifier J #LE# 3. If
the conditional qualifier evaluates to true, DEMAND(J) will be added to the sum. The end result is
LINGO sums up the first three terms in DEMAND, omitting the fourth and fifth terms, for a total sum
of 9.
Note: Before leaving this example, one subtle aspect to note in this last sum expression is the value
that the set index J is returning. Note, we are comparing the set index variable to the quantity
3 in the conditional qualifier J #LE# 3. In order for this to be meaningful, J must represent a
numeric value. Because a set index is used to loop over set members, one might imagine a set
index is merely a placeholder for the current set member. In a sense, this is true, but what set
indices really return is the index of the current set member in its parent primitive set. The
index returned is one-based. In other words, the value 1 is returned when indexing the first set
member, 2 when indexing the second, and so on. Given that set indices return a numeric
value, they may be used in arithmetic expressions along with other variables in your model.
USING SETS 45
To find the minimum and maximum DEMAND, all one need do is add the two expressions:
MIN_DEMAND = @MIN( VENDORS( J): DEMAND( J));
MAX_DEMAND = @MAX( VENDORS( J): DEMAND( J));
The resulting model with the new statements in bold would then be as follows:
MODEL:
SETS:
VENDORS: DEMAND;
ENDSETS
DATA:
VENDORS, DEMAND = V1,5 V2,1 V3,3 V4,4 V5,6;
ENDDATA
MIN_DEMAND = @MIN( VENDORS( J): DEMAND( J));
MAX_DEMAND = @MAX( VENDORS( J): DEMAND( J));
END
As with the @SUM example, we can use an implied index list since the attributes are defined on the
index set. Using implied indexing, we can recast our expressions as:
MIN_DEMAND = @MIN( VENDORS: DEMAND);
MAX_DEMAND = @MAX( VENDORS: DEMAND);
In either case, when we solve this model, LINGO returns the expected minimum and maximum
DEMAND of:
Variable Value
MIN_DEMAND 1.000000
MAX_DEMAND 6.000000
For illustration purposes, suppose we had just wanted to compute the minimum and maximum values
of the first three elements of DEMAND. As with the @SUM example, all we need do is add the
conditional qualifier J #LE# 3. We then have:
MIN_DEMAND3 =
@MIN( VENDORS( J) | J #LE# 3: DEMAND( J));
MAX_DEMAND3 =
@MAX( VENDORS( J) | J #LE# 3: DEMAND( J));
46 CHAPTER 2
with solution:
Variable Value
MIN_DEMAND3 1.000000
MAX_DEMAND3 5.000000
Since the reciprocal of zero is not defined, we could put a conditional qualifier on our @FOR
statement that causes us to skip the reciprocal computation whenever a zero is encountered. The
following @FOR statement accomplishes this:
@FOR(NUMBERS(I)| VALUE(I) #NE# 0:
RECIPROCAL(I) = 1 / VALUE(I)
);
The conditional qualifier (listed in bold) tests to determine if the value is not equal (#NE#) to zero. If
so, the computation proceeds.
This was just a brief introduction to the use of the @FOR statement. There will be many additional
examples in the sections to follow.
Here we have a system of three components arranged in a series. The probability that each component
functions successfully (.95, .99, and .98) is loaded into attribute P in the model’s data section. We
then compute the probability that the entire system will fail, P_FAIL, by taking the product of the
component probabilities and subtracting it from 1:
P_FAIL = 1 - @PROD( COMPONENTS( I): P( I));
As an aside, an interesting feature to note about this model is that we never initialized the
COMPONENTS set. When LINGO sees that an attribute of an undefined primitive set being
initialized to n values in a data section, it automatically initializes the parent primitive set to contain
the members: 1, 2, …, n. So, in this example, LINGO automatically assigned the member 1, 2 and 3 to
the COMPONENTS set.
48 CHAPTER 2
Specifically, for each vendor, we sum up the shipments going from all the warehouses to that vendor
and set the quantity equal to the vendor’s demand. In this case, we have nested an @SUM function
within an @FOR function.
@SUM, @MAX, and @MIN can be nested within any set looping function. @FOR functions, on the
other hand, may only be nested within other @FOR functions.
Summary
This section demonstrated that set looping functions can be very powerful and can simplify the
modeler's task. If you aren't making use of sets and set looping functions, you will have a considerably
more difficult time building your models. Furthermore, the difficulty will grow dramatically as the
sizes of your models grow.
We now know how to create sets, how to initialize sets and attributes using the data section, and how
to work with sets using set looping functions. At this point, we now have the ability to start
constructing some meaningful example models.
The Formulation
The first question to consider when building a set based model is, "What are the relevant sets and their
attributes?". In this model, we have a single primitive set, the days of the week. If we call this set
DAYS, we can begin by writing our sets section as:
SETS:
DAYS;
ENDSETS
Next, we can add a data section to initialize the set members of the DAYS set:
SETS:
DAYS;
ENDSETS
DATA:
DAYS = MON TUE WED THU FRI SAT SUN;
ENDDATA
Alternatively, we could use LINGO’s implicit set definition capability and express this equivalently as:
SETS:
DAYS;
ENDSETS
DATA:
DAYS = MON..SUN;
ENDDATA
50 CHAPTER 2
We will be concerned with two attributes of the DAYS set. The first is the number of staff required on
each day, and the second is the number of staff to start on each day. If we call these attributes
REQUIRED and START, then we may add them to the sets section to get:
SETS:
DAYS: REQUIRED, START;
ENDSETS
After defining the sets and attributes, it is useful to determine which of the attributes are data, and
which are decision variables. In this model, the REQUIRED attribute is given to us and is, therefore,
data. The START attribute is something we need to determine and constitutes the decision variables.
Once you've identified the data in the model, you may go ahead and initialize it. We can do this by
extending the data section as follows:
DATA:
DAYS = MON TUE WED THU FRI SAT SUN;
REQUIRED = 20 16 13 16 19 14 12;
ENDDATA
We are now at the point where we can begin entering the model's mathematical relations (i.e., the
objective and constraints). Let's begin by writing out the mathematical notation for the objective. Our
objective is to minimize the total number of employees we start during the week. Using standard
mathematical notation, this objective may be expressed as:
Minimize: ∑i STARTi
The equivalent LINGO statement is very similar. Substitute "MIN=" for "Minimize:" and "@SUM(
DAYS( I):" for ∑i and we have:
MIN = @SUM( DAYS( I): START( I));
Now, all that is left is to come up with our constraints. There is only one set of constraints in this
model. Namely, we must have enough staff on duty each day to meet or exceed staffing requirements.
In words, what we want is:
Staff on duty today ≥ Staff required today, for each day of the week
The right-hand side of this expression, Staff required today, is easy to calculate. It is simply the
quantity REQUIRED( I). The left-hand side, Staff on duty today, is a bit trickier to compute. Given that
all employees are on a "five day on, two day off" schedule, the number of employees working today is:
Number working today = Number starting today +
Number starting 1 day ago + Number starting 2 days ago +
Number starting 3 days ago + Number starting 4 days ago.
In other words, to compute the number of employees working today, we sum up the number of people
starting today plus those starting over the previous four days. The number of employees starting five
and six days back don't count because they are on their days off. So, using mathematical notation, what
one might consider doing is adding the constraint:
∑i = j-4, j STARTi ≥ REQUIRED j , for j∈ DAYS
USING SETS 51
Translating into LINGO notation, we can write this as:
@FOR( DAYS( J):
@SUM( DAYS( I) | I #LE# 5: START( J - I + 1))
>= REQUIRED( J)
);
In words, the LINGO statement says, for each day of the week, the sum of the employees starting over
the five day period beginning four days ago and ending today must be greater-than-or-equal-to the
required number of staff for the day. This sounds correct, but there is a slight problem. If we try to
solve our model with this constraint we get the error message:
To see why we get this error message, consider what happens on Thursday. Thursday has an index of 4
in our set DAYS. As written, the staffing constraint for Thursday will be:
START( 4 - 1 + 1) + START( 4 - 2 + 1) +
START( 4 - 3 + 1) + START( 4 - 4 + 1) +
START( 4 - 5 + 1) >= REQUIRED( 4);
Simplifying, we get:
START( 4) + START( 3) +
START( 2) + START( 1) +
START( 0) >= REQUIRED( 4);
The START( 0) term is the root of our problem. START is defined for days 1 through 7. START( 0) does
not exist. An index of 0 on START is considered "out of range.”
We would like to have any indices less-than-or-equal-to 0 wrap around to the end of the week.
Specifically, 0 would correspond to Sunday (7), -1 to Saturday (6), and so on. LINGO has a function
that does just this: @WRAP.
The @WRAP function takes two argumentscall them INDEX and LIMIT. Formally speaking,
@WRAP returns J such that J = INDEX - K * LIMIT, where K is an integer such that J is in the interval
[1, LIMIT]. Informally speaking, @WRAP will subtract or add LIMIT to INDEX until it falls in the
range 1 to LIMIT. Therefore, this is just what we need to "wrap around" an index in multiperiod
planning models.
52 CHAPTER 2
Incorporating the @WRAP function, we get the corrected, final version of our staffing constraint:
@FOR( DAYS( J):
@SUM( DAYS( I) | I #LE# 5:
START( @WRAP( J - I + 1, 7)))
>= REQUIRED( J)
);
The Solution
Below is our staffing model in its entirety:
MODEL:
SETS:
DAYS: REQUIRED, START;
ENDSETS
DATA:
DAYS = MON TUE WED THU FRI SAT SUN;
REQUIRED = 20 16 13 16 19 14 12;
ENDDATA
The Formulation
The primitive sets in this model are the nut types and the brands of mixed nuts. We can add them to the
sets section as follows:
SETS:
NUTS / PEANUTS, CASHEWS/: SUPPLY;
BRANDS / PAWN, KNIGHT, BISHOP, KING/:
PRICE, PRODUCE;
ENDSETS
The NUTS set has the single SUPPLY attribute, which we will use to store the daily supply of nuts in
pounds. The BRANDS set has PRICE and PRODUCE attributes, where PRICE stores the selling price
of the brands and PRODUCE represents the decision variables of how many pounds of each brand to
produce per day.
We need one more set, which is the dense derived set we have been promising. In order to input the
brand formulas, we will need a two dimensional table defined on the nut types and the brands. To do
this, we will generate a derived set from the cross of the NUTS and BRANDS sets. Adding this derived
set, we get the completed sets section:
SETS:
NUTS / PEANUTS, CASHEWS/: SUPPLY;
BRANDS / PAWN, KNIGHT, BISHOP, KING/:
PRICE, PRODUCE;
FORMULA(NUTS, BRANDS): OUNCES;
ENDSETS
We have titled the derived set FORMULA. It has the single OUNCES attribute, which will be used to
store the ounces of nuts used per pound of each brand. Since we have not specified the members of this
derived set, LINGO assumes we want the complete, dense set that includes all pairs of nuts and brands.
USING SETS 55
Now that our sets are defined, we can move on to building the data section. We initialize SUPPLY,
PRICE, and OUNCES in the data section as follows:
DATA:
SUPPLY = 750 250;
PRICE = 2 3 4 5;
OUNCES = 15 10 6 2
1 6 10 14;
ENDDATA
With the sets and data established, we can begin to enter our objective function and constraints. The
objective function of maximizing total revenue is straightforward. We can express this as:
MAX = @SUM(BRANDS(I):
PRICE(I) * PRODUCE(I));
Our model has only one class of constraints. Namely, we can’t use more nuts than we are supplied
with on a daily basis. In words, we would like to ensure that:
For each nut i, the number of pounds of nut i used must be less-than-or-equal-to the
supply of nut i.
We can express this in LINGO as:
@FOR(NUTS(I):
@SUM(BRANDS(J):
OUNCES(I, J) * PRODUCE(J) / 16) <=
SUPPLY(I)
);
We divide the sum on the left-hand side by 16 to convert from ounces to pounds.
The Solution
Our completed blending model is:
SETS:
NUTS / PEANUTS, CASHEWS/: SUPPLY;
BRANDS / PAWN, KNIGHT, BISHOP, KING/:
PRICE, PRODUCE;
FORMULA(NUTS, BRANDS): OUNCES;
ENDSETS
DATA:
SUPPLY = 750 250;
PRICE = 2 3 4 5;
OUNCES= 15 10 6 2
1 6 10 14;
ENDDATA
MAX = @SUM(BRANDS(I):
PRICE(I) * PRODUCE(I));
@FOR(NUTS(I):
@SUM(BRANDS(J):
OUNCES(I, J) * PRODUCE(J) / 16) <=
SUPPLY(I)
);
Model: CHESS
56 CHAPTER 2
An abbreviated solution report to the model follows.
Global optimal solution found.
Objective value: 2692.308
Total solver iterations: 2
Forecast Schedule
Demand Production Run Cost Out
Finalize
Design
The Formulation
We will need a primitive set to represent the tasks of the project. We can add such a set to the model
using the set definition:
TASKS / DESIGN, FORECAST, SURVEY, PRICE,
SCHEDULE, COSTOUT, TRAIN/: TIME, ES, LS, SLACK;
We have associated four attributes with the TASKS set. The definitions of the attributes are:
TIME Time to complete the task
ES Earliest possible start time for the task
LS Latest possible start time for the task
SLACK Difference between LS and ES for the task
58 CHAPTER 2
The TIME attribute is given to us as data. We will compute the values of the remaining three attributes.
If a task has a 0 slack time, it means the task must start on time or the whole project will be delayed.
The collection of tasks with 0 slack time constitutes the critical path for the project.
In order to compute the start times for the tasks, we will need to examine the precedence relations.
Thus, we will need to input the precedence relations into the model. The precedence relations can be
viewed as a list of ordered pairs of tasks. For instance, the fact that the DESIGN task must be
completed before the FORECAST task could be represented as the ordered pair (DESIGN,
FORECAST). Creating a two-dimensional derived set on the TASKS set will allow us to input the list
of precedence relations. Specifically, we add the set definition:
PRED(TASKS, TASKS) /
DESIGN,FORECAST,
DESIGN,SURVEY,
FORECAST,PRICE,
FORECAST,SCHEDULE,
SURVEY,PRICE,
SCHEDULE,COSTOUT,
PRICE,TRAIN,
COSTOUT,TRAIN /;
Keep in mind that the first member of this set is the ordered pair (DESIGN, FORECAST)not just the
single task DESIGN. Therefore, this set has a total of 8 members that all correspond to an arc in the
precedence relations diagram.
The set PRED is the sparse derived set with an explicit listing we want to highlight in this example.
The set is a subset derived from the cross of the TASKS set upon itself. The set is sparse because it
contains only 8 out of 49 possible members found in the complete cross of TASKS on TASKS. The set
is said to be an “explicit list” set, because we have explicitly listed the members we want included in
the set. Explicitly listing the members of a sparse set may not be convenient in cases where there are
thousands of members to select from, but it does make sense whenever set membership conditions are
not well defined and the sparse set size is small relative to the dense alternative.
Next, we can input the task times in the data section by including:
DATA:
TIME = 10, 14, 3, 3, 7, 4, 10;
ENDDATA
Now, with our sets and data established, we can turn our attention to building the formulas of the
model. We have three attributes to compute: earliest start (ES), latest start (LS), and slack time
(SLACK). The trick is computing ES and LS. Once we have these times, SLACK is merely the
difference of the two.
Let’s start by coming up with a formula to compute ES. A task cannot begin until all its predecessor
tasks are completed. Thus, if we find the latest finishing time of all predecessors to a task, then we
have also found its earliest start time. Therefore, in words, the earliest start time for task t is equal to
the maximum over all predecessors of task t of the sum of the earliest start time of the predecessor plus
its completion time. The corresponding LINGO notation is:
@FOR(TASKS(J)| J #GT# 1:
ES(J) = @MAX(PRED(I, J): ES(I) + TIME(I))
);
USING SETS 59
Note that we skip the computation for the first task by adding the conditional qualifier J #GT# 1. We
do this because the first task has no predecessors. We will give the first task an arbitrary start time as
shown below.
Computing LS is slightly trickier, but very similar to ES. In words, the latest time for task t to start is
the minimum over all successor tasks of the sum of the successor’s earliest start minus the time to
perform task t. If task t starts any later than this, it will prohibit at least one successor from starting at
its earliest start time. Converting into LINGO syntax gives:
@FOR(TASKS(I)| I #LT# LTASK:
LS(I) = @MIN(PRED(I, J): LS(J) - TIME(I))
);
Here, we omit the computation for the last task since it has no successor tasks.
Computing slack time is just the difference between LS and ES, and may be written as:
@FOR(TASKS(I): SLACK(I) = LS(I) - ES(I));
We can set the start time of the first task to some arbitrary value. For our purposes, we will set it to 0
with the statement:
ES(1) = 0;
We have now input formulas for computing the values of all the variables with the exception of the
latest start time for the last task. It turns out, if the last project were started any later than its earliest
start time, the entire project would be delayed. So, by definition, the latest start time for the last project
is equal to its earliest start time. We can express this in LINGO using the equation:
LS(7) = ES(7);
This would work, but it’s probably not the best way to express the relation. Suppose you were to add
some tasks to your model. You’d have to change the 7 in this equation to the new number of tasks was.
The whole idea behind LINGO’s set based modeling language is the equations in the model should be
independent of the data. Expressing the equation in this form violates data independence. Here’s a
better way to do it:
LTASK = @SIZE(TASKS);
LS(LTASK) = ES(LTASK);
The @SIZE function returns the size of a set. In this case, it will return the value 7, as desired.
However, if we changed the number of tasks, @SIZE would also return the new, correct value. Thus,
we preserve the data independence of our model’s equations.
60 CHAPTER 2
The Solution
The entire PERT formulation and portions of its solution appear below:
SETS:
TASKS / DESIGN, FORECAST, SURVEY, PRICE,
SCHEDULE, COSTOUT, TRAIN/: TIME, ES, LS, SLACK;
PRED(TASKS, TASKS) /
DESIGN,FORECAST,
DESIGN,SURVEY,
FORECAST,PRICE,
FORECAST,SCHEDULE,
SURVEY,PRICE,
SCHEDULE,COSTOUT,
PRICE,TRAIN,
COSTOUT,TRAIN /;
ENDSETS
DATA:
TIME = 10, 14, 3, 3, 7, 4, 10;
ENDDATA
@FOR(TASKS(J)| J #GT# 1:
ES(J) = @MAX(PRED(I, J): ES(I) + TIME(I))
);
@FOR(TASKS(I)| I #LT# LTASK:
LS(I) = @MIN(PRED(I, J): LS(J) - TIME(I));
);
@FOR(TASKS(I): SLACK(I) = LS(I) - ES(I));
ES(1) = 0;
LTASK = @SIZE(TASKS);
LS(LTASK) = ES(LTASK);
Model: PERT
USING SETS 61
Feasible solution found at step: 0
Variable Value
LTASK 7.000000
ES(DESIGN) 0.0000000
ES(FORECAST) 10.00000
ES(SURVEY) 10.00000
ES(PRICE) 24.00000
ES(SCHEDULE) 24.00000
ES(COSTOUT) 31.00000
ES(TRAIN) 35.00000
LS(DESIGN) 0.0000000
LS(FORECAST) 10.00000
LS(SURVEY) 29.00000
LS(PRICE) 32.00000
LS(SCHEDULE) 24.00000
LS(COSTOUT) 31.00000
LS(TRAIN) 35.00000
SLACK(DESIGN) 0.0000000
SLACK(FORECAST) 0.0000000
SLACK(SURVEY) 19.00000
SLACK(PRICE) 8.000000
SLACK(SCHEDULE) 0.0000000
SLACK(COSTOUT) 0.0000000
SLACK(TRAIN) 0.0000000
Solution to PERT
The interesting values are the slacks for the tasks. Both SURVEY and PRICE have slack in their start
times of 19 weeks and 8 weeks, respectively. Their start times may be delayed by as much as these
slack values without compromising the completion time of the entire project. The tasks DESIGN,
FORECAST, SCHEDULE, COSTOUT, and TRAIN, on the other hand, have 0 slack times. These tasks
constitute the critical path for the project and, if any of their start times are delayed, the entire project
will be delayed. Management will want to pay close attention to these critical path projects to be sure
they start on time and are completed within the allotted amount of time. Finally, the ES(TRAIN) value
of 35 tells us the estimated time to the start of the roll out of the new Solar Widget will be 45
weeks35 weeks to get to the start of training, plus 10 weeks to complete training.
The Formulation
The first set of interest in this problem is the set of eight analysts. This is a primitive set that can be
written simply as:
ANALYSTS / 1..8/;
The final set we want to construct is a set consisting of all the potential pairings. This will be a derived
set that we will build by taking the cross of the ANALYSTS set on itself. As a first pass, we could build
the dense derived set:
PAIRS(ANALYSTS, ANALYSTS);
This set, however, would include both PAIRS(I, J) and PAIRS(J, I). Since only one of these pairs is
required, the second is wasteful. Furthermore, this set will include “pairs” of the same analyst of the
form PAIRS(I, I). As much as each analyst might like an office of his or her own, such a solution is not
feasible. The solution is to put a membership filter on our derived set requiring each pair (I,J) in the
final set to obey the condition J be greater than I. We do this with the set definition:
PAIRS(ANALYSTS, ANALYSTS)|&2 #GT# &1;
USING SETS 63
The start of the membership filter is denoted with the vertical bar character (|). The &1 and &2
symbols in the filter are known as set index placeholders. Set index placeholders are valid only in
membership filters. When LINGO constructs the PAIRS set, it generates all combinations in the cross
of the ANALYSTS set on itself. Each combination is then “plugged” into the membership filter to see if
it passes the test. Specifically, for each pair (I,J) in the cross of the ANALYSTS set on itself, I is
substituted into the placeholder &1 and J into &2 and the filter is evaluated. If the filter evaluates to
true, (I,J) is added to the PAIRS set. Viewed in tabular form, this leaves us with just the above diagonal
elements of the (I,J) pairing table.
We will also be concerned with two attributes of the PAIRS set. First, we will need an attribute that
corresponds to the incompatibility rating of the pairings. Second, we will need an attribute to indicate
if analyst I is paired with analyst J. We will call these attributes RATING and MATCH. We append
them to the PAIRS set definition as follows:
PAIRS(ANALYSTS, ANALYSTS)|&2 #GT# &1:
RATING, MATCH;
We initialize the RATING attribute to the incompatibility ratings listed in the table above using the data
section:
DATA:
RATING =
9 3 4 2 1 5 6
1 7 3 5 2 1
4 4 2 9 2
1 5 5 2
8 7 6
2 3
4;
ENDDATA
We will use the convention of letting MATCH(I, J) be 1 if we pair analyst I with analyst J, otherwise 0.
Given this, the MATCH attribute contains the decision variables for the model.
Our objective is to minimize the sum of the incompatibility ratings of all the final pairings. This is just
the inner product on the RATING and MATCH attributes and is written as:
MIN = @SUM(PAIRS(I, J):
RATING(I, J) * MATCH(I, J));
There is just one class of constraints in the model. In words, it is:
For each analyst, ensure that the analyst is paired with exactly one other analyst.
Putting the constraint into LINGO syntax, we get:
@FOR(ANALYSTS(I):
@SUM(PAIRS(J, K)|J #EQ# I #OR# K #EQ# I:
MATCH(J, K)) = 1
);
The feature of interest in this constraint is the conditional qualifier (J #EQ# I #OR# K #EQ# I) on the
@SUM function. For each analyst I, we sum up all the MATCH variables that contain I and set them
equal to 1. In so doing, we guarantee analyst I will be paired up with exactly one other analyst. The
conditional qualifier guarantees we only sum up the MATCH variables that include I in its pairing.
64 CHAPTER 2
One other feature is required in this model. We are letting MATCH(I, J) be 1 if we are pairing I with J.
Otherwise, it will be 0. Unless specified otherwise, LINGO variables can assume any value from 0 to
infinity. Because we want MATCH to be restricted to being only 0 or 1, we need to apply the @BIN
variable domain function to the MATCH attribute. Variable domain functions are used to restrict the
values a variable can assume. Unlike constraints, variable domain functions do not add equations to a
model. The @BIN function restricts a variable to being binary (i.e., 0 or 1). When you have a model
that contains binary variables, it is said to be an integer programming (IP) model. IP models are much
more difficult to solve than models that contain only continuous variables. Carelessly formulated large
IPs (with several hundred integer variables or more) can literally take forever to solve! Thus, you
should limit the use of binary variables whenever possible. To apply @BIN to all the variables in the
MATCH attribute, add the @FOR expression:
@FOR(PAIRS(I, J): @BIN(MATCH(I, J)));
The Solution
The entire formulation for our matching example and parts of its solution appear below:
SETS:
ANALYSTS / 1..8/;
PAIRS(ANALYSTS, ANALYSTS)|&2 #GT# &1:
RATING, MATCH;
ENDSETS
DATA:
RATING =
9 3 4 2 1 5 6
1 7 3 5 2 1
4 4 2 9 2
1 5 5 2
8 7 6
2 3
4;
ENDDATA
MIN = @SUM(PAIRS(I, J):
RATING(I, J) * MATCH(I, J));
@FOR(ANALYSTS(I):
@SUM(PAIRS(J, K)|J #EQ# I #OR# K #EQ# I:
MATCH(J, K)) = 1
);
@FOR(PAIRS(I, J): @BIN(MATCH(I, J)));
Model: MATCHD
USING SETS 65
Global optimal solution found.
Objective value: 6.000000
Extended solver steps: 0
Total solver iterations: 0
Variable Value Reduced Cost
MATCH(1, 2) 0.0000000 9.000000
MATCH(1, 3) 0.0000000 3.000000
MATCH(1, 4) 0.0000000 4.000000
MATCH(1, 5) 0.0000000 2.000000
MATCH(1, 6) 1.000000 1.000000
MATCH(1, 7) 0.0000000 5.000000
MATCH(1, 8) 0.0000000 6.000000
MATCH(2, 3) 0.0000000 1.000000
MATCH(2, 4) 0.0000000 7.000000
MATCH(2, 5) 0.0000000 3.000000
MATCH(2, 6) 0.0000000 5.000000
MATCH(2, 7) 1.000000 2.000000
MATCH(2, 8) 0.0000000 1.000000
MATCH(3, 8) 1.000000 2.000000
MATCH(4, 5) 1.000000 1.000000
MATCH(4, 6) 0.0000000 5.000000
MATCH(4, 7) 0.0000000 5.000000
MATCH(4, 8) 0.0000000 2.000000
MATCH(5, 6) 0.0000000 8.000000
MATCH(5, 7) 0.0000000 7.000000
MATCH(5, 8) 0.0000000 6.000000
MATCH(6, 7) 0.0000000 2.000000
MATCH(6, 8) 0.0000000 3.000000
MATCH(7, 8) 0.0000000 4.000000
Solution to MATCHD
From the objective value, we know the total sum of the incompatibility ratings for the optimal pairings
is 6. Scanning the Value column for ones, we find the optimal pairings: (1,6), (2,7), (3,8), and (4,5).
Summary
In this chapter, we’ve discussed the concept of sets, how to declare sets, and demonstrated the power
and flexibility of set based modeling. You should now have a foundation of knowledge in the
definition and use of both primitive and derived sets. The next chapter will discuss the use of variable
domain functions, which were briefly introduced in this chapter when we used @BIN in the previous
matching model.
3 Using Variable Domain
Functions
Unless specified otherwise, variables in a LINGO model default to being nonnegative and continuous.
More specifically, variables can assume any real value from zero to positive infinity. In many cases,
this default domain for a variable may be inappropriate. For instance, you may want a variable to
assume negative values, or you might want a variable restricted to purely integer values. LINGO
provides four variable domain functions, which allow you to override the default domain of a variable.
The names of these functions and a brief description of their usage are:
@GIN restricts a variable to being an integer value,
@BIN makes a variable binary (i.e., 0 or 1),
@FREE allows a variable to assume any real value, positive or negative, and
@BND limits a variable to fall within a finite range.
In the remainder of this chapter, we’ll investigate the mechanics of using these functions, and present a
number of examples illustrating their usage.
Integer Variables
LINGO gives the user the ability to define two types of integer variablesgeneral and binary. A
general integer variable is required to be a whole number. A binary integer variable is further required
to be either zero or one. Any model containing one or more integer variables is referred to as an
integer programming (IP) model.
In many modeling projects, you will be faced with Yes/No types of decisions. Some examples would
include Produce/Don’t Produce, Open Plant/Close Plant, Supply Customer I from Plant J/Don’t
Supply Customer I from Plant J, and Incur a Fixed Cost/Don’t Incur a Fixed Cost. Binary variables are
the standard method used for modeling these Yes/No decisions.
General integer variables are useful where rounding of fractional solutions is problematic. For
instance, suppose you have a model that dictates producing 5,121,787.5 blue crayons in your crayon
factory. Whether you round the solution to 5,121,787 or 5,121,788 is inconsequential. On the other
hand, suppose your planning model for NASA determines the optimal number of space stations to
deploy is 1.5. Because building 0.5 space stations is impossible, you must very carefully consider how
to round the results. When whole numbers are required and rounding can make a significant
difference, general integer variables are appropriate.
67
68 CHAPTER 3
LINGO does not simply round or truncate values to come up with an integer answer. Rounding of a
solution will typically lead to either infeasible or suboptimal solutions. To illustrate this point, consider
the small model:
MAX = X;
X + Y = 25.5;
X <= Y;
By examining this model, one can deduce the optimal solution is X=Y=12.75. Now, suppose we want
an optimal solution with X being integer. Simply rounding X to 13 would make the model infeasible,
because there would be no value for Y that would satisfy both the constraints. Clearly, the optimal
solution is X=12 and Y=13.5. Unfortunately, “eyeballing” the optimal solution on larger models with
many integer variables is virtually impossible.
To solve these problems, LINGO performs a complex algorithm called branch-and-bound that
implicitly enumerates all combinations of the integer variables to determine the best feasible answer to
an IP model. Because of the extra computation time required by this algorithm, formulating your
problem to avoid the use of integer variables is advised whenever possible. Even so, although
computation times may grow dramatically when you add integer variables, it often makes sense to ask
LINGO for integer solutions when fractional values are of little or no use.
Note the new optimal number of Turbo computers, 28.5, is no longer an integer quantity. CompuQuick
must produce whole numbers of computers each day. To guarantee this, we add @GIN statements to
make both the STANDARD and TURBO variables general integer. The revised model follows:
! Here is the total profit objective function;
MAX = 100 * STANDARD + 150 * TURBO;
! Constraints on the production line capacity;
STANDARD <= 103;
TURBO <= 120;
! Our labor supply is limited;
STANDARD + 2 * TURBO <= 160;
! Integer values only;
@GIN(STANDARD); @GIN(TURBO);
70 CHAPTER 3
Solving the modified model results in the integer solution we were hoping for:
Global optimal solution found at step: 4
Objective value: 14550.00
Branch count: 1
Note that we now have a new statistic called Extended solver steps. For models with integer variables,
such as this one, the extended solver steps statistic is a tally of the number of times integer variables
had to be forced to an integer value during the branch-and-bound solution procedure. In general, this
value is not of much practical use to the normal user, other than to give you a notion of how hard
LINGO is working at finding an integer solution. If the number of steps gets quite large, LINGO is
having a hard time finding good integer solutions to your model.
DATA:
DAYS = MON TUE WED THU FRI SAT SUN;
REQUIRED = 20 12 18 16 19 14 12;
ENDDATA
In this particular model, we can always round the solution up and remain feasible. (In most models, we
won’t tend to be as lucky. Rounding the continuous solution in one direction or the other can lead to an
infeasible solution.) There may be some extra staff on some of the days, but, by rounding up, we will
never have a day without enough staff. Rounding the continuous solution up gives an objective of
10+2+2+6+5=25 employees.
Now, let’s apply integer programming to the revised staffing model. First, we will need to use the
@GIN function to make the START variables general integers. We could do this by adding the
following to our model:
@GIN(@START(MON));
@GIN(@START(TUE));
@GIN(@START(WED));
@GIN(@START(THU));
@GIN(@START(FRI));
@GIN(@START(SAT));
@GIN(@START(SUN));
However, an easier approach would be to embed the @GIN function in an @FOR function, so we can
apply @GIN to each member of START using the single statement:
@FOR(DAYS(I): @GIN(START(I)));
This new statement says, for each day of the week, make the variable corresponding to the number of
people to start on that day a general integer variable.
72 CHAPTER 3
After inserting this @FOR statement at the end of our model and reoptimizing, we get the pure integer
solution:
Global optimal solution found.
Objective value: 24.00000
Extended solver steps: 0
Total solver iterations: 6
Note that the objective of 24 beats the objective of 25 obtained by rounding. Thus, had we gone with
the rounded solution, we would have hired one more employee than required.
The Problem
As an example, suppose you are planning a picnic. You’ve constructed a list of items you would like to
carry with you on the picnic. Each item has a weight associated with it and your knapsack is limited to
carrying no more than 15 pounds. You have also come up with a 1 to 10 rating for each item, which
indicates how strongly you want to include the particular item in the knapsack for the picnic. This
information is listed below:
Item Weight Rating
Ant Repellent 1 2
Beer 3 9
Blanket 4 3
Bratwurst 3 8
Brownies 3 10
Frisbee 1 6
Salad 5 4
Watermelon 10 10
The Formulation
We have only one set in this model—the set of items we are considering carrying in the knapsack. This
is a primitive set, and we can create it using the sets section:
SETS:
ITEMS / ANT_REPEL, BEER, BLANKET,
BRATWURST, BROWNIES, FRISBEE, SALAD,
WATERMELON/:
INCLUDE, WEIGHT, RATING;
ENDSETS
We have associated the INCLUDE, WEIGHT, and RATING attributes with the set. INCLUDE will be
the binary variables used to indicate if an item is to be included in the knapsack. WEIGHT is used to
store the weight of each item, and RATING is used to store each item’s rating.
Next, we will need to construct a data section to input the weights and ratings of the items. Here is a
data section that accomplishes the task:
DATA:
WEIGHT RATING =
1 2
3 9
4 3
3 8
3 10
1 6
5 4
10 10;
KNAPSACK_CAPACITY = 15;
ENDDATA
74 CHAPTER 3
Note that we have also included the knapsack’s capacity in the data section. This is a good practice in
that it isolates data from the constraints of the model.
Given that all the sets and data have been defined, we can turn to building our objective function. We
want to maximize the sum of the ratings of the items included in our knapsack. Note that INCLUDE(I)
will be 1 if item I is included. Otherwise, it will be 0. Therefore, if we take the inner product of
INCLUDE with the RATING attribute, we will get the overall rating of a combination of included
items. Putting this into LINGO syntax, we have:
MAX = @SUM(ITEMS: RATING * INCLUDE);
Note that we did not specify a set index variable in the @SUM function. Since all the attributes in the
function (RATING and INCLUDE) are defined on the index set (ITEMS), we can drop the set index
variable and use implicit indexing.
Our next step is to input our constraints. There is only one constraint in this model. Specifically, we
must not exceed the capacity of the knapsack. In a similar manner as the objective, we compute the
weight of a given combination of items by taking the inner product of the INCLUDE attribute with the
WEIGHT attribute. This sum must be less-than-or-equal-to the capacity of the knapsack. In LINGO
syntax, we express this as:
@SUM(ITEMS: WEIGHT * INCLUDE) <= KNAPSACK_CAPACITY;
Finally, we must make the INCLUDE variable binary. We could do this by adding:
@BIN(INCLUDE(@INDEX(ANT_REPEL)));
@BIN(INCLUDE(@INDEX(BEER)));
@BIN(INCLUDE(@INDEX(BLANKET)));
@BIN(INCLUDE(@INDEX(BRATWURST)));
@BIN(INCLUDE(@INDEX(BROWNIES)));
@BIN(INCLUDE(@INDEX(FRISBEE)));
@BIN(INCLUDE(@INDEX(SALAD)));
@BIN(INCLUDE(@INDEX(WATERMELON)));
(Note that the @INDEX function simply returns the index of a primitive set member in its set. )
However, a more efficient and data independent way of doing this would be to embed an @BIN
function in an @FOR function as follows:
@FOR(ITEMS: @BIN(INCLUDE));
VARIABLE DOMAIN FUNCTIONS 75
The Solution
The entire model for our knapsack example and excerpts from its solution are listed below. The model
formulation file may be found in your SAMPLES subdirectory off the main LINGO directory under the
name KNAPSACK:
SETS:
ITEMS / ANT_REPEL, BEER, BLANKET,
BRATWURST, BROWNIES, FRISBEE, SALAD,
WATERMELON/:
INCLUDE, WEIGHT, RATING;
ENDSETS
DATA:
WEIGHT RATING =
1 2
3 9
4 3
3 8
3 10
1 6
5 4
10 10;
KNAPSACK_CAPACITY = 15;
ENDDATA
MAX = @SUM(ITEMS: RATING * INCLUDE);
@SUM(ITEMS: WEIGHT * INCLUDE) <=
KNAPSACK_CAPACITY;
@FOR(ITEMS: @BIN(INCLUDE));
Model: KNAPSACK
Global optimal solution found
Objective value: 38.00000
Extended solver steps: 0
Total solver iterations: 0
SETS:
ITEMS / ANT_REPEL, BEER, BLANKET,
BRATWURST, BROWNIES, FRISBEE, SALAD,
WATERMELON/:
INCLUDE, WEIGHT, RATING;
MUST_EAT_ONE(ITEMS)
/ SALAD, WATERMELON/;
ENDSETS
DATA:
WEIGHT RATING =
1 2
3 9
4 3
3 8
3 10
1 6
5 4
10 10;
KNAPSACK_CAPACITY = 15;
ENDDATA
MAX = @SUM(ITEMS: RATING * INCLUDE);
@SUM(ITEMS: WEIGHT * INCLUDE) <=
KNAPSACK_CAPACITY;
@FOR(ITEMS: @BIN(INCLUDE));
@SUM(MUST_EAT_ONE(I): INCLUDE(I)) >= 1;
We have derived a set called MUST_EAT_ONE from the original picnic items, and used an explicit list
to include the items we must carry as members. Then, at the end of the model, we added a constraint
that forces at least one of the “must eat” items into the solution.
VARIABLE DOMAIN FUNCTIONS 77
For those interested, the solution to the modified model is:
Global optimal solution found.
Objective value: 37.00000
Extended solver steps: 0
Total solver iterations: 0
In short, we drop the ant repellent and blanket, and replace them with the salad.
The Problem
You’re the manager of an airplane plant and you want to determine the best product-mix of your six
models to produce. The six models currently under production are the Rocket, Meteor, Streak, Comet,
Jet, and Biplane. Each plane has a known profit contribution. There is also a fixed cost associated with
the production of any plane in a period. The profit and fixed costs are given in the following table:
Plane Profit Setup
Rocket 30 35
Meteor 45 20
Streak 24 60
Comet 26 70
Jet 24 75
Biplane 30 30
78 CHAPTER 3
Each plane is produced using six raw materials—steel, copper, plastic, rubber, glass, and paint. The
units of these raw materials required by the planes as well as the total availability of the raw materials
are:
Rocket Meteor Streak Comet Jet Biplane Available
Steel 1 4 0 4 2 1 800
Copper 4 5 3 0 1 0 1160
Plastic 0 3 8 0 1 0 1780
Rubber 2 0 1 2 1 5 1050
Glass 2 4 2 2 2 4 1360
Paint 1 4 1 4 3 4 1240
The problem is to determine the final mix of products that maximizes net profit (gross profit - setup
costs) without exceeding the availability of any raw material. Your brand new Meteor model has the
highest profit per unit of anything you’ve ever manufactured and the lowest setup cost. Maybe you
should build nothing but Meteors? Then again, maybe not.
The Formulation
As you might guess, we will need two primitive sets in this model—one to represent the airplane
models and one to represent the raw materials. We can construct these sets as follows:
PLANES:
PROFIT, SETUP, QUANTITY, BUILD;
RESOURCES: AVAILABLE;
We added the following four attributes to the PLANES set:
The Solution
The formulation in its entirety and a selected portion of the solution appear below. The formulation file
may be found in file PRODMIX.
MODEL:
SETS:
PLANES:
PROFIT, SETUP, QUANTITY, BUILD;
RESOURCES: AVAILABLE;
RXP( RESOURCES, PLANES): USAGE;
ENDSETS
DATA:
PLANES PROFIT SETUP =
ROCKET 30 35
METEOR 45 20
STREAK 24 60
COMET 26 70
JET 24 75
BIPLANE 30 30;
RESOURCES AVAILABLE =
STEEL,800 COPPER,1160 PLASTIC,1780
RUBBER,1050 GLASS,1360 PAINT,1240;
USAGE = 1 4 0 4 2 0
4 5 3 0 1 0
0 3 8 0 1 0
2 0 1 2 1 5
2 4 2 2 2 4
1 4 1 4 3 4;
ENDDATA
END
Model: PRODMIX
82 CHAPTER 3
Global optimal solution found.
Objective value: 14764.00
Extended solver steps: 11
Total solver iterations: 189
Summary
You should now be familiar with the use of the variable domain functions @BIN and @GIN, and how
they are used to introduce integer variables into a model. This section has shown how integer variables
bring a whole new dimension of power to the mathematical modeler. Given that we have only briefly
delved into the topic of modeling with integer variables, the user that would like to become more
familiar with the many practical applications of integer programming and their formulations can refer
to Schrage (2006), or Winston (1995).
VARIABLE DOMAIN FUNCTIONS 83
Free Variables
By default, a LINGO variable has a lower bound of zero and an upper bound of infinity. @FREE
removes the lower bound of zero and lets a variable take negative values, rendering it unconstrained in
sign, or free. The syntax is:
@FREE(variable_name);
where variable_name is the name of the variable you wish to make free.
The @FREE function may be used in a model anywhere you would normally enter a constraint. The
@FREE function can be embedded in an @FOR statement to allow you to easily make all, or selected,
variables of an attribute to be free. Some examples of @FREE are:
Example 1: @FREE(X);
makes the scalar variable, X, free,
Example 2: @FREE(QUANTITY(4));
makes the variable QUANTITY(4) free,
Example 3: @FOR(ITEMS: @FREE(QUANTITY));
makes all variables in the QUANTITY attribute free.
The Formulation
We will need two primitive sets in our model. The first set will have eight members to represent the
quarters that we have historical data for. The second set will have four members corresponding to the
four quarters of the year. This second set is used for defining the four seasonal factors. Here is our sets
section that incorporates these two sets:
SETS:
PERIODS: OBSERVED, PREDICT, ERROR;
QUARTERS: SEASFAC;
ENDSETS
The three attributes on the PERIODS set—OBSERVED, PREDICT, and ERROR—correspond to the
observed sales values, predicted sales values, and the prediction error. The prediction error is simply
predicted sales minus observed sales. The SEASFAC attribute on the SEASONS set corresponds to the
seasonal sales factors and will be computed by LINGO.
We will also need to add a data section to initialize the set members and the OBSERVED attribute with
the historical sales data. We can do this with the following:
DATA:
PERIODS = P1..P8;
QUARTERS = Q1..Q4;
OBSERVED = 10 14 12 19 14 21 19 26;
ENDDATA
Next, we must add a formula to compute the error terms. As mentioned, the error term in a period is
the difference between the observed and predicted sales. We can express this in LINGO as:
@FOR(PERIODS: ERROR =
PREDICT - OBSERVED);
Our objective is to minimize the sum of the squared error terms, which may be written as:
MIN = @SUM(PERIODS: ERROR ^ 2);
VARIABLE DOMAIN FUNCTIONS 85
We choose to use squared error terms as a measure to minimize because we want to weight large errors
relatively more heavily. Another option might be to minimize the sum of the absolute values of the
errors, which would weight large and small errors proportionally the same.
In order to compute the error terms, we will also need to compute predicted sales. Using our
theoretical formula, we compute predicted sales as follows:
@FOR(PERIODS(P): PREDICT(P) =
SEASFAC(@WRAP(P, 4))
* (BASE + P * TREND));
The @WRAP function is used here to allow us to apply the four seasonal factors over a time horizon
exceeding four periods. Had we simply used the index P, instead of @WRAP(P, 4), we would have
generated a subscript out of range error. For a more in depth explanation of the use of the @WRAP
function, please see the staff-scheduling example on page 49.
For esthetic reasons, we would like the seasonal factors to average out to a value of one. We can do
this by adding the constraint:
@SUM(QUARTERS: SEASFAC) = 4;
Finally, it is possible for the error terms to be negative as well as positive. Given that variables in
LINGO default to a lower bound of zero, we will need to use the @FREE function to allow the error
terms to go negative. By embedding the @FREE function in an @FOR loop, we can apply @FREE to
all the ERROR variables in the statement:
@FOR(PERIODS: @FREE(ERROR));
86 CHAPTER 3
The Solution
The entire formulation and excerpts from the solution appear below.
MODEL:
SETS:
PERIODS: OBSERVED, PREDICT, ERROR;
QUARTERS: SEASFAC;
ENDSETS
DATA:
PERIODS = P1..P8;
QUARTERS = Q1..Q4;
OBSERVED = 10 14 12 19 14 21 19 26;
ENDDATA
END
Model: SHADES
VARIABLE DOMAIN FUNCTIONS 87
Local optimal solution found.
Objective value: 1.822561
Total solver iterations: 32
Variable Value
BASE 9.718878
TREND 1.553017
OBSERVED( P1) 10.00000
OBSERVED( P2) 14.00000
OBSERVED( P3) 12.00000
OBSERVED( P4) 19.00000
OBSERVED( P5) 14.00000
OBSERVED( P6) 21.00000
OBSERVED( P7) 19.00000
OBSERVED( P8) 26.00000
PREDICT( P1) 9.311820
PREDICT( P2) 14.10136
PREDICT( P3) 12.85213
PREDICT( P4) 18.80620
PREDICT( P5) 14.44367
PREDICT( P6) 20.93171
PREDICT( P7) 18.40496
PREDICT( P8) 26.13943
ERROR( P1) -0.6881796
ERROR( P2) 0.1013638
ERROR( P3) 0.8521268
ERROR( P4) -0.1938024
ERROR( P5) 0.4436688
ERROR( P6) -0.6828722E-01
ERROR( P7) -0.5950374
ERROR( P8) 0.1394325
SEASFAC( Q1) 0.8261096
SEASFAC( Q2) 1.099529
SEASFAC( Q3) 0.8938789
SEASFAC( Q4) 1.180482
Solution to SHADES
The solution is: TREND, 1.55; BASE, 9.72. The four seasonal factors are .826, 1.01, .894, and 1.18.
The spring quarter seasonal factor is .826. In other words, spring sales are 82.6% of the average. The
trend of 1.55 means, after the effects of season are taken into account, sales are increasing at an
average rate of 1,550 sunglasses per quarter. As one would expect, a good portion of the error terms
are negative, so it was crucial to use the @FREE function to remove the default lower bound of zero
on ERROR.
88 CHAPTER 3
Our computed function offers a very good fit to the historical data as the following graph illustrates:
Using this function, we can compute the forecast for sales for the upcoming quarter (quarter 9). Doing
so gives:
Predicted_Sales(9) = Seasonal_Factor(1) * (Base + Trend * 9)
= 0.826 * (9.72 + 1.55 * 9)
= 19.55
Given this, inventory levels should be brought to a level sufficient to support an anticipated sales level
of around 19,550 pairs of sunglasses.
Bounded Variables
Whereas @FREE sets the upper and lower bounds of the specified variable to plus and minus infinity
(effectively removing any bounds on the variable), the @BND function lets you set specific upper and
lower bounds on a variable. In other words, @BND limits a variable’s range within some specified
interval. The syntax for @BND is:
@BND(lower_bound, variable_name, upper_bound);
where variable_name is the variable to be bounded below by the quantity lower_bound and bounded
above by the quantity upper_bound. Both lower_bound and upper_bound must be either numeric
values or variables whose values have been set in a data section. @BND may be used wherever you
would normally enter a constraint in a model—including inside an @FOR looping function.
In mathematical terms, LINGO interprets this @BND function as:
lower_bound ≤ variable_name ≤ upper_bound
VARIABLE DOMAIN FUNCTIONS 89
It is certainly possible to add constraints in lieu of the @BND function, but, from the standpoint of the
optimizer, @BND is an extremely efficient way of representing simple bounds on variables.
Specifying variable bounds using @BND rather than explicitly adding constraints can noticeably speed
up the solution times for larger models. Furthermore, @BND does not count against the limit on the
total number of constraints LINGO imposes on some versions. So, in general, it is a good idea to use
@BND in place of constraints whenever possible.
Some examples of @BND are:
Example 1: @BND(-1, X, 1);
constrains the variable X to lie in the interval [-1,1],
Example 2: @BND(100, QUANTITY(4), 200);
constrains the variable QUANTITY(4) to fall within 100 to 200,
Example 3: @FOR(ITEMS: @BND(10, Q, 20));
sets the bounds on all variables in the Q attribute to 10 and 20,
Example 4: @FOR(ITEMS: @BND(QL, Q, QU));
sets the bounds on all variables in the Q attribute to QL and QU (QL and
QU must have been previously set to some values in a data section).
4 Data, Init and Calc
Sections
Typically, when dealing with a model’s data, you need to assign set members to sets and give values to
some set attributes before LINGO can solve your model. For this purpose, LINGO gives the user three
optional sections, the data section for inputting set members and data values, the init section for setting
the starting values for decision variables, and the calc section for performing computations on raw
input data.
Basic Syntax
The data section begins with the keyword DATA: (including the colon) and ends with the keyword
ENDDATA. In the data section, you can have statements to initialize set members and/or the attributes
of the sets you instantiated in a previous sets section. These expressions have the syntax:
object_list = value_list;
The object_list contains the names of the attributes and/or a set whose members you want to initialize,
optionally separated by commas. There can be no more than one set name in object_list, while there
may be any number of attributes. If there is more than one attribute name in object_list, then the
attributes must be defined on the same set. If there is a set name in object_list, then all attributes in
object_list must be defined on this set.
The value_list contains the values you want to assign to the members of object_list, optionally
separated by commas. As an example, consider the following model:
SETS:
SET1 /A, B, C/: X, Y;
ENDSETS
DATA:
X = 1, 2, 3;
Y = 4, 5, 6;
ENDDATA
91
92 CHAPTER 4
We have two attributes, X and Y, defined on the SET1 set. The three values of X are set to 1, 2, and 3,
while Y is set to 4, 5, and 6. We could have also used the following compound data statement to the
same end:
SETS:
SET1 /A, B, C/: X, Y;
ENDSETS
DATA:
X, Y = 1, 4,
2, 5,
3, 6;
ENDDATA
Looking at this example, you might imagine X would be assigned the values 1, 4, and 2 because they
are first in the values list, rather than the true values of 1, 2, and 3. When LINGO reads a data
statement's value list, it assigns the first n values to the first position of each of the n attributes in the
attribute list, the second n values to the second position of each of the n attributes, and so on. In other
words, LINGO is expecting the input data in column form rather than row form.
As mentioned, we can also initialize the set members in the data section. Modifying our sample model
to use this approach by moving the set members from the sets section to the data section, we get:
SETS:
SET1: X, Y;
ENDSETS
DATA:
SET1, X, Y = A 1 4
B 2 5
C 3 6;
ENDDATA
This final method is, perhaps, the most elegant in that all model data—attribute values and set
members—are isolated within the data section.
Parameters
You are not limited to putting attributes and sets on the left-hand side of data statements. You may also
initialize scalar variables in the data section. When a scalar variable's value is fixed in a data section,
we refer to it as a parameter.
As an example, suppose your model uses an interest rate of 8.5% as a parameter. You could input the
interest rate as a parameter in the data section as follows:
DATA:
INTEREST_RATE = .085;
ENDDATA
As with set attributes, you can initialize multiple parameters in a single statement. Suppose you also
add the inflation rate to your model. You could initialize both the interest rate and inflation rate in the
same data statement as follows:
DATA:
INTEREST_RATE, INFLATION_RATE = .085, .03;
ENDDATA
DATA, INIT AND CALC SECTIONS 93
What If Analysis
In some cases, you may not be sure what values to input for the data in your model. For example,
suppose your model uses the inflation rate as a parameter. You may be uncertain as to the exact rate of
inflation in the future, but you know it will most likely fall within a range of 2 to 6 percent. What you
would like to do is run your model for various values of the inflation rate within this range to see how
sensitive the model's results are to inflation. We refer to this as what if analysis, and LINGO has a
feature to facilitate this. To set up a parameter for what if analysis, input a question mark (?) as its
value in place of a number as in the following example:
DATA:
INFLATION_RATE = ?;
ENDDATA
LINGO will prompt you for a value for the INFLATION_RATE parameter each time you solve the
model. Under Windows, you will receive a dialog box resembling:
Simply input the desired value for the inflation rate and then press the OK button. LINGO will then set
INFLATION_RATE to the value you input and proceed with solving the model.
On platforms other than Windows, LINGO will write a prompt to your screen asking you to input a
value for INFLATION_RATE. Type in the value and then press the Enter key.
In addition to parameters, you can perform what if analysis on individual members of attributes by
initializing them to question marks in the data section, as well.
For an example of a model that uses what if analysis to compute the value of a home mortgage, see the
Home Mortgage Calculation model in Appendix A, Additional Examples of LINGO Modeling.
DATA:
NEEDS = 20;
ENDDATA
LINGO will initialize all the members of the NEEDS attribute to the value 20.
If there are multiple attributes on the left-hand side of the data statement, you will need one value on
the right-hand side for each attribute on the left. For instance, let's extend the previous example, so we
have an additional attribute called COST:
SETS:
DAYS / MO, TU, WE, TH, FR, SA, SU/:
NEEDS, COST;
ENDSETS
DATA:
NEEDS, COST = 20, 100;
ENDDATA
All seven members of NEEDS will be initialized to 20 and all seven members of COST to 100.
We have set CAPACITY for the first two years to 34, but have omitted values for the last three years.
LINGO will assume, therefore, that it is free to determine the values for CAPACITY in the last three
years.
Note: You must use commas when omitting values. If you do not use the commas, LINGO will
think you did not enter the correct number of values for the attribute, which will trigger an
error message.
DATA, INIT AND CALC SECTIONS 95
Note: Starting points specified in an init section are only of use in nonlinear models. Starting points
currently offer no help in purely linear models. If you are not sure whether your model is
linear or nonlinear, you can check the count of nonlinear constraints in the solver status
window. If there are any nonlinear constraints, then your model is nonlinear. For more
information on the nature of nonlinear models and how good starting points can be of
assistance, please see Chapter 14, On Mathematical Modeling.
As an example, in a set defining a group of stocks, you may have a known price of each stock, but the
amount to buy or sell of each stock is unknown. You would typically initialize the price attribute in the
data section. If approximate values of the buy and sell attributes are known, you can tell LINGO this
information by entering it in the init section. LINGO then uses the values specified as a starting point
in its search for the optimal solution. If your starting point is relatively close to an optimal solution,
you may save on the solution time required to run your model.
An init section begins with the keyword INIT: and ends with the keyword ENDINIT. The syntax rules
for init statements in the init section are identical to the rules for data section statements. You can have
multiple attributes on the left-hand side of a statement, you can initialize an entire attribute to a single
value, you can omit values in an attribute, and you can use the question mark to have LINGO prompt
you for an initialization value whenever you solve the model.
As an example of how a good starting point may help to reduce solution times, consider the small
model:
Y <= @LOG(X);
X^2 + Y^2 <=1;
The function @LOG(X) returns the natural logarithm of X. This model has only one feasible point of
(X,Y) = (1,0). If we solve this model without an init section, we get the solution:
Feasible solution found at step: 12
Variable Value
Y 0.5721349E-03
X 1.000419
Note that it required 12 iterations to solve. Now, let’s add an init section to initialize X and Y to a point
close to the solution, so we have:
INIT:
X = .999;
Y = .002;
ENDINIT
Y <= @LOG(X);
X^2 + Y^2 <=1;
96 CHAPTER 4
Solving this modified model, we get the solution:
Feasible solution found at step: 3
Variable Value
X 0.9999995
Y 0.0000000
Note that our solution required only 3 iterations compared to the 12 iterations required without the init
section.
DATA:
X, Y, Z = 1, 2, 3;
ENDDATA
CALC:
AVG = ( X + Y + Z) / 3;
ENDCALC
END
Example of a valid calc section
DATA, INIT AND CALC SECTIONS 97
Now, suppose we did not know the value of Y beforehand. The following model with Y dropped from
the data section would trigger an error in LINGO. The error occurs because the value of Y is an
unknown, which violates the requirement that all right-hand side variables in a calc expression must
have already had their values established in a previous data or calc section:
MODEL:
DATA:
X, Z = 1, 3;
ENDDATA
CALC:
AVG = ( X + Y + Z) / 3;
ENDCALC
END
Example of an invalid calc section
You may perform running calculations in a calc section, which means that you may break complex
calc expressions down into a series of smaller expressions. Here we break the computation from
above into two steps:
MODEL:
DATA:
X, Y, Z = 1, 2, 3;
ENDDATA
CALC:
AVG = X + Y + Z;
AVG = AVG / 3;
ENDCALC
END
Example of a running calc expression
98 CHAPTER 4
There is no limit to the number of times that a variable may appear on the left-hand side of a calc
expression. However, the final calc expression for the variable will determine its value in the final
solution report.
Calc expressions are computed sequentially in the order in which they appear in the model. So, if one
calc expression feeds its value into a subsequent expression, then it must appear before its dependent
expression. For example, the following calc section is valid:
CALC:
X = 1;
Y = X + 1;
ENDCALC
while this variation is not valid:
CALC:
Y = X + 1;
X = 1;
ENDCALC
In the second example, Y depends on X, but X is not defined until after Y.
Of course, Set looping functions may also be used in calc expressions. For example, consider the
following portfolio optimization model. In this model, we take the annual returns for three stocks and
in a calc section compute the following three pieces of information for the stocks: average return, the
covariance matrix, and the correlation matrix. This information is then used in a standard Markowitz
model to determine an optimal portfolio that meets a desired level of return while minimizing overall
risk.
MODEL:
SETS:
STOCKS: AVG_RET, WEIGHT;
DAYS;
SXD( DAYS, STOCKS): RETURN;
SXS( STOCKS, STOCKS): COVR, CORR;
ENDSETS
DATA:
DAYS = 1..12;
TARGET = .15;
STOCKS = ATT GMC USX;
RETURN = 0.300 0.225 0.149
0.103 0.290 0.260
0.216 0.216 0.419
-0.046 -0.272 -0.078
-0.071 0.144 0.169
0.056 0.107 -0.035
0.038 0.321 0.133
0.089 0.305 0.732
0.090 0.195 0.021
0.083 0.390 0.131
0.035 -0.072 0.006
DATA, INIT AND CALC SECTIONS 99
0.176 0.715 0.908;
ENDDATA
CALC:
!Average annual return for each stock;
@FOR( STOCKS( S):
AVG_RET( S) =
( @SUM( SXD( D, S): RETURN( D, S)) /
@SIZE( DAYS))
);
!Covariance matrix;
@FOR( SXS( S1, S2):
COVR( S1, S2) =
@SUM( DAYS( D):( RETURN( D, S1) - AVG_RET( S1)) *
( RETURN( D, S2) - AVG_RET( S2))) / @SIZE( DAYS)
);
!Although not required, compute the correlation matrix;
@FOR( SXS( S1, S2):
CORR( S1, S2) = COVR( S1, S2) /
( COVR( S1, S1) * COVR( S2, S2))^.5;
);
ENDCALC
!Minimize the risk of the portfolio
(i.e., its variance);
[R_OBJ] MIN = @SUM( SXS( S1, S2):
WEIGHT( S1) * WEIGHT( S2) * COVR( S1, S2));
!Must be fully invested;
[R_BUDGET] @SUM( STOCKS: WEIGHT) = 1;
!Must exceed target return;
[R_TARGET] @SUM( STOCKS: AVG_RET * WEIGHT) >= TARGET;
END
Model: MARKOW
Summary
You should now be comfortable with adding basic data, init and calc sections to your models. Keep in
mind that initialization performed in a data section permanently fixes a variable's value. Initialization
done in an init section is used only as a temporary starting point, which may be of benefit in finding
solutions to nonlinear models. Initialization in a calc section holds until another calc expression
redefining a variable’s value is encountered. The benefit of placing computations in a calc section as
opposed to placing them in the general constraint section is that calc expressions are treated as side
computations and aren’t fed into the main solver, thereby improving execution times.
We have only touched on some of the basic features of data and init sections in this chapter. In
subsequent sections, you will see how to add hooks in your data and init sections to external files,
spreadsheets, and databases.
5 Windows Commands
In this chapter, we will discuss the pull down menu commands available in the Windows version of
LINGO. The following chapter, Command-line Commands, deals with the commands available
through LINGO’s command-line interface. If you’re not using a Windows version of LINGO, then you
will be primarily interested in the following chapter. If you are using a Windows version of LINGO,
then you will be primarily interested in this chapter. Windows users will also be interested in the
command-line interface if they plan to build command scripts to automate LINGO.
Menus
LINGO groups commands under the following five menus:
♦ File
♦ Edit
♦ LINGO
♦ Window
♦ Help
The File menu contains commands that primarily deal with handling input and output. The Edit menu
contains commands for editing the document in the current window. The LINGO menu contains
commands to solve a model and generate solution reports. The Window menu has commands that deal
with the mechanics of handling multiple windows. The Help menu provides access to LINGO’s help
facility.
The Toolbar
By default, the toolbar runs along the top of the screen and is illustrated in the following picture:
LINGO’s toolbar “floats”. Thus, you can reposition it by dragging it to any part of the screen. You can
also choose to suppress the toolbar by clearing the Toolbar button on the Interface tab of the
LINGO|Options dialog box.
Each button on the toolbar corresponds to a menu command. Not all menu commands have a toolbar
button, but, in general, the most frequently used commands have an equivalent button.
101
102 CHAPTER 5
LINGO displays “tool tips” for each button. When you position the mouse over a button, a short
description of what the button does appears in a pop up window and in the status bar at the bottom of
the screen.
Here is a list of the buttons and their equivalent commands:
File|Save LINGO|Solution
File|Print LINGO|Options
Edit|Undo LINGO|Picture
Edit|Paste Help|Topics
Edit|Find Help|Pointer
Edit|Go To Line
Accelerator Keys
Along with accessing commands via the menus and toolbar, most commands may also be accessed by
a single, unique keystroke known as an accelerator. The equivalent accelerator key is listed alongside
each command in the menus.
1. File Menu
LINGO’s File menu, pictured at left, contains
commands that generally pertain to the movement of
files in and out of LINGO. Each command contained
in the File menu is discussed below.
106 CHAPTER 5
File|New F2
The New command opens a new, blank window. When you select the New command, you will be
presented with the following dialog box:
You may then select the type of file you want to create. The file must be one of the four types:
1. LINGO Model (*.lg4)
The LG4 format was established with release 4.0 of LINGO. LG4 is the primary
file format used by LINGO to store models under Windows. This format
supports multiple fonts, custom formatting, and OLE (Object Linking and
Embedding). LG4 files are saved to disk using a proprietary binary format.
Therefore, these files can’t be read directly into other applications or transported
to platforms other than the PC. Use the LNG format (discussed next) to port a
file to other applications or platforms.
2. LINGO Model (Text Only) (*.lng)
The LNG format is a portable format for storing your models. It was the
standard file format used by LINGO in releases prior to 4.0 and remains in use
on all platforms other than Windows. LNG files are saved to disk as ASCII text
and may be read into any application or word processor that supports text files.
LNG files may also be ported to platforms besides the PC. LNG files do not
support multiple fonts, custom formatting, or OLE.
3. LINGO Data (*.ldt)
LDT files are data files typically imported into LINGO models using the @FILE
function. @FILE can only read text files. Given this, all LDT files are stored as
ASCII text. LDT files do not support multiple fonts, custom formatting, or OLE.
4. LINGO Command Script (*.ltf)
LTF files are LINGO command scripts. These are ASCII text files containing a
series of LINGO commands that can be executed with the File|Take Commands
command. For more information on commands that can be used in a LINGO
script, refer to the following chapter, Command-line Commands. LTF files do
not support multiple fonts, custom formatting, or OLE.
WINDOWS COMMANDS 107
5. LINDO Model (*.ltx)
LTX files are model files that use the LINDO syntax. Longtime LINDO users may
prefer LINDO syntax over LINGO syntax. LINDO syntax is convenient for quickly
entering small to medium sized linear programs. As long as a file has an extension
of .ltx, LINGO will assume that the model is written using LINDO syntax. Readers
interested in the details of LINDO syntax may contact LINDO Systems to obtain a
LINDO user’s manual.
When you simply press either the New toolbar button or the F2 key, LINGO assumes you want a
model file. Thus, LINGO does not display the file type dialog box and immediately opens a model file
of type LG4.
If you have used the LINGO|Options command to change the default model file format from LG4 to
LNG, LINGO will automatically open a model of type LNG when you press either the New button or
the F2 key.
You may begin entering text directly into a new model window or paste in text from other applications
using the Windows clipboard and the Edit|Paste command in LINGO.
File|Open... Ctrl+O
The Open command reads a saved file from disk and places it in a LINGO Window. The file can be a
LINGO model file (*.LG4), or any other file. If the file is not in LG4 format, it must be in ASCII text
format.
After issuing the Open command, you will be presented with a dialog box resembling the following:
You can enter a file name in the File name edit box, or select a file name from the list of existing files
by double-clicking on a file. Press the Open button to open the file, the Cancel button to exit without
opening a file, or the Help button for assistance.
108 CHAPTER 5
You may select a different file type from the Files of type list box causing LINGO to list only the files
of that type.
If the file to be opened has an extension of .MPS, then LINGO will invoke its MPS reader to parse the
file. The MPS file format is an industry standard format developed by IBM, which is useful for passing
models from one solver or platform to another. When importing an MPS file, LINGO reads the MPS
file from disk, converts it to an equivalent LINGO model, and places the model into a new model
window. Refer to the section below, Importing MPS Files, for more information. LINGO can also
write MPS format files, which is discussed in the File|Export File section below.
If you have just read in a LINGO model file (a LG4 or LNG file) and wish to solve it, use the
LINGO|Solve command.
One thing to notice at this point is that the MPS format is not a very compact method for storing a
model.
WINDOWS COMMANDS 109
Using the File|Open command to read this file into LINGO, we are presented with the following
window containing an equivalent LINGO model:
Note how the model is automatically converted from MPS format to LINGO format. Should you wish
to save the file again using MPS format rather than LINGO format, you may use the File|Export
File|MPS Format… command discussed below.
When it comes to acceptable constraint and variable names, MPS format is less restrictive than
LINGO. To compensate for this fact, LINGO attempts to patch names when reading an MPS file, so
that all the incoming names are compatible with its syntax. LINGO does this by substituting an
underscore for any character in a name that is not admissible. In most cases, this will work out OK.
However, there is a chance for name collisions where two or more names get mapped into one. For
instance, the variable names X.1 and X%1 would both get mapped into the single LINGO name X_1.
Of course, situations such as this entirely alter the structure of the model, rendering it incorrect.
However, you will be warned whenever LINGO has to patch a name with the following error message:
This message displays the number of variable and row names that were patched to get them to conform
to LINGO syntax.
110 CHAPTER 5
If name collisions are a problem, then LINGO has an option to ensure that all names remain unique.
This option involves using RC format for names encountered during MPS I/O. RC format involves
renaming each row (constraint) in a model to be Rn, where n is the row’s index. Similarly, each
column (variable) is renamed to Cn. In addition, LINGO renames the objective row to be ROBJ. To
switch to RC format for MPS names, run the LINGO|Options command, select the General Solver tab,
then click the checkbox titled Use R/C format names for MPS I/O, as illustrated here:
As an example, we will once again import the same MPS format model as above. However, this time
we will use RC naming conventions. Here is the model as it appears after importing it into LINGO:
Notice how the variable names now use RC format, guaranteeing that name collisions will not occur.
Another potential conflict is that MPS allows variable names to be duplicated as constraint names and
vice versa. LINGO does not allow for this. When you go to solve the model, you will receive either
error message 28 (Invalid use of a row name), or error message 37 (Name already in use). Once again,
you can switch to using RC name format to avoid this conflict.
As a final note, LINGO only supports free format MPS files, and does not support fixed format MPS
files. Therefore, variable and row names may not contain embedded blanks.
File|Save Ctrl+S
The Save command saves the contents of the active window to disk using the existing file name for the
window. If the window has not been saved before, you will be prompted to provide a name for the file.
WINDOWS COMMANDS 111
File|Save As... F5
The Save As command allows you to save the contents of the active window under a new file name.
When issuing the Save As command, you will be presented with a dialog box that resembles the
following:
You can enter a new file name in the File name edit box, or select a file name from the list of existing
files by double-clicking on it. If you do not specify a file extension, LINGO will append the extension
of the default model format to the name. If you want to prevent LINGO from appending an extension
to the name, place the file name in double quotes.
Press the Save button to save the model, the Cancel button to exit without saving, or the Help button
for assistance.
You may select a different file type from the Save as type list box. If your model has special fonts or
embedded objects, you must save it using the LG4 file format to preserve them. The LG4 format is a
special binary format readable only by LINGO. If you wish to create a text copy of your model, then
use the LNG file format. For further discussion of the available file formats under LINGO, refer to the
New command above.
File|Close F6
Use the Close command to close the active (front most) window. If the window has been modified
without being saved, you’ll be asked whether you want to save the changes.
112 CHAPTER 5
File|Print... F7
Use the Print command to send the contents of the active window to your printer. First, LINGO will
display the Print dialog box:
Select the printer to route the output to from the Name list box. Modify the printer’s properties by
pressing the Properties button. Select a range of pages to print in the Print range group box. If you
need multiple copies, input the number desired in the Number of copies field and specify if you want
the copies collated (assuming your printer is capable of collating). Finally, press the OK button to
begin printing. Press the Cancel button to exit without printing.
WINDOWS COMMANDS 113
File|Print Setup... F8
Use the Print Setup command to configure your printer. You should see a dialog box that resembles
the following:
Select the target printer from the Name list box. Press the Properties button to set additional printer
properties. Select the type of paper and tray from the Paper group box. In the Orientation group box,
select whether you want portrait or landscape output. Press the Cancel button to exit without changing
the printer configuration. Press the OK button to save changes and exit the Print Setup command.
114 CHAPTER 5
The Print button sends the file to the printer. The Next Page button brings the next page into the
viewer. The Prev Page button brings the previous page into the viewer. The One Page button puts the
viewer into single page mode, while the Two Page button puts the viewer into double page mode. The
Zoom In button is used to have the viewer zoom in on a region of the document. The Zoom Out button
undoes the effect of a Zoom In. Press the Close button to close the print viewer and return to the
normal command mode of LINGO.
If you would like to change some of the printer specifications, such as landscape output, use the Print
Setup command (described above) before issuing the Print Preview command.
WINDOWS COMMANDS 115
File|Log Output... F9
Normally, when you are using LINGO for Windows, it is operating in a menu driven mode, where you
choose commands from the pull down menus and reports are displayed in individual windows. LINGO
can also operate in command mode, where text commands or command script files drive the
application and all output is routed to a window known as the command window. All input and output
passes through the command window when LINGO is in command mode. You can open a command
window at anytime by issuing the Window|Command Window command.
In general, you will only be interested in running LINGO in command mode if you are planning to
embed LINGO in a larger application. If you do use LINGO in command mode, you will find that the
command window can only hold a limited amount of output. Should you need to keep a disk-based
copy of all that transpires in the command window, you will need to use the Log Output command.
The Log Output command opens a standard Windows file dialog box from which you can name the
log file. You can echo the output to the command window as well as the file by checking the Echo to
screen checkbox. If you would like to append output to the end of an existing file, check the Append
output checkbox.
When you have selected a file for logging output, a check mark will appear in the File menu before the
Log Output command. To turn off Log Output, select the command again and the check mark will
disappear.
Select item 4, LINGO Command Script, and press the OK button. LINGO will open a blank script file.
116 CHAPTER 5
Now, enter the following into the script file:
This is a command script that inputs a small product-mix model, solves it, and puts the solution in a
text file. Save the command script to a file titled MyScript.ltf using the File|Save As command.
To run the script, issue the File|Take Commands command. You should see the following:
WINDOWS COMMANDS 117
Double-click on the icon for MyScript.ltf to begin processing the command script. LINGO’s command
window will now appear, and you should be able to watch LINGO’s progress at processing the script
by watching commands and output as they are logged in the command window. When LINGO finishes
the command script, the command window will resemble the following:
118 CHAPTER 5
Also of interest is the solution file, SOLU.LGR, created as part of our command script. If you open this
file, you should find the following solution to the model:
Variable Value Reduced Cost
X 50.00000 0.000000
Y 35.00000 0.000000
The output that was routed to the command window can be routed to a file using the Log Output
command described above.
File|Export File
The File|Export File command allows you to either export MPS or MPI format files. The MPS file
format is an industry standard format developed by IBM, and is useful for passing models from one
solver or platform to another. MPI file format was developed by LINDO Systems as a way to store all
math programs, from linear models to, in particular, nonlinear models
Note 2: When exporting an MPS file, LINGO truncates all variable names to 8 characters. For
instance, the two distinct LINGO names SHIP(WH1, C1) and SHIP(WH1, C2) would both be
truncated to the single 8 character name SHIPWH1C under MPS. Either choose names to
avoid collisions of truncated names, or enable the option for converting names to RC format
when doing MPS I/O. LINGO will display an error message if potential collisions exist.
Note 3: The MPS file format is intended primarily for exporting models to other applications or
platforms. The MPS format is purely scalar in nature—all set-based information is lost upon
converting a LINGO model to MPS format. Thus, when saving copies of a model on your
own machine, you should always use the File|Save command in order to preserve your model
in its entirety.
120 CHAPTER 5
Exporting MPI Files
MPI file format was developed by LINDO Systems as a way to store all math programs, from linear
models to, in particular, nonlinear models. As with MPS files, the MPI format is scalar-based. Thus,
you will lose any sets in your model when saving it in this format. Most users will not have a need for
MPI formatted files. However, LINDO API users can load these files directly and may find this feature
useful.
File|License
Some versions of LINGO require the user to input a password. Think of the password as a “key” that
unlocks the LINGO application. If you upgrade your copy of LINGO, then you will need to enter a
new password. The File|License command prompts you for a new password.
When you run the File|License command, you will be presented with the dialog box:
Carefully enter the password into the edit field, including hyphens, making sure that each character is
correct. Click the OK button and, assuming the password was entered correctly, LINGO will display
the Help|About LINGO dialog box listing the features in the upgraded license. Verify that these
features correspond to the license you intended to install.
Note: If you were e-mailed your password, then you have the option of cutting-and-pasting it into
the password dialog box. Cut the password from the e-mail that contains it. Then, press
Ctrl+V to paste it into the LINGO File|License dialog box.
WINDOWS COMMANDS 121
Enter any user id and/or password into the appropriate fields. For security reasons, LINGO does not
store this information from one session to the next. So, you will need to run this command at the start
of each session.
If security is not a concern, and you would like to store your database user information, then you can
create an AUTOLG.DAT file containing a DBUID command and a DBPWD command. Commands in
the AUTOLG.DAT file are executed automatically each time LINGO starts. Thus, DBUID and
DBPWD commands contained in an AUTOLG.DAT file will restore your database user information at
the start of each LINGO run. For more information on the use of AUTOLG.DAT files, refer to LINGO
Command Scripts section in Chapter 8, Interfacing with External Files.
File|Exit F10
Use the Exit command to quit LINGO. If any unsaved files are open, you will be prompted to save
them before LINGO shuts down.
122 CHAPTER 5
2. Edit Menu
LINGO’s Edit menu, pictured at left,
contains commands that generally
pertain to editing and modifying the text
within a window. Each command
contained in the Edit menu is discussed
below.
Edit|Undo Ctrl+Z
Use the Undo Command to undo the last modification made to the contents of a Window. Undo can
undo all operations except drag-and-drop. LINGO stores a limited amount of undo operations, so you
won’t be able to depend on LINGO to undo extensive changes.
Edit|Redo Ctrl+Y
This command will redo the last undo operation. LINGO stores a limited amount of redo operations, so
you won’t be able to depend on LINGO to redo extensive changes.
WINDOWS COMMANDS 123
Edit|Cut Ctrl+X
Use the Cut command to clear the selected block of text and place it on the clipboard for pasting. To
select a block of text for cutting, place the cursor immediately before the block and press down on the
left mouse button. Now, drag the mouse until the cursor appears immediately after the block of text.
The text block should now be displayed in reverse video. Now, issue the Cut command to remove the
selected text from the document, placing it in the Windows clipboard.
Edit|Copy Ctrl+C
Use the Copy command to copy the selected text to the clipboard for pasting. To select a block of text
for copying, place the cursor immediately before the block and press down on the left mouse button.
Now, drag the mouse until the cursor appears immediately after the block of text. The text block
should now be displayed in reverse video. Now, issue the Copy command to place a copy of the
selected text in the Windows clipboard.
The Copy command is a convenient way to transfer small amounts of data from LINGO to other
applications.
Edit|Paste Ctrl+V
Use the Paste command to replace the current selection in the active window with the contents of the
Windows clipboard. The Paste command is a convenient way to import small amounts of data from
other applications into your LINGO models.
124 CHAPTER 5
Edit|Paste Special...
Use the Paste Special command to insert the contents from the Windows clipboard into the active
window at the cursor insertion point. This command can do much more than insert just plain text as
done by the standard Paste command. Paste Special can be used to insert other objects and links to
other objects. This is particularly useful for adding links to supporting data for your model. By
inserting a link to your data sources, it is much easier to find and view them.
As an example, suppose we have the following transportation model:
! A 3 Warehouse, 4 Customer
Transportation Problem;
SETS:
WAREHOUSE / WH1, WH2, WH3/: CAPACITY;
CUSTOMER / C1, C2, C3, C4/: DEMAND;
ROUTES(WAREHOUSE, CUSTOMER): COST, VOLUME;
ENDSETS
! The objective;
MIN = @SUM(ROUTES: COST * VOLUME);
As we can see from the data section, we are importing data from the Excel file TRANLINKS.XLS and
writing the solution back out to the same file.
WINDOWS COMMANDS 125
A nice feature would be to insert a link to the spreadsheet into our model file. This way, we could view
the data and the solution without having to start Excel and load the spreadsheet. To do this, open Excel
and load the spreadsheet as we have done here:
For complete information on importing data from Excel, see Chapter 9, Interfacing with Spreadsheets.
126 CHAPTER 5
Now, select the range B2:F21 in the spreadsheet. Next, from Excel’s Edit menu, choose the Copy
command. Now, click on LINGO, place the cursor right before the data section, and give the
Edit|Paste Special command. Click on the Paste Link button in the dialog box, so you see the
following:
WINDOWS COMMANDS 127
Finally, click the OK button, and you should be able to see the spreadsheet contents in the LINGO
model:
This link will be saved as part of your LINGO file. Therefore, whenever you open the model, the
spreadsheet will be visible. Note that whenever you reopen the LINGO model, you may want to open
the link, so the contents are updated automatically. You can do this by selecting the spreadsheet in the
LINGO model, giving the Edit|Links command, and pressing the Open Links button in the dialog box.
As a final note, LINGO's compiler ignores all embedded links and objects. Thus, you are free to insert
links and objects wherever you choose in a model.
128 CHAPTER 5
Edit|Select All
Use the Select All command to select the entire contents of the active window. This is useful when you
want to copy the entire contents of the window elsewhere, or if you want to delete the contents of the
window.
Edit|Find... Ctrl+F
Use the Find command to search for a desired string of text in the active window. When you issue the
Find command, you should see the following dialog box:
Enter the text you wish to search for in the Find what box. Check the Match whole word only box to
have LINGO find only whole words of text (i.e., don’t search for occurrences of the text embedded in
other words). Check the Match case box to have LINGO search only for instances of the text with the
same capitalization. Click the Find Next button to find the next instance of the text.
Edit|Replace Ctrl+H
Use the Replace command to replace one string of text with another in the active window. When you
issue the Replace command, you will see the following dialog box:
WINDOWS COMMANDS 129
Enter the name of the text you want to replace in the Find what box. Enter the text you want to replace
the old text with in the Replace with box. Clicking the Find Next button will cause LINGO to find the
next occurrence of the old text. Clicking the Replace button will cause the next occurrence of the old
text to be replaced by the new text. The Replace All button will replace all occurrences of the old text
with the new text throughout the entire document.
Check the Match whole word only box to have LINGO replace only whole words of the text (i.e., don’t
replace occurrences of the text embedded in other words). Check the Match case box to have LINGO
replace only instances of the text with the same capitalization.
Enter a line number in the Go to line number box. Then, press the OK button and LINGO will jump to
the desired line number. Press the Top button to go to the top of the document, or the Bottom button to
go to the bottom.
Edit|Paste Function
Use the Paste Function command to paste any of LINGO’s built-in functions at the current insertion
point. Choose the category of the LINGO function you want to paste from the secondary menu, and
then select the function from the cascading menu.
In the following illustration, we have chosen the External Files category from the secondary menu:
On the right are all the functions that deal with external files. By selecting one of these functions,
LINGO will paste a template for the selected function into your document, with a suggestive
placeholder for each argument. You should then replace the argument placeholders with actual
arguments that are relevant to your model.
Edit|Select Font
Use the Select Font command to select a new font, size, style, color, or effect in which to display the
selected text. You may find it easier to read models and solution reports if you select a mono-spaced
font such as Courier. Custom fonts are preserved only when saving in the LG4 file format. (Refer to
the File|New command above for a description of LINGO’s various file types.)
Note: You cannot change the display color of text if syntax coloring is enabled. If you need to use
specific display colors in your document, you will need to disable syntax coloring.
WINDOWS COMMANDS 131
From the model’s data section, we see that we are using the @ODBC function to retrieve the values for
the NEED attribute from the STAFFING ODBC data source. We are also using the @ODBC function
to send the optimal values for the START attribute back to the same data source. Because this data
source is an integral part of our model, it would be nice to place a link to it in our model, so we can
retrieve it easily each time we want to refer to it. We can do this with the Edit|Insert New Object
command as follows:
1. Position the cursor in the model where you would like the icon for the link
to appear (Note, the LINGO parser ignores links to external objects, so you
can insert the link anywhere you like).
132 CHAPTER 5
2. Issue the Edit|Insert New Object command. You should see the following
dialog box:
Now, whenever you want to edit or view the supporting database, all you need do is double-click on
the icon. In this case, Microsoft Access will start and load the staffing database, so you will see the
following on the screen:
As a final note, keep in mind linked objects are preserved only when a model is saved in LG4 format
(see the File|New command above for details on the LG4 file format).
For complete information on exchanging data and solution values with data sources, see Chapter 10,
Interfacing with Databases.
134 CHAPTER 5
Edit|Links
Use the Links command to modify the properties of the links to external objects in a LINGO
document. The dialog box appears as follows:
Select the Automatic radio button to have LINGO automatically update the object when the source file
is changed. The Manual radio button allows you to update the object only when you select the Update
Now button.
The Open Source button is used to open the connection to an automatic link. Once the link has been
opened, any changes to the source document will be reflected in the view of the object in your LINGO
model.
The Change Source button is used to attach the link to a different source file.
Finally, the Break Link button is used to break the connection to the external object.
3. LINGO Menu
The LINGO menu, pictured at left,
contains commands that generally
pertain to solving a model and
generating reports. This menu also
contains the Options command for
customizing LINGO’s configuration.
LINGO|Solve Ctrl+U
Use the Solve command to have LINGO solve the model in the active window. The Solve command is
available only for model windows—report, script, and data windows cannot be solved.
When you solve a model, LINGO first examines the model’s syntax to determine if it is valid. If
LINGO finds a mistake in the syntax, you will be presented with a dialog box similar to the following:
In the Error Text box, LINGO prints the line number where the syntax error occurred, the text of the
line, and points to where LINGO determines that the error occurred. In most cases, LINGO is good at
pointing to where the error occurred. Sometimes, however, the error may not be located exactly where
LINGO is pointing. Be sure to examine neighboring lines for possible flaws as well. In this particular
example, the syntax error occurred in line 2, where we forgot to insert the multiplication signs (*)
between the two coefficients and variable names.
136 CHAPTER 5
When you issue the Solve command (assuming your model has no further syntax errors), LINGO will
post the solver status window. This window contains information about the composition of your model
and keeps you posted as to the progress of the solver. The solver status window resembles the
following:
For more information on the various fields in the solver status window, refer to Chapter 1, Getting
Started with LINGO.
WINDOWS COMMANDS 137
Once the solver has completed processing your model, it will create a new window containing the
Solution Report for your model. You can scroll through this window to examine its contents, save it to
a text file, or queue it to your printer. The following is a sample solution report window:
138 CHAPTER 5
LINGO|Solution Ctrl+W
Use the Solution command to generate a solution report for the active window. The solution report
may be in text or graphical format.
After selecting the model window that you want to generate a solution for, issue the LINGO|Solution
command and you will be presented with this dialog box:
In the Attribute or Row Name list box, select an attribute or row name that you would like a report for.
If you do not select a name in this box, LINGO will generate a full solution report that includes all
attributes and rows.
In the Header Text box, enter whatever text (e.g., “Values for X”) you would like to appear at the head
of the report.
In the box labeled Type of Output, you can select either text or graphical output. If you select Text,
LINGO will create a new window containing the solution in text format. If you select Graph, a new
window containing the solution in one of several different graphical formats will be created. Current
supported graph formats are bar, line, and pie charts.
Check the Nonzeros Only box to see a report that contains only the variables with a nonzero value
and/or only the constraints that are binding.
WINDOWS COMMANDS 139
If you select to have the solution displayed as a graph, the Graph Properties box will be undimmed,
which allows you to select options influencing the display of the graph. In the Graph Type box, you
have the option of selecting a Bar, Line, or Pie chart. In the Values box, you can select to graph either
Primal or Dual values. The Bounds box gives you the option of placing bounds on the values
displayed in the graph. If a number is entered in the Lower bound field, LINGO will only display
points in the graph that are greater-than-or-equal-to the value. Conversely, if a value is placed in the
Upper bound field, LINGO will only graph points less-than-or-equal-to the bound. If the Include
Labels box is checked, LINGO will label each point on the graph with the name of the corresponding
variable.
When you click OK, LINGO creates a new solution window containing the solution report. You can
use Cut and Paste commands to move the contents of the report to other applications. If the report is in
text format, you can also save it to a file.
Note: LINGO maintains only one solution in memory. This is the solution to the last window you
issued the LINGO|Solve command for. If you try to issue the Solution command for a window
that LINGO does not currently have a solution for, you will receive an error message. Thus, if
you plan to work with two or more models that take a long time to solve, be sure to save
copies of your solutions. This will allow you to refer to them later without having to re-solve
your models.
LINGO|Range Ctrl+R
Use the Range command to generate a range report for the model in the active window. A range report
shows over what ranges you can: 1) change a coefficient in the objective without causing any of the
optimal values of the decision variables to change, or 2) change a row’s constant term (also referred to
as the right-hand side coefficient) without causing any of the optimal values of the dual prices or
reduced costs to change.
Note: The solver computes range values when you solve a model. Range computations must be
enabled in order for the solver to compute range values. Range computations are not enabled
by default. To enable range computations, select the General Solver Tab under
LINGO|Options and, in the Dual Computations list box, choose the Prices and Ranges option.
Range computations can take a fair amount of computation time, so, if speed is a concern,
you don’t want to enable range computations unnecessarily.
The example model below, when solved, yields the range report that follows:
[OBJECTIVE] MAX = 20 * A + 30 * C;
[ALIM] A <= 60;
[CLIM] C <= 50;
[JOINT] A + 2 * C <= 120;
140 CHAPTER 5
Here is the range report:
Ranges in which the basis is unchanged:
Objective Coefficient Ranges
Current Allowable Allowable
Variable Coefficient Increase Decrease
A 20.00000 INFINITY 5.000000
C 30.00000 10.00000 30.00000
The first section of the report is titled Objective Coefficient Ranges. In the first column, titled Variable,
all the optimizable variables are listed by name. The next column, titled Current Coefficient, lists the
current coefficient of the variable in the objective row. The next column, Allowable Increase, tells us
the amount that we could increase the objective coefficient without changing the optimal values for the
variables. The final column, Allowable Decrease, lists the amount that the objective coefficient of the
variable could decrease before the optimal values of the variables would change. Information on the
allowable increases and decreases on objective coefficients can be useful when you need answers to
questions like, “How much more (less) profitable must this activity be before we should be willing to
do more (less) of it?”
Referring to the Objective Coefficient Ranges report for our example, we can say, as long as the
objective coefficient of A is greater-than-or-equal-to 15, the optimal values of the variables will not
change. The same may be said for the objective coefficient of variable C, as long as it falls within the
range of [0,40].
Note: Ranges are valid only if you are planning to alter a single objective or right-hand side
coefficient. The range information provided by LINGO cannot be applied in situations where
one is simultaneously varying two or more coefficients. Furthermore, ranges are only lower
bounds on the amount of change required in a coefficient to actually force a change in the
optimal solution. You can change a coefficient by any amount up to the amount that is
indicated in the range report without causing a change in the optimal solution. Whether the
optimal solution will actually change if you exceed the allowable limit is not certain.
The second section of the range report is Right-hand Side Ranges. The first column, Row, lists the
names of all the optimizable rows, or constraints, in the model. The second column, Current RHS,
gives the constant term, or right-hand side value, for the row. The next two columns, Allowable
Increase and Allowable Decrease, tell us how far we can either increase or decrease the right-hand
side coefficient of the row without causing a change in the optimal values of the dual prices or reduced
costs. If you recall, the dual prices on rows are, effectively, shadow prices that tell us at what price we
should be willing to buy (or sell) our resources for. The dual prices do not, however, tell us what
quantity we should be willing to buy (or sell) at the dual price. This information is obtained from the
allowable increases and decreases on the right-hand side coefficients for the row. So, for our example,
the dual prices and reduced costs will remain constant as long as the right-hand side of row ALIM falls
within the range [20,120], the right-hand side of CLIM is greater-than-or-equal-to 30, and the
right-hand side of JOINT is in [60,160].
WINDOWS COMMANDS 141
Note: We preceded all the constraints in our model with a name enclosed in square brackets. This is
an important practice if you wish to generate range reports. If you do not name your
constraints, LINGO assigns them a name that corresponds to the internal index of the
constraint. This internal index will not always correspond to the order of the constraint in the
text of the original model. So, to make the Right-hand Side Ranges section of range reports
meaningful, be sure to name all your constraints (See page 29 for details on assigning
constraint names).
If a variable is nonlinear in the objective, its value in the Current Coefficient column will be displayed
as NONLINEAR. Similarly, if a row is nonlinear, the value in the Current RHS column will be
displayed as NONLINEAR.
Coefficients that can be increased or decreased indefinitely will display a range of INFINITY.
Fixed variables are substituted out of a model and will not appear in a range report. Rows that contain
only fixed variables are also substituted out of models, and will also not appear in range reports. As an
example, suppose we changed the following inequality in our sample model from:
[ALIM] A <= 60;
to the equality:
[ALIM] A = 60;
LINGO can now solve directly for the value of A. The variable A is considered fixed, as is the row
ALIM (since it contains no optimizable variables). Given this, the variable A will no longer appear in
the Objective Coefficient Ranges section of the range report, and the row ALIM will not appear in the
Right-hand Side Ranges section. We can verify this by examining the updated range report:
Ranges in which the basis is unchanged:
Note: LINGO maintains the range report for only one model in memory. This is the report for the
window that you last issued the LINGO|Solve command for. If you try to issue the Range
command for a window that LINGO does not currently have range information for, you will
receive an error message. If you plan to work with two or more models that take a long time
to solve, be sure to save copies of your range reports to disk, so you can refer to them later
without having to re-solve your models.
142 CHAPTER 5
LINGO|Options Ctrl+I
Use the LINGO|Options command to change a number of parameters that affect LINGO's user
interface, as well as the way LINGO solves your model. When issuing the Options command, you will
be presented with the following dialog box:
Set these parameters to your personal preference and press the Apply button to set them for the extent
of the current LINGO session. The currently selected settings are also applied when you click the OK
button, with the one difference being that the OK button closes the dialog box. If you would like the
current parameter settings to be maintained for use in subsequent LINGO sessions, click the Save
button. The original default settings can be restored at any time by clicking the Default button.
WINDOWS COMMANDS 143
Note: LINGO uses the LINDO API as its solver engine. The LINDO API has a wealth of advanced
parameter settings to control its various solvers. Most of the more relevant parameters may
be set through the LINGO|Options command. However, some of the more advanced
parameters must be set using the APISET command.
144 CHAPTER 5
Interface Tab
The Interface tab on the Options dialog box (shown above) can be used to control the appearance of
LINGO, LINGO’s output, and the default file format.
General Box
The General box on the Interface tab:
Errors In Dialogs
If the Errors In Dialogs box is checked, LINGO will display error messages issued by the solver in a
modal dialog box. This dialog box must be cleared before LINGO proceeds with any other operation.
In some instances, you may have LINGO embedded in other applications, where it may not be
desirable, or possible, to have users clearing error dialogs. By unchecking this option, LINGO will
route the solver's error messages to the command window, where they will be displayed, and no user
intervention will be required to clear the messages.
The default is for solver errors to be displayed in dialog boxes.
WINDOWS COMMANDS 145
Note: This option allows you to route only those error messages generated by LINGO's solver to the
report window. Error messages displayed by LINGO's interactive front-end will always be
posted in dialog boxes.
Splash Screen
If the Splash Screen box is checked, LINGO will display its splash screen each time it starts up. The
splash screen lists the release number of LINGO and the software’s copyright notice. If you disable
this option, LINGO will not display the splash screen.
The default is for the splash screen to be displayed.
Status Bar
If the Status Bar box is checked, LINGO displays a status bar along the bottom of the main frame
window. Among other things, the status bar displays the time of day, location of the cursor, menu tips,
and the current status of the program. To remove the status bar from the screen, clear the Status Bar
checkbox.
The default is for LINGO to display the status bar.
146 CHAPTER 5
Status Window
If the Status Window box is checked, LINGO displays a solver status window whenever you issue the
LINGO|Solve command. This window resembles the following:
The solver status window is useful for monitoring the progress of the solver and the dimensions of
your model. It is updated every n seconds, where n is the value in the Update interval field in the lower
right corner of the window. For a detailed description of the various fields in the solver status window,
see the section Solver Status Window in Chapter 1, Getting Started with LINGO.
WINDOWS COMMANDS 147
Output Level
You can use the Output Level setting to control the amount of output LINGO generates. There are
four settings available:
Verbose—Causes LINGO to display the maximum amount of output, including full solution
reports.
Terse—Less output than Verbose, with full solution reports suppressed. This is a good output
level if you tend to solve large models. LINGO also suppresses Export Summary Reports
generated when exporting data to spreadsheets or databases.
Errors Only—All output is suppressed, with the exception of error messages.
Nothing—LINGO suppresses all output. This level may be useful when taking advantage of
the programming capabilities in LINGO, in which case, you will add statements to your
model to generate all required output.
The default is for LINGO to be in verbose mode.
Toolbar
If the Toolbar box is checked, LINGO displays its command toolbar containing buttons, which act as
shortcuts to various commands contained in the LINGO menu. For definitions of the buttons on the
toolbar, please see the section The Toolbar at the beginning of this chapter. If the Toolbar checkbox is
unchecked, LINGO does not display its toolbar.
The default is for LINGO to display its toolbar.
Solution Cutoff
On occasion, due to roundoff error, some of the values returned by LINGO’s solver will be very small
(less than 1e-10). In reality, the true values of these variables are either zero or so small as to be of no
consequence. These tiny values can be distracting when interpreting a solution report. The Solution
Cutoff parameter can be used to suppress small solution values. Any solution value
less-than-or-equal-to Solution Cutoff will be reported as being zero.
The default value for Solution Cutoff is 1e-9.
148 CHAPTER 5
File Format Box
The File Format box on the Interface tab:
is used to select the default file format that LINGO uses to save models to disk. There are three
different formats to choose from: LG4, LNG, or LTX.
The LG4 format is a binary format readable only by LINGO. This format enables you to have custom
formatting, fonts in your models, and to use LINGO as an OLE server and container. Files saved in the
LG4 format are readable only by Windows versions of LINGO.
The LNG format is a text-based format. Thus, models saved in the LNG format can be read into other
applications. LNG format models are transferable to other platforms running LINGO. Models saved in
LNG format cannot contain custom formatting or embedded objects.
LTX files are model files that use the LINDO syntax. Longtime LINDO users may prefer LINDO
syntax over LINGO syntax. LINDO syntax is convenient for quickly entering small to medium sized
linear programs. As long as a file has an extension of .ltx, LINGO will assume that the model is
written using LINDO syntax. Readers interested in the details of LINDO syntax may contact LINDO
Systems to obtain a LINDO user’s manual.
The default file format is LG4.
is used to control the syntax coloring capability in LINGO’s editor. LINGO’s editor is “syntax aware.”
In other words, when it encounters LINGO keywords, it displays them in blue. Comments are
displayed in green, and all remaining text is displayed in black. Matching parentheses are also
highlighted in red when you place the cursor immediately following a parenthesis.
The controls available in this box are: Line Limit, Delay, and Paren Match.
WINDOWS COMMANDS 149
Line Limit
Syntax coloring can take a long time if you have very large files. The Line Limit field sets the
maximum acceptable file size for syntax coloring. Files with line counts exceeding this parameter will
not be syntax colored.
Setting this parameter to 0 will disable the syntax coloring feature. The default line limit is 1000 lines.
Delay
The Delay field sets the number of seconds LINGO waits after the last keystroke was typed before
re-coloring modified text. Users on slower machines may want to set this higher to avoid having
syntax coloring interfere with typing. Users on faster machines may want to decrease this value, so text
is re-colored quickly.
The default is 0 seconds.
Paren Match
If the Paren Match box is checked, LINGO will highlight matching parentheses in red when you place
the cursor immediately following a parenthesis. In other words, by placing the cursor immediately
after one of the parentheses of interest, you will notice that the color of the parenthesis changes from
black to red. LINGO will simultaneously display the matching parenthesis in red. These parentheses
will remain displayed in red until you move the cursor to another position, at which point they will be
returned to a black color.
The default is for parenthesis matching to be enabled.
Echo Input
When you run a LINGO command script with File|Take Commands, the commands LINGO processes
are normally not displayed. If the Echo Input box is checked, processed commands will be displayed in
the command window. This can be a useful feature when you are trying to develop and debug a
LINGO command script.
The default is to not echo input.
is used to control the total number of output lines that can be stored in the command window.
When LINGO sends output to the command window, it places it at the bottom of the window. All
previous output is scrolled up to make way for the new output. The Maximum field sets the maximum
number of output lines allowed in the command window. When LINGO hits this limit, it starts
removing lines from the top of the command window until there are n lines left, where n is the value of
the Minimum field.
In general, output to the command window will become slower as the maximum and minimum line
counts are increased, or the difference between the maximum and minimum is decreased. If you have a
long session you need to save, you can use the File|Log Output command to log all command window
output to disk.
The default value for Line Count Limits is 800 lines maximum and 400 lines minimum.
is used to control the page length and width of the command window.
WINDOWS COMMANDS 151
If you would like LINGO to pause after a certain number of lines have been written to the command
window, you can do so by setting the Length field in the Page Size Limits box. When LINGO hits this
limit, it will display the following button on your screen:
LINGO will wait until you press the More button to display any subsequent output in the command
window. The default is None, meaning no page length limit is imposed.
When LINGO generates reports, it limits output lines to a certain width. In some reports, lines will be
wrapped, so they fall within the line limit. In other reports, lines may be truncated. Because LINGO
concatenates variable names in performing set operations, a variable name such as
SHIPMENTS(WAREHOUSE1, CUSTOMER2) may result. This could be truncated in a solution report
if too narrow an output width is used. You can control this line width limit through the Width field of
the Page Size Limits box. You may set it anywhere between 64 and 200, with the default being 76.
152 CHAPTER 5
can be used to control several general parameters related to the functioning of LINGO’s solver.
WINDOWS COMMANDS 153
Debugger Box
The Debugger box on the General Solver tab:
gives you control over the output level and the solver used as part of the model debugging command,
LINGO|Debug. The debugger is very useful in tracking down problems in models that are either
infeasible or unbounded .
The Output Level option controls how much output the model debugger generates. Possible output
levels range from 1 (minimum output) to 15 (maximum output). In general, you will want to generate
as much output as possible. The only reason to restrict the amount of output would be to speed
debugging times on large models.
The default setting for the debugger output level is 15.
The Cold Start Solver and Warm Start Solver options control the solver used on linear models for cold
starts (starting without an existing basis in memory) and warm starts (restarting from an existing basis)
during the debugging process. In either case, the available options are
♦ Solver Decides — LINGO selects the solver it believes is the most appropriate,
♦ Primal — the primal simplex solver will be used,
♦ Dual — the dual simplex solver will be used, and
♦ Barrier — the barrier solver will be used (requires a barrier solver license).
With some models, you may find that choosing a particular solver improves overall performance of the
debugger.
LINGO defaults to Solver Decides for both the cold and warm debug solver.
154 CHAPTER 5
Runtime Limits Box
The Runtime Limits box on the General Solver tab:
is used to control the length of time the solver spends on your model.
The first field, Iterations, allows you to place an upper limit on the number of iterations the solver will
perform. An iteration is the fundamental operation performed by the solver. At the risk of
oversimplification, it is a process that involves forcing a variable, currently 0, to become nonzero until
some other variable is driven to zero, improving the objective as we go. In general, larger models will
take longer to perform an iteration and nonlinear models will take longer than linear models. The
default iteration limit is None, meaning no limit is imposed on the iteration count.
The second field in the Runtime Limits box, Time (sec), is a limit on the amount of elapsed time the
solver is allowed when optimizing a model. The default time limit is None, meaning no limit is
imposed on the length of time the solver can run.
If the solver hits either of these limits, it returns to normal command mode. If the model contains
integer variables, LINGO will restore the best solution found so far. You may need to be patient,
however, because the solver may have to perform a fair amount of work to reinstall the current best
solution after it hits a runtime limit.
Note: When the solver is interrupted, the only time it will return a valid solution is when the model
contains integer variables and an incumbent integer solution exists. In which case, the solver
backtracks to the incumbent solution before exiting. Interrupting a model without integer
variables will result in an undefined solution. Interrupting a model with integer variables but
no incumbent solution will also return an undefined solution.
Note: If solution times are a concern, you should avoid unnecessarily enabling range computations.
tells LINGO to convert all variable and row names to RC notation when performing MPS file format
Input/Output.
RC format involves renaming each row (constraint) in a model to be Rn, where n is the row’s index.
Similarly, each column (variable) is renamed to Cn. In addition, LINGO renames the objective row to
be ROBJ. Refer to the Importing MPS Files section under the File|Open command earlier in this
chapter for a discussion of RC notation and why this option is useful.
By default, LINGO disables the use of RC format names.
tells LINGO to place a default lower bound of 0 on all variables. In other words, unless otherwise
specified, variables will not be allowed to go negative. Should you want a variable to take on a
negative value, you may always override the default lower bound of 0 using the @BND() function. If
this option is disabled, then LINGO’s default assumption is that variables are unconstrained and may
take on any value, positive or negative. Unconstrained variables are also referred to as be being free
156 CHAPTER 5
is used to control several parameters related to the generation of the model. The model generator takes
the expressions in your LINGO model and converts them to a format understood by the solver engines
that find the actual solutions to the model.
WINDOWS COMMANDS 157
Generator Memory Limit Box
The Generator Memory Limit box on the Model Generator tab:
is used to control the amount of memory set aside to use as workspace for generating a model.
Large models may run out of generator memory when you attempt to solve them. In this case, you will
receive the error message "The model generator ran out of memory.” To avoid this error, increase the
amount of memory in the Generator Memory Limit field. You will then need to click the Save button
and restart LINGO. Since LINGO sets aside this memory when it starts, changes in LINGO's generator
memory limit are not established until you restart the program.
To determine exactly how much generator memory LINGO was able to successfully allocate, run the
Help|About LINGO command. The About LINGO dialog box displays the amount of generator
memory allocated at startup.
The memory allocated to LINGO’s generator will not be available to the various solver engines
contained in LINGO. Thus, you should not allocate overly excessive amounts of memory to the
generator.
If you set LINGO's generator memory limit to None, LINGO will allocate all available memory when
it starts up. This is not a recommended practice. The default size for the workspace is 32Mb.
Note: By setting LINGO's generator memory limit abnormally high, both LINGO and Windows
will resort to swapping virtual memory to and from the hard drive, which can slow down your
machine dramatically and result in poor performance. In general, set the memory allocation to
a level high enough to comfortably handle your largest models, but not too much higher than
that. You can view the amount of memory used in the allotted workspace at any time by
opening the solver status window and examining the Generator Memory Used field.
is used to set the priority of the unary minus operator. The two available options are High and Low.
158 CHAPTER 5
There are two theories as to the priority that should be assigned to the unary minus (i.e., negation)
operator in mathematical expressions. On the one hand, there is the Excel practice that the unary
minus operator should have the highest priority, in which case, the expression -3^2 would evaluate to
+9. On the other hand, there is the mathematicians’ preference for assigning a lower priority to unary
minus than is assigned to exponentiation, in which case, -3^2 evaluates to -9. Note that regardless
which relative priority is used, one can force the desired result through the use of parenthesis.
LINGO defaults to the Excel approach of setting a higher priority (High) on negation than on
exponentiation.
is used to control the degree to which fixed variables are substituted out of the ultimate math program
passed to the solver engines.
For example, consider the model:
MAX= 20*X + 30*Y + 12*Z;
X = 2*Y;
X + Y + Z <= 110;
Y = 30;
If we run the LINGO|Generate command, we see that LINGO is able to reduce this model down to the
equivalent, but smaller model:
MAX= 12 * Z + 2100;
Z <= 20;
From the third constraint of the original model, it is obvious that Y is fixed at the value 30. Plugging
this value for Y into the first constraint, we can conclude that X has a value of 60. Substituting these
two fixed variables out of the original formulation yields the reduced formulation above.
In most cases, substituting out fixed variables yields a smaller, more manageable model. In some
cases, however, you may wish to avoid this substitution. An instance in which you might want to
avoid substitution would be when equations have more than one root. When multiple roots are
present, reduction may select a suboptimal root for a particular equation. On the other hand, the global
and multistart solvers are adept at handling equations containing multiple roots. Thus, when using
these solvers one may wish to forgo fixed variable reduction.
The available options are:
♦ None,
♦ Always,
♦ Always, but linear only with global and multi, and
♦ Linear rows only.
WINDOWS COMMANDS 159
Selecting None disables all fixed variable reduction. Selecting Always enables reduction. When
Always, but linear only with global and multi is selected, LINGO always enables reduction except
when either the global or multistart solvers are selected, in which case it will only perform reduction
on rows where the key variable appears linearly. The Linear rows only option always limits reduction
to rows in which the key variable is linear.
Note: You should be careful when turning off fixed variable reduction. If the model generator is
unable to substitute out fixed variables, you may end up turning a linear model into a more
difficult nonlinear model.
LINGO defaults to selecting Always, but linear only with global and multi for fixed variable reduction.
is used to control the frequency with which LINGO regenerates a model. Commands that will trigger
the model generator are LINGO|Solve, LINGO|Generate, LINGO|Model Statistics, LINGO|Picture,
LINGO|Debug, and File|Export File.
The choices available under this option are:
♦ Only when text changes - LINGO regenerates a model only when a change has been
made to the model’s text since the last generation took place.
♦ When text changes or with external references – (default) LINGO regenerates whenever
a change is made to the model text or when the model contains references to external
data sources (e.g., text files, databases, or spreadsheets).
♦ Always - LINGO always regenerates the model each time information regarding the
generated model is needed.
160 CHAPTER 5
Linearization
The Linearization box on the Model Generator tab:
controls the linearization option in LINGO. Many nonlinear operations can be replaced by linear
operations that are mathematically equivalent. The ultimate goal is to replace all the nonlinear
operations in a model with equivalent linear ones, thereby allowing use of the faster and more robust
linear solvers. We refer to this process as linearization. For more information on linearization, please
refer to the Linearization section in Chapter 14, On Mathematical Modeling. Degree determines the
extent to which LINGO will attempt to linearize models.
The available options are:
♦ Solver Decides,
♦ None,
♦ Low, and
♦ High
Under the None option, no linearization occurs. With the Low option, LINGO linearizes @ABS(),
@FLOOR(), @IF(), @MAX(), @MIN(), @SIGN(). @SMAX(), and @SMIN() function references
along with any products of binary and continuous variables. The High option is equivalent to the Low
option plus LINGO will linearize all logical operators (#LT#, #LE#, #EQ#, #GT#, #GE#, and #NE#).
Under the Solver Decides option, LINGO will do maximum linearization if the number of variables is
less-than-or-equal-to 12. Otherwise, LINGO will not perform any linearization. LINGO defaults to the
Solver Decides setting.
The Delta Coefficient is a tolerance indicating how closely you want the additional constraints added
as part of linearization to be satisfied. Most models won’t require any changes to this parameter.
However, some numerically challenging formulations may benefit from increasing Delta slightly.
LINGO defaults to a Delta of 1.e-6.
When LINGO linearizes a model, it will add forcing constraints to the mathematical program
generated to optimize your model. These forcing constraints are of the form:
f( Adjustable Cells) = M • y
where M is the BigM Coefficient and y is a 0/1 variable. The idea is that if some activity in the
variables is occurring, then the forcing constraint will drive y to take on the value of 1. Given this, if
we set the BigM value to be too small, we may end up with an infeasible model. Therefore, the astute
reader might conclude that it would be smart to make BigM quite large, thereby minimizing the chance
of an infeasible model. Unfortunately, setting BigM to a large number can lead to numerical stability
problems in the solver resulting in infeasible or sub-optimal solutions. So, getting a good value for the
BigM Coefficient may take some experimentation.
WINDOWS COMMANDS 161
As an example of linearization, consider the following model:
MODEL:
SETS:
projects: baths, sqft, beds, cost, est;
ENDSETS
DATA:
projects, beds, baths, sqft, cost =
p1 5 4 6200 559608
p2 2 1 820 151826
p3 1 1 710 125943
p4 4 3 4300 420801
p5 4 2 3800 374751
p6 3 1 2200 251674
p7 3 2 3400 332426
;
ENDDATA
MIN = @MAX( projects: @abs( cost - est));
@FOR( projects:
est = a0 + a1 * beds + a2 * baths + a3 * sqft
);
END
Model: COSTING
This model estimates the cost of home construction jobs based on historical data on the number of
bedrooms, bathrooms, and square footage. The objective minimizes the maximum error over the
sample project set. Both the @MAX() and @ABS() functions in the objective are non-smooth nonlinear
functions, and, as a result, can present problems for LINGO’s default, local search NLP solver.
Running the model under the default settings with linearization disabled, we get the following result:
Local optimal solution found at step: 91
Objective value: 3997.347
Variable Value Reduced Cost
A0 37441.55 0.000000
A1 27234.51 0.000000
A2 23416.53 0.000000
A3 47.77956 0.000000
Note that the maximum error has been reduced from 3,997 to 1,426!
Linearization will substantially increase the size of your model. The sample model above, in un-
linearized form, has a mere 8 rows and 11 continuous variables. On the other hand, the linearized
version has 51 rows, 33 continuous variables, and 14 binary variables! Although linearization will
162 CHAPTER 5
cause your model to grow in size, you will tend to get much better solution results if the model can be
converted entirely to an equivalent linear form.
Note: Linearization will be of most use when a nonlinear model can be 100% linearized. If LINGO
can only linearize a portion of your model, then you may actually end up with a more difficult
nonlinear model.
You may view the linearization components that are added to your model with the GEN command. We
will illustrate with the following model:
model:
min = @abs( x - 5) + 4 * @abs( y -3);
x + 2 * y <= 10;
end
Setting the Linearization option to High and running the LINGO|Generate command on this model
yields the following linearized model:
MODEL:
MIN= _C03 + 4 * _C07 ;
X + 2 * Y <= 10 ;
[_R01] - _C01 - _C02 + _C03 = 0 ;
[_R02] _C01 - 100000 * _C04 <= 0 ;
[_R03] _C02 + 100000 * _C04 <= 100000 ;
[_R04] X - _C01 + _C02 = 5 ;
[_R05] - _C05 - _C06 + _C07 = 0 ;
[_R06] _C05 - 100000 * _C08 <= 0 ;
[_R07] _C06 + 100000 * _C08 <= 100000 ;
[_R08] Y - _C05 + _C06 = 3 ;
@BND( 0, _C01, 100000) ; @BND( 0, _C02, 100000) ; @BND( 0,
_C03, 100000) ; @BIN( _C04); @BND( 0, _C05, 100000) ;
@BND( 0, _C06, 100000) ; @BND( 0, _C07, 100000) ; @BIN(
_C08);
END
Columns added due to linearization have names beginning with _C, while linearization rows have
names starting with _R. In this particular example, rows _R01 through _R08 and columns _C01
through _C08 were added due to linearization.
The linearization option is set to Solver Decides by default.
allows for backward compatibility with models created in earlier releases of LINGO.
WINDOWS COMMANDS 163
In many instances, you will need to get the index of a primitive set member within its set. Prior to
release 4 of LINGO, you could do this by using the primitive set member’s name directly in the
model’s equations. This can create problems when you are importing set members from an external
source. In this case, you will not necessarily know the names of the set members beforehand. When
one of the imported primitive set members happens to have the same name as a variable in your model,
unintended results can occur. More specifically, LINGO will not treat such a variable as optimizable.
In fact, it would treat it as if it were a constant equal to the value of the index of the primitive set
member!
In short, different primitive set names can potentially lead to different results. Therefore, starting with
release 4.0 of LINGO, models such as the following were no longer permitted:
MODEL:
SETS:
DAYS /MO TU WE TH FR SA SU/;
ENDSETS
INDEX_OF_FRIDAY = FR;
END
If you want the index of FR in the DAYS set, you should now use the @INDEX function:
INDEX_OF_FRIDAY = @INDEX(DAYS, FR);
By default, LINGO disables the use of primitive set member names.
This option can be used for minimizing memory usage on models that are entirely linear. When this
option is in effect, the model generator can take steps to dramatically reduce overall memory
consumption without sacrificing performance. In fact, if all your models are linear, we recommend
that you enable this option permanently as the default for your installation. The one restriction is that
the model must prove to be entirely linear. If a single nonlinearity is detected, you will receive an
error message stating that the model is nonlinear and model generation will cease. At which point, you
should clear this option and attempt to solve the model again.
By default, the Assume model is linear option is disabled.
164 CHAPTER 5
Check for Duplicate Names
The Check for duplicate names in data and model checkbox on the General Solver tab:
allows you to test your LINGO models from older releases for instances where primitive set members
appear in the model’s equations. The next time you run a model, LINGO will issue an error message if
duplicate names appear as set members and as variables in the model.
Earlier releases of LINGO allowed you to use primitive set names in the equations of a model.
Primitive set names in a model’s equations returned the index of the set member. Starting with release
4.0, LINGO required you to use the @INDEX function (see the Chapter 7, LINGO's Operators and
Functions) to get the index of a primitive set member.
By default, this option is disabled.
may be used to guide LINGO’s memory usage. Enabling Minimize memory usage causes LINGO to
opt for less memory usage when solving a model. The downside is that opting for less memory may
result in longer runtimes.
LINGO defaults to disabling Minimize memory usage.
WINDOWS COMMANDS 165
can be used to control several options, discussed below, for tailoring the operation of LINGO’s linear
solver. The linear solver is used on linear models and on mixed integer linear models as part of the
branch-and-bound process.
166 CHAPTER 5
Method Box
The Method box on the Linear Solver tab:
is used to control the amount of model reduction performed by LINGO’s linear solver.
Your options are:
♦ Off - Disables reduction,
♦ On - Reduction is used on all models, and
♦ Solver Decides - LINGO decides whether or not to use reduction.
When this option is enabled, LINGO attempts to identify and remove extraneous variables and
constraints from the formulation before solving. In certain cases, this can greatly reduce the size of the
final model to be solved. Sometimes, however, reduction merely adds to solution times without
trimming back much on the size of the model.
LINGO defaults to the Solver Decides option.
WINDOWS COMMANDS 167
Feasibility Tolerance Boxes
The Initial Linear Feasibility Tol. and the Final Linear Feasibility Tol. boxes on the Linear Solver tab:
are used to control the feasibility tolerances for the linear solver. These tolerances are related to how
closely constraints must be satisfied in linear models. In general, if your models are well formulated,
you should not have to modify these tolerances. However, access to these tolerances is provided for the
expert user.
Due to the finite precision available for floating point operations on digital computers, LINGO can’t
always satisfy each constraint exactly. Given this, LINGO uses these two tolerances as limits on the
amount of violation allowed on a constraint while still considering it “satisfied”. These two tolerances
are referred to as the Initial Linear Feasibility Tolerance (ILFT) and the Final Linear Feasibility
Tolerance (FLFT). The default values for these tolerances are, respectively, .000003 and .0000001.
The ILFT is used when the solver begins iterating. In the early stages of the solution process, having
the solver less concerned with accuracy issues can boost performance. When LINGO thinks it has an
optimal solution, it switches to the more restrictive FLFT. At this stage in the solution process, you
want a relatively high degree of accuracy. Thus, the FLFT should be smaller than the ILFT.
One instance where these tolerances can be of use is when LINGO returns a solution that is almost
feasible. You can verify this by checking the values in the Slack or Surplus column in the model’s
solution report. If there are only a few rows with small negative values in this column, then you have a
solution that is close to being feasible. Loosening (i.e., increasing) the ILFT and FLFT may help you
get a feasible solution. This is particularly true in a model where scaling is poor (i.e., very large and
very small coefficients are used in the same model), and the units of measurement on some constraints
are such that minor violations are insignificant. For instance, suppose you have a budget constraint
measured in millions of dollars. In this case, a violation of a few pennies would be of no consequence.
Short of the preferred method of rescaling your model, loosening the feasibility tolerances may be the
most expedient way around a problem of this nature.
is used to control the pricing strategy used by LINGO’s simplex solvers. Pricing determines the
relative attractiveness of the variables during the simplex algorithm.
168 CHAPTER 5
For the Primal Solver, you have the following choices:
♦ Solver Decides - LINGO selects the pricing method it believes is the most appropriate.
♦ Partial - LINGO prices out a small subset of variables at each iteration and intermittently
prices out all the variables to determine a new subset of interesting variables.
♦ Devex - Devex prices out all columns at each iteration using a steepest-edge
approximation (see below).
Partial pricing tends to yield faster iterations. Devex, while slower, results in fewer overall iteration
and can be helpful when models are degenerate. Thus, it is difficult to determine what method is
superior beforehand.
For the Dual Solver, you have these options:
♦ Solver Decides - LINGO selects the pricing method it believes is the most appropriate.
♦ Dantzig - The dual simplex solver will tend to select variables that offer the highest
absolute rate of improvement to the objective regardless of how far other variables may
have to move per unit of movement in the newly introduced variable.
♦ Steepest-Edge - The dual solver spends a little more time selecting variables by looking
at the total improvement in the objective when adjusting a particular variable.
Dantzig pricing generally yields faster iterations. However, the other variables in the model may
quickly hit a bound resulting in little gain to the objective. With the steepest-edge option, each iteration
will tend to lead to larger gains in the objective resulting in fewer overall iteration. However, each
iteration will tend to take more compute time.
LINGO defaults to Solver Decides for both the primal and dual solvers.
allows you to control the setting for the linear optimality tolerance. This tolerance is used to determine
whether a reduced cost on a variable is significantly different from zero. You may wish to loosen this
tolerance (make it larger) on poorly scaled and/or large models to improve performance.
The default setting for the Linear Optimality Tolerance is 1.e-7.
WINDOWS COMMANDS 169
Matrix Decomposition
The Matrix Decomposition box on the Linear Solver tab:
controls several options that affect the operation of LINGO’s solver on nonlinear models.
WINDOWS COMMANDS 171
Initial Nonlinear Feasibility Tolerance
Final Nonlinear Feasibility Tolerance
The Initial Nonl Feasibility Tol and the Final Nonl Feasibility Tol boxes on the Nonlinear Solver tab:
are used to control the feasibility tolerances for the nonlinear solver in the same manner that the Initial
Linear and Final Linear Feasibility Tolerance are used by the linear solver. For information on how
and why these tolerances are useful, refer to the Feasibility Tolerances section in the Linear Solver
Tab section immediately above.
Default values for these tolerances are, respectively, .001 and .000001.
is used to terminate the solution process if little or no progress is being made in the objective value.
Specifically, if the objective function’s value has not improved significantly in n iterations, where n is
the value of SPIL, the nonlinear solver will terminate the solution process. Increasing this tolerance’s
value will tend to force the solver to run longer and may be useful in models with relatively “flat”
objective functions around the optimal solution.
The default value for SPIL is 5 iterations.
172 CHAPTER 5
Nonlinear Solver Version
The Nonlinear Solver Version box on the Nonlinear Solver tab:
♦ Solver Decides — LINGO selects the solver (ver 2.0 in this case),
♦ Ver 1.0, and
♦ Ver 2.0.
This option is available on the off chance that the older version of the nonlinear solver performs better
on a particular model.
LINGO defaults to Solver Decides for the nonlinear solver version.
Derivative Computation
The Derivatives box on the Nonlinear Solver tab:
The First Order option determines how the nonlinear solver computes first order derivatives. There are
two general methods available: numerical or analytical derivatives. Analytical derivatives are
computed directly by symbolically analyzing the arithmetic operations in a constraint. Numerical
derivatives are computed using finite differences. There are two types of numerical derivatives
available using either central differences or forward differences. There are also two types of analytical
derivatives available: backward analytical and forward analytical. Finally, a Solver Decides option is
also available. LINGO defaults to the Solver Decides setting for the First Order option, which
presently involves LINGO using backward analytical derivatives. However, one of the other choices
may be more appropriate for certain classes on nonlinear models. We suggest you try the various
derivative options to see which works best for your particular models.
The Use Second Order option determines if the nonlinear solver will use second order derivates. If
used, second order derivatives will always be computed analytically. Computing second order
derivatives will take more time, but the additional information they provide may lead to faster runtimes
and/or more accurate solutions. LINGO defaults to not using second order derivatives.
WINDOWS COMMANDS 173
Strategies Box
The Strategies box on the Nonlinear Solver tab:
Global Solver
If the Global Solver box is checked, LINGO will invoke the global solver when you solve a model.
Many nonlinear models are non-convex and/or non-smooth (for more information, see Chapter 14, On
Mathematical Modeling). Nonlinear solvers that rely on local search procedures (as does LINGO’s
default nonlinear solver) will tend to do poorly on these types of models. Typically, they will converge
to a local, sub-optimal point that may be quite distant from the true, global optimal point. Global
solvers overcome this weakness through methods of range bounding (e.g., interval analysis and convex
analysis) and range reduction techniques (e.g., linear programming and constraint propagation) within
a branch-and-bound framework to find the global solutions to non-convex models.
The only drawback to the global solver is that it runs considerably slower than the default local solver.
Therefore, the preferred option is to always try and write smooth, convex nonlinear models, so the
faster, default local solver can successfully solve them.
The global solver is disabled by default.
174 CHAPTER 5
Quadratic Recognition
If the Quadratic Recognition box is checked, LINGO will use algebraic preprocessing to determine if
an arbitrary nonlinear model is actually a quadratic programming (QP) model. If a model is found to
be a QP model, then it can be passed to the faster quadratic solver. Note that the QP solver is not
included with the standard, basic version of LINGO, but comes as part of the barrier option.
LINGO defaults to not using quadratic recognition
SLP Directions
If the SLP Directions box is checked, LINGO’s nonlinear solver will use successive linear
programming to compute new search directions. This technique uses a linear approximation in search
computations in order to speed iteration times. In general, however, the number of total iterations will
tend to rise when SLP Directions are used.
LINGO defaults to using SLP Directions.
Steepest Edge
If the Steepest Edge box is checked, LINGO’s nonlinear solver will use the steepest-edge strategy
when selecting variables to iterate on.
When LINGO is not in steepest-edge mode, the nonlinear solver will tend to select variables that offer
the highest absolute rate of improvement to the objective, regardless of how far other variables may
have to move per unit of movement in the newly introduced variable. The problem with this strategy is
that other variables may quickly hit a bound, resulting in little gain to the objective.
With the steepest-edge option, the nonlinear solver spends a little more time in selecting variables by
looking at the rate that the objective will improve relative to movements in the other nonzero
variables. Thus, on average, each iteration will lead to larger gains in the objective. In general, the
steepest-edge option will result in fewer iterations. However, each iteration will take longer.
LINGO defaults to not using the Steepest-Edge option.
WINDOWS COMMANDS 175
can be used to control several options for tailoring the operation of LINGO’s integer programming
pre-solver. The integer pre-solver does a great deal of model reformulation, so that the final
formulation passed to the branch-and-bound solver may be solved as fast as possible. The reformulated
model is always mathematically equivalent to the original formulation, but it is structured in such a
way that it is best suited for solution by the branch-and-bound integer programming algorithm.
The integer pre-solver operates only with linear integer models (i.e., models that make use of the
@BIN and @GIN functions to restrict one or more variables to integer values). Integer pre-solver
option settings have no effect on nonlinear integer models.
176 CHAPTER 5
Heuristics
The Heuristics box on the Integer Pre-Solver tab:
controls the level of integer programming heuristics used by the integer solver. These heuristics use the
continuous solution at each node in the branch-and-bound tree to attempt to quickly find a good integer
solution. Heuristics are only applied to linear, integer programming models. Requesting heuristics on
nonlinear models and/or pure linear programs will result in no benefits.
The Level field controls the level and number of heuristics that are applied, and may run from 0 (none)
to 100 (highest level). The Min Seconds field specifies the minimum amount of time to spend on
heuristics at each node.
The default values are 3 for Level and 0 for Min Seconds.
Probing Level
The Probing Level option on the Integer Pre-Solver tab can be used on mixed integer linear programs
to perform an operation known as probing. Probing involves taking a close look at the integer
variables in a model and deducing tighter variable bounds and right-hand side values. In many cases,
probing can tighten an integer model sufficiently to speed overall solution times. In other cases,
however, probing may not be able to do much tightening, and the overall solution time will increase
due to the extra time spent probing.
Pulling down the selection list for the Probing Level field:
you will see that you can choose one of eight different probing levels. A probing level of 1 means
probing is disabled, while levels 2 through 7 indicate successively higher degrees of probing. The
default setting for this option, Solver Decides, leaves the decision up to LINGO to select the probing
level.
WINDOWS COMMANDS 177
Constraint Cuts Box
The tolerances contained in the Constraint Cuts box on the Integer Pre-Solver tab:
can be used to control the solver’s cut generation phase on linear models.
LINGO’s integer programming pre-solver performs extensive evaluation of your model in order to add
constraint cuts. Constraint cuts are used to “cut” away sections of the feasible region of the continuous
model (i.e., the model with integer restrictions dropped) that are not contained in the feasible region to
the integer model.
On most integer models, this will accomplish two things. First, solutions to the continuous problem
will tend to be more naturally integer. Thus, the branch-and-bound solver will have to branch on fewer
variables. Secondly, the bounds derived from intermediate solutions will tend to be tighter, allowing
the solver to “fathom” (i.e., drop from consideration) branches higher in the branch-and-bound tree.
These improvements should dramatically speed solution times on most integer models.
Note: Cuts are not applied to nonlinear models. Thus, modifying any of the tolerances in the
Constraint Cuts box will have no bearing on nonlinear models.
Application
In the Application drop-down box of the Constraint Cuts box:
you can control the nodes in the solution tree where the branch-and-bound solver adds cuts.
178 CHAPTER 5
If you pull down the selection list, you will find three options:
♦ Root Only,
♦ All Nodes, and
♦ Solver Decides.
Under the Root Only option, the solver appends cuts only at the first node, or root node, in the solution
tree. With the All Nodes option, cuts are appended at each node of the tree. The Solver Decides option
causes the solver to dynamically decide when it is best to append cuts at a node.
The default is to let the solver decide when to append cuts. In general, this will offer superior
performance. There may be instances, however, where one of the other two options prevails.
Relative Limit
In the Relative Limit field of the Constraint Cuts box:
you can control the number of constraint cuts that are generated by the integer pre-solver. Most integer
programming models benefit from the addition of some constraint cuts. However, at some point
additional cuts take more time to generate than they save in solution time. For this reason, LINGO
imposes a relative limit on the number of constraint cuts.
The default limit is set to .75 times the number of true constraints in the original formulation. This
relative limit may be overridden by changing it in the Relative Limit field.
Max Passes
In the Max Passes box of the Constraint Cuts box:
you can control the number of iterative passes the integer pre-solver makes through a model to
determine appropriate constraint cuts to append to the formulation. In general, the benefits of each
successive pass decline. At some point, additional passes will only add to the total solution time. Thus,
LINGO imposes a limit on the maximum number of passes.
The default limit is 200 passes at the root node of the branch-and-bound tree, and 2 passes at all
subsequent nodes. You can override these limits by changing the values in the Root and Tree fields.
WINDOWS COMMANDS 179
Types
The Types box of the Constraint Cuts box:
is used to enable or disable the different strategies LINGO uses for generating constraint cuts. LINGO
uses twelve different strategies for generating constraint cuts. The default is for all cut generation
strategies to be enabled with the exception of Basis cuts.
It is beyond the scope of this manual to go into the details of the various strategies. Interested readers
may refer to any good text on integer programming techniques. In particular, see Nemhauser and
Wolsey (1988).
180 CHAPTER 5
can be used to control several tolerances for tailoring the operation of LINGO’s branch-and-bound
solver used on integer models (i.e., models making use of the @BIN and @GIN functions to restrict
one or more variables to integer values).
WINDOWS COMMANDS 181
Branching Box
The Branching box on the Integer Solver tab contains the following two options for controlling the
branching strategy used by LINGO’s branch-and-bound solver:
♦ Direction, and
♦ Priority.
Direction
LINGO uses a branch-and-bound solution procedure when solving integer programming models. One
of the fundamental operations involved in the branch-and-bound algorithm is branching on variables.
Branching involves forcing an integer variable that is currently fractional to either the next greatest or
the next lowest integer value. As an example, suppose there is a general integer variable that currently
has a value of 5.6. If LINGO were to branch on this variable, it would have to choose whether to set
the variable first to 6 or 5. The Direction field controls how LINGO makes this branching decision.
If you pull down the drop-down box for the Direction option, you’ll find the following:
The default, Both, involves LINGO making an intelligent guess as to whether it should branch up or
down first on each individual variable. If Up is selected, LINGO will always branch up first. If Down
is selected, LINGO will always branch down first. In most cases, the Both option will result in the best
performance.
Priority
When branching on variables, the branch-and-bound procedure can give priority to branching on the
binary variables first, or it can make an intelligent guess as to the next best variable to branch on,
regardless of whether it is binary or general. The Priority field controls how LINGO makes this
branching decision.
If you pull down the drop-down box for Priority, you’ll find the following:
Select Binary to have LINGO give branching priority to the binary variables. Select LINGO Decides to
have LINGO select the next integer variable for branching based on an intelligent guess, regardless of
whether it is binary or general.
The default is LINGO Decides, which should generally give the best results.
182 CHAPTER 5
Integrality Box
Due to the potential for round-off error on digital computers, it is not always possible for LINGO to
find exact integer values for the integer variables. The Integrality box on the Integer Solver tab
contains the following three options for controlling the amount of deviation from integrality that will
be tolerated:
♦ Absolute Integrality,
♦ Relative Integrality, and
♦ BigM Threshhold.
Absolute Integrality
The Absolute Integrality tolerance is used by LINGO as a test for integrality in integer programming
models. Due to round-off errors, the “integer” variables in a solution may not have values that are
precisely integer. The absolute integrality tolerance specifies the absolute amount of violation from
integrality that is acceptable. Specifically, if X is an "integer" variable and I is the closest integer to X,
then X would be accepted as being integer valued if:
| X – I | <= Absolute Integrality Tolerance.
The default value for the absolute integrality tolerance is .000001. Although one might be tempted to
set this tolerance to 0, this may result in feasible models being reported as infeasible.
Relative Integrality
The Relative Integrality tolerance is used by LINGO as a test for integrality in integer programming
models. Due to round-off errors, the “integer” variables in a solution may not have values that are
precisely integer. The relative integrality tolerance specifies the relative amount of violation from
integrality that is acceptable. Specifically, if I is the closest integer value to X, X will be considered an
integer if:
| X – I | <= Relative Integrality Tolerance.
|X|
The default value for the relative integrality tolerance is .000008. Although one might be tempted to
set this tolerance to 0, this may result in feasible models being reported as infeasible.
BigM Threshold
Many integer programming models have constraints of the form:
f(x) ≤ M * z
where f(x) is some function of the decision variables, M is a large constant term, and z is a binary
variable. These types of constraints are called forcing constraints and are used to force the binary
variable, z, to 1 when f(x) is nonzero. In many instances, the binary variable is multiplied by a fixed
cost term in the objective; a fixed cost that is incurred when a particular activity, represented by f(x),
occurs. The large constant tem, M, Is frequently referred to as being a BigM coefficient.
WINDOWS COMMANDS 183
Setting BigM too small can lead to infeasible or suboptimal models. Therefore, the BigM value will
typically have to be rather large in order to exceed the largest activity level of f(x). When BigM is
large, the solver may discover that by setting z slightly positive (within normal integrality tolerances),
it can increase f(x) to a significant level and thereby improve the objective. Although such solutions
are technically feasible to tolerances, they are invalid in that the activity is occurring without incurring
its associated fixed cost.
The BigM threshold is designed to avoid this problem by allowing LINGO to identify the binary
variables that are being set by forcing constraints. Any binary variable with a coefficient larger than
the BigM threshold will be subject to a much tighter integrality tolerance.
The default value for the BigM Threshold is 1.e8.
LP Solver Box
In a mixed linear integer programming model, LINGO’s branch-and-bound solver solves a linear
programming model at each node of the solution tree. LINGO has a choice of using the primal
simplex, dual simplex, or barrier solver (assuming the barrier option was purchased with your license)
for handling these linear programs. The LP Solver box on the Integer Solver tab contains the following
two options for controlling this choice of linear program solver:
♦ Warm Start, and
♦ Cold Start
Warm Start
The Warm Start option controls the linear solver that is used by the branch-and-bound solver at each
node of the solution tree when a previous solution is present to use as a “warm start”. The cold start
option, discussed below, determines the solver to use when a previous solution does not exist.
If you pull down the drop-down box for Warm Start, you’ll find the following:
Optimality Box
The Optimality Box on the Integer Solver tab:
is used to control three tolerances: Absolute, Relative, and Time to Relative. These tolerances control
how close you want the solver to come to the optimal solution. Ideally, we’d always want the solver to
find the best solution to a model. Unfortunately, integer programming problems are very complex, and
the extra computation required to seek out the absolute best solution can be prohibitive. On large
integer models, the alternative of getting a solution within a few percentage points of the true optimum
after several minutes of runtime, as opposed to the true optimum after several days, makes the use of
these tolerances quite attractive.
Absolute
The Absolute Optimality tolerance is a positive value r, indicating to the branch-and-bound solver that
it should only search for integer solutions with objective values at least r units better than the best
integer solution found so far. In many integer programming models, there are huge numbers of
branches with roughly equivalent potential. This tolerance helps keep the branch-and-bound solver
from being distracted by branches that can’t offer a solution significantly better than the incumbent
solution.
WINDOWS COMMANDS 185
In general, you shouldn’t have to set this tolerance. Occasionally, particularly on poorly formulated
models, you might need to increase this tolerance slightly to improve performance. In most cases, you
should experiment with the relative optimality tolerance, discussed below, rather than the absolute
optimality tolerance in order to improve performance.
The default value for the absolute optimality tolerance is 8e-8.
Relative
The Relative Optimality tolerance is a value r, ranging from 0 to 1, indicating to the branch-and-bound
solver that it should only search for integer solutions with objective values at least 100*r% better than
the best integer solution found so far.
The end results of modifying the search procedure in this way are twofold. First, on the positive side,
solution times can be improved tremendously. Second, on the negative side, the final solution obtained
by LINGO may not be the true optimal solution. You will, however, be guaranteed the solution is
within 100*r% of the true optimum.
Typical values for the relative optimality tolerance would be in the range .01 to .05. In other words,
you would be happy to get a solution within 1% to 5% of the true optimal value. On large integer
models, the alternative of getting a solution within a few percentage points of the true optimum after
several minutes of runtime, as opposed to the true optimum after several days, makes the use of an
optimality tolerance quite attractive.
Note: Generally speaking, the relative optimality tolerance is the tolerance that will most likely
improve runtimes on integer models. You should be sure to set this tolerance whenever
possible.
The default for the relative optimality tolerance is 5e-8.
Time to Relative
If an integer programming model is relatively easy to solve, then we would like to have the solver
press on to the true optimal solution without immediately resorting to a relative optimality tolerance,
discussed above. On the other hand, if, after running for a while, it becomes apparent that the optimal
solution won’t be immediately forthcoming, then you might want the solver to switch to using a
relative optimality tolerance.
The Time to Relative tolerance can be used in this manner. This tolerance is the number of seconds
before the branch-and-bound solver begins using the relative optimality tolerance. For the first n
seconds, where n is the value of the time to relative tolerance, the branch-and-bound solver will not
use the relative optimality tolerance and will attempt to find the true optimal solution to the model.
Thereafter, the solver will use the relative optimality tolerance in its search.
The default value for the time to relative tolerance is 100 seconds.
186 CHAPTER 5
Tolerances Box
The Tolerances box on the Integer Solver tab:
contains three miscellaneous tolerances for controlling the branching strategy used by the
branch-and-bound solver on integer programming models. The three tolerances are Hurdle, Node
Selection, and Strong Branch.
Hurdle
If you know the objective value of a solution to a model, you can enter it as the Hurdle tolerance. This
value is used in the branch-and-bound solver to narrow the search for the optimum. More specifically,
LINGO will only search for integer solutions in which the objective is better than the hurdle value.
This comes into play when LINGO is searching for an initial integer solution. LINGO can ignore
branches in the search tree with objective values worse than the hurdle value, because a better solution
exists (i.e., the solution whose objective value equals the hurdle tolerance) on some alternate branch.
Depending on the problem, a good hurdle value can greatly reduce solution time. Once LINGO finds
an initial integer solution, however, the hurdle tolerance no longer has an effect. At this point, the
Relative Optimality tolerance comes into play.
Note: Be sure when entering a hurdle value that a solution exists that is at least as good or better
than your hurdle. If such a solution does not exist, LINGO will not be able to find a feasible
solution to the model.
The default hurdle value is None. In other words, the solver does not use a hurdle value.
Node Selection
The branch-and-bound solver has a great deal of freedom in deciding how to span the
branch-and-bound solution tree. The Node Selection option allows you to control the order in which
the solver selects branch nodes in the tree.
If you examine the pull down list for Node Selection, you will see the following:
WINDOWS COMMANDS 187
The four choices function as follows:
♦ LINGO Decides – This is the default option. LINGO makes an educated guess as to the
best node to branch on next.
♦ Depth First – LINGO spans the branch-and-bound tree using a depth first strategy.
♦ Worst Bound – LINGO picks the node with the worst bound.
♦ Best Bound – LINGO picks the node with the best bound.
In general, LINGO Decides will offer the best results. Experimentation with the other three choices
may be beneficial with some classes of models.
Strong Branch
The Strong Branch field uses a more intensive branching strategy during the first n levels of the
branch-and-bound tree, where n is the value in Strong Branch. During these initial levels, LINGO
picks a subset of the fractional variables as branching candidates. LINGO then performs a tentative
branch on each variable in the subset, selecting as the final candidate the variable that offers the
greatest improvement in the bound on the objective. Although strong branching is useful in tightening
the bound quickly, it does take additional computation time. Therefore, you may want to try different
settings to determine what works best for your model.
The default strong branch setting is 10 levels.
188 CHAPTER 5
can be used to control the operation of LINGO’s global solver capabilities. Please keep in mind that
the global solver toolkit is an add-on option to LINGO. You must specifically purchase the global
solver option as part of your LINGO license in order to make use of its capabilities.
LINGO exploits the convex nature of linear models to find globally optimal solutions. However, we
aren’t as fortunate with nonlinear models. LINGO’s default NLP solver uses a local search procedure.
This can lead to LINGO stopping at locally optimal points when a model is non-convex and perhaps
missing a global point lying elsewhere. You may refer to Chapter 14, On Mathematical Modeling, for
more information on how and why this can happen. The global solver toolkit contains features
designed to sift through the local points in search of the globally optimal point.
The two primary features in LINGO’s global toolkit are a global solver and a multistart solver. The
global solver uses range bounding and reduction techniques within a branch-and-bound framework to
convert a non-convex model into a series of smaller, convex models. This divide-and-conquer strategy
WINDOWS COMMANDS 189
ultimately results in convergence to the guaranteed globally optimal point. The multistart solver, on the
other hand, uses a heuristic approach of restarting the NLP solver several times from different initial
points. It is not uncommon for a different starting point to lead to a different local solution point. Thus,
if we restart from enough unique points, saving the best local solution as we go, we stand a much
better chance of finding the true global solution.
The objective function has three local minimal points over the feasible range. These points are
summarized in the following table:
Point X Objective
1 1.09 -1.05
2 3.03 -3.02
3 5.02 -5.01
Clearly, the third local point is also the globally best point, and we would like the NLP solver to
converge to this point. Below is the solution LINGO produces if the default nonlinear solver is
invoked:
Local optimal solution found at step: 11
Objective value: -1.046719
Variable Value Reduced Cost
X 1.090405 0.1181082E-07
Row Slack or Surplus Dual Price
1 -1.046719 -1.000000
WINDOWS COMMANDS 191
Unfortunately, as you can see, we converged to the least preferable of the local minimums. However,
after enabling the global solver by checking the Use Global Solver box, we do obtain the global
solution:
Global optimal solution found at step: 35
Objective value: -5.010083
Variable Value Reduced Cost
X 5.020143 -0.7076917E-08
Row Slack or Surplus Dual Price
1 -5.010083 -1.000000
Note: There is one drawback to using the global solver; it runs considerably slower than the default
nonlinear solver. Therefore, the preferred option is to always try and write smooth, convex
nonlinear models. By doing this, the faster, default local solver can be successfully invoked.
Keep in mind that the global solver supports most, but not all, of the functions available in the LINGO
language. The following is a list of the nonlinear functions not currently supported by the global
solver:
♦ @PBN()—Cumulative binomial probability
♦ @PCX()—Cumulative Chi-squared distrribution
♦ @PFD()—Cumulative F distribution
♦ @PHG()—Cumulative hypergeometric probability
♦ @PFS()—Poisson finite source
♦ @PPL()—Poisson linear loss
♦ @PTD()—Cumulative t distribution
♦ @USER()-User supplied function
Note: The global solver will not operate on models containing one or more unsupported nonlinear
operations that reference optimizable quantities; the default NLP solver will be called in this
case.
The global solver is disabled by default.
sets the default variable bounds while the global solver is running. If this parameter is set to d, then
variables will not be permitted to assume values outside the range of [-d, d]. Setting this parameter as
tightly as possible in the Value Field restricts the global solver from straying into uninteresting regions
and will reduce run times. The default value for the Value Field is 1.e10.
192 CHAPTER 5
The Application list box has three options available: None, All and Selected. Selecting None removes
the variable bounds entirely, and is not recommended. The All setting applies the bound to all
variables. Finally, the Selected setting causes the global solver to apply the bound after an initial
solver pass to find the first local solution. The bound will only be applied to a variable if it does not
cutoff the initial local solution. LINGO defaults to the Selected setting.
Tolerances Box
The Tolerances box:
contains two tolerances used by the global solver: Optimality and Delta.
The Optimality tolerance specifies by how much a new solution must beat the objective value of the
incumbent solution in order to become the new incumbent. The default value for Optimality 1.e-6.
The Delta tolerance specifies how closely the additional constraints, added as part of the global
solver’s convexification process, must be satisfied. The default value for Delta is 1.e-7.
Strategies Box
The Strategies box:
allows you to control three strategies used by the global solver: Branching, Box Selection and
Reformulation.
The Branching strategy consists of six options to use when branching on a variable for the first time:
♦ Absolute Width,
♦ Local Width,
♦ Global Width,
♦ Global Distance,
♦ Absolute Violation, and
♦ Relative Violation.
The default setting for Branching is Relative Violation.
The Box Selection option specifies the strategy to use for choosing between all active nodes in the
global solver’s branch-and-bound tree. The choices are: Depth First and Worst Bound, with the
default being Worst Bound.
WINDOWS COMMANDS 193
The Reformulation option sets the degree of algebraic reformulation performed by the global solver.
Algebraic reformulation is critical for construction of tight, convex sub-regions to enclose the
nonlinear and nonconvex functions. The available settings are None, Low, Medium and High, with
High being the default.
MultiStart Solver
LINGO exploits the convex nature of linear models to find globally optimal solutions. However, we
aren’t as fortunate with nonlinear models. With NLP models, LINGO’s default NLP solver uses a local
search procedure. This can lead to LINGO stopping at locally optimal points, perhaps missing a global
point lying elsewhere. You may refer to On Mathematical Modeling for more information on how and
why this can happen.
A strategy that has proven successful in overcoming this problem is to restart the NLP solver several
times from different initial points. It is not uncommon for a different starting point to lead to a
different local solution point. Thus, if we restart from enough unique points, saving the best local
solution as we go, then we stand a much better chance of finding the true global solution. We refer to
this solution strategy as multistart.
The Multistart Solver Attempts box on the Global Solver tab:
is used to set the number of times the multistart solver restarts the standard NLP solver in its attempt to
find successively better local solutions. Each new starting point is intelligently generated to maximize
the chances of finding a new local point.
The default option, Solver Decides, entails restarting 5 times on small NLPs and disabling multistart
on larger models. Setting multistart to 1 causes the NLP solver to be invoked only once, effectively
disabling multistart. Setting multistart to any value greater than 1 will cause the NLP solver to restart
that number of times on all NLPs. In general, we have found that setting the number of multistarts to
around 5 tends to be adequate for most models. Highly nonlinear models may require a larger setting.
Note: Keep in mind that multistart will dramatically increase runtimes, particularly if a large
number of restarts is selected. Thus, one should avoid using multistart unnecessarily on
convex models that will converge to a global point in a single pass without any additional
prodding.
194 CHAPTER 5
The following example illustrates the usefulness of multistart. Consider the simple, yet highly
nonlinear, model:
MODEL:
MIN = X * @COS( 3.1416 * X);
@BND( 0, X, 6);
END
The objective function has three local minimal points over the feasible range. These points are
summarized in the following table:
Point X Objective
1 1.09 -1.05
2 3.03 -3.02
3 5.02 -5.01
WINDOWS COMMANDS 195
Clearly, the third local point is also the globally best point, and we would like the NLP solver to
converge to this point. Below is the solution you will get from LINGO if the multistart option is
disabled:
Local optimal solution found at step: 11
Objective value: -1.046719
Variable Value Reduced Cost
X 1.090405 0.1181082E-07
Row Slack or Surplus Dual Price
1 -1.046719 -1.000000
Unfortunately, as you can see, we converged to the least preferable of the local minimums. However,
after setting the number of multistarts to five and re-solving, we do obtain the global solution:
Note: Unlike the global solver, the multistart solver can only claim its solution to be locally optimal.
This is because there may always be a better solution out there that the multistart solver may,
or may not, be able to find with additional runs. The global solver, on the other hand, can
claim global optimality by having partitioned the original model into a series of smaller,
convex models.
LINGO|Generate... Ctrl+G
Once you remove all the syntax errors from your LINGO model, there is still one very important step
required: model verification. LINGO’s set-based modeling capabilities are very powerful, and they
allow you to generate large, complex models quickly and easily. However, when you first develop a
model you will need to verify that the model being generated matches up to the model you actually
intended to generate. Many set-based models can be quite complex, and it is highly likely that logic
errors may creep into one or more expressions, thereby causing your generated model to be flawed.
The LINGO|Generate command is very useful for debugging such errors. It expands all of the model's
compact set-based expressions and then writes out the full scalar-based equivalent of the LINGO
model. The expanded model report explicitly lists all the generated constraints and variables in your
model. You will find that the Generate report can be an invaluable tool in tracking down errors.
196 CHAPTER 5
When selecting the Generate command, you will be presented with a pop-up menu prompting you for
one of the two following options:
♦ Display model, and
♦ Don’t display model.
If you choose the Display model option, LINGO will place a copy of the generated model in a new
window, which you may scroll through to examine, print, or save to disk. If you choose the Don’t
display model option, LINGO will generate the model without displaying it, but will store the
generated model for later use by the appropriate solver.
As an example of the output from the Generate command, consider the transportation model
developed in Chapter 1:
MODEL:
! A 6 Warehouse 8 Vendor Transportation Problem;
SETS:
WAREHOUSES: CAPACITY;
VENDORS: DEMAND;
LINKS( WAREHOUSES, VENDORS): COST, VOLUME;
ENDSETS
DATA:
!set members;
WAREHOUSES = WH1 WH2 WH3 WH4 WH5 WH6;
VENDORS = V1 V2 V3 V4 V5 V6 V7 V8;
!attribute values;
CAPACITY = 60 55 51 43 41 52;
DEMAND = 35 37 22 32 41 32 43 38;
COST = 6 2 6 7 4 2 5 9
4 9 5 3 8 5 8 2
5 2 1 9 7 4 3 3
7 6 7 3 9 2 7 1
2 3 9 5 7 2 6 5
5 5 2 2 8 1 4 3;
ENDDATA
! The objective;
[OBJECTIVE] MIN = @SUM( LINKS( I, J):
COST( I, J) * VOLUME( I, J));
! The demand constraints;
@FOR( VENDORS( J): [DEMAND_ROW]
@SUM( WAREHOUSES( I): VOLUME( I, J)) =
DEMAND( J));
! The capacity constraints;
@FOR( WAREHOUSES( I): [CAPACITY_ROW]
@SUM( VENDORS( J): VOLUME( I, J)) <=
CAPACITY( I));
END
Model: WIDGETS
The objective will generate one expression, there should be one demand constraint generated for each
of the eight vendors and one supply constraint generated for each of the six warehouses, for a grand
WINDOWS COMMANDS 197
total of 15 rows in the expanded model. Running the generate command to verify this reveals the
following report:
MODEL:
[OBJECTIVE] MIN= 6 * VOLUME_WH1_V1 + 2 * VOLUME_WH1_V2 + 6 *
VOLUME_WH1_V3 + 7 * VOLUME_WH1_V4 + 4 * VOLUME_WH1_V5 + 2 *
VOLUME_WH1_V6 + 5 * VOLUME_WH1_V7 + 9 * VOLUME_WH1_V8 + 4 *
VOLUME_WH2_V1 + 9 * VOLUME_WH2_V2 + 5 * VOLUME_WH2_V3 + 3 *
VOLUME_WH2_V4 + 8 * VOLUME_WH2_V5 + 5 * VOLUME_WH2_V6 + 8 *
VOLUME_WH2_V7 + 2 * VOLUME_WH2_V8 + 5 * VOLUME_WH3_V1 + 2 *
VOLUME_WH3_V2 + VOLUME_WH3_V3 + 9 * VOLUME_WH3_V4 + 7 *
VOLUME_WH3_V5 + 4 * VOLUME_WH3_V6 + 3 * VOLUME_WH3_V7 + 3 *
VOLUME_WH3_V8 + 7 * VOLUME_WH4_V1 + 6 * VOLUME_WH4_V2 + 7 *
VOLUME_WH4_V3 + 3 * VOLUME_WH4_V4 + 9 * VOLUME_WH4_V5 + 2 *
VOLUME_WH4_V6 + 7 * VOLUME_WH4_V7 + VOLUME_WH4_V8 + 2 *
VOLUME_WH5_V1 + 3 * VOLUME_WH5_V2 + 9 * VOLUME_WH5_V3 + 5 *
VOLUME_WH5_V4 + 7 * VOLUME_WH5_V5 + 2 * VOLUME_WH5_V6 + 6 *
VOLUME_WH5_V7 + 5 * VOLUME_WH5_V8 + 5 * VOLUME_WH6_V1 + 5 *
VOLUME_WH6_V2 + 2 * VOLUME_WH6_V3 + 2 * VOLUME_WH6_V4 + 8 *
VOLUME_WH6_V5 + VOLUME_WH6_V6 + 4 * VOLUME_WH6_V7 + 3 *
VOLUME_WH6_V8 ;
[DEMAND_ROW_V1] VOLUME_WH1_V1 + VOLUME_WH2_V1 +
VOLUME_WH3_V1 + VOLUME_WH4_V1 + VOLUME_WH5_V1 +
VOLUME_WH6_V1 = 35 ;
[DEMAND_ROW_V2] VOLUME_WH1_V2 + VOLUME_WH2_V2 +
VOLUME_WH3_V2 + VOLUME_WH4_V2 + VOLUME_WH5_V2 +
VOLUME_WH6_V2 = 37 ;
[DEMAND_ROW_V3] VOLUME_WH1_V3 + VOLUME_WH2_V3 +
VOLUME_WH3_V3 + VOLUME_WH4_V3 + VOLUME_WH5_V3 +
VOLUME_WH6_V3 = 22 ;
[DEMAND_ROW_V4] VOLUME_WH1_V4 + VOLUME_WH2_V4 +
VOLUME_WH3_V4 + VOLUME_WH4_V4 + VOLUME_WH5_V4 +
VOLUME_WH6_V4 = 32 ;
[DEMAND_ROW_V5] VOLUME_WH1_V5 + VOLUME_WH2_V5 +
VOLUME_WH3_V5 + VOLUME_WH4_V5 + VOLUME_WH5_V5 +
VOLUME_WH6_V5 = 41 ;
[DEMAND_ROW_V6] VOLUME_WH1_V6 + VOLUME_WH2_V6 +
VOLUME_WH3_V6 + VOLUME_WH4_V6 + VOLUME_WH5_V6 +
VOLUME_WH6_V6 = 32 ;
[DEMAND_ROW_V7] VOLUME_WH1_V7 + VOLUME_WH2_V7 +
VOLUME_WH3_V7 + VOLUME_WH4_V7 + VOLUME_WH5_V7 +
VOLUME_WH6_V7 = 43 ;
[DEMAND_ROW_V8] VOLUME_WH1_V8 + VOLUME_WH2_V8 +
VOLUME_WH3_V8 + VOLUME_WH4_V8 + VOLUME_WH5_V8 +
VOLUME_WH6_V8 = 38 ;
[CAPACITY_ROW_WH1] VOLUME_WH1_V1 + VOLUME_WH1_V2 +
VOLUME_WH1_V3 + VOLUME_WH1_V4 + VOLUME_WH1_V5 +
VOLUME_WH1_V6 + VOLUME_WH1_V7 + VOLUME_WH1_V8 <= 60 ;
[CAPACITY_ROW_WH2] VOLUME_WH2_V1 + VOLUME_WH2_V2 +
VOLUME_WH2_V3 + VOLUME_WH2_V4 + VOLUME_WH2_V5 +
VOLUME_WH2_V6 + VOLUME_WH2_V7 + VOLUME_WH2_V8 <= 55 ;
[CAPACITY_ROW_WH3] VOLUME_WH3_V1 + VOLUME_WH3_V2 +
VOLUME_WH3_V3 + VOLUME_WH3_V4 + VOLUME_WH3_V5 +
198 CHAPTER 5
VOLUME_WH3_V6 + VOLUME_WH3_V7 + VOLUME_WH3_V8 <= 51 ;
[CAPACITY_ROW_WH4] VOLUME_WH4_V1 + VOLUME_WH4_V2 +
VOLUME_WH4_V3 + VOLUME_WH4_V4 + VOLUME_WH4_V5 +
VOLUME_WH4_V6 + VOLUME_WH4_V7 + VOLUME_WH4_V8 <= 43 ;
[CAPACITY_ROW_WH5] VOLUME_WH5_V1 + VOLUME_WH5_V2 +
VOLUME_WH5_V3 + VOLUME_WH5_V4 + VOLUME_WH5_V5 +
VOLUME_WH5_V6 + VOLUME_WH5_V7 + VOLUME_WH5_V8 <= 41 ;
[CAPACITY_ROW_WH6] VOLUME_WH6_V1 + VOLUME_WH6_V2 +
VOLUME_WH6_V3 + VOLUME_WH6_V4 + VOLUME_WH6_V5 +
VOLUME_WH6_V6 + VOLUME_WH6_V7 + VOLUME_WH6_V8 <= 52 ;
END
Model: WIDfGETS
As expected, there are 15 rows in the generated model: [OBJECTIVE], [DEMAND_ROW_V1]
through [DEMAND_ROW_V8], and [CAPACITY_ROW_WH1] through
[CAPACITY_ROW_WH6].
As a side note, it’s interesting to compare the generated model to the original, set-based model. We
think most would agree that the set-based model is much easier to comprehend, thereby illustrating one
of the primary benefits of modern algebraic languages over more traditional, scalar-based languages.
In addition to verifying that the correct number of rows is being generated, you should also examine
each of the rows to determine that the correct variables are appearing in each row along with their
correct coefficients.
Note: Starting with release 9.0 of LINGO, the reports generated by the LINGO|Generate command
are valid LINGO models. You may load Generate reports into a model window and solve
them as you would any other LINGO model.
One thing to keep in mind when examining generated model reports is that the LINGO model
generator performs fixed variable reduction. This means that any variables that are fixed in value are
substituted out of the generated model. For example, consider the simple model:
MODEL:
MAX = 200 * WS + 300 * NC;
WS = 60;
NC <= 40;
WS + 2 * NC <= 120;
END
LINGO|Picture Ctrl+K
The Picture command displays a model in matrix form. Viewing the model in matrix form can be
helpful in a couple of instances. First and perhaps most importantly, is the use of nonzero pictures in
debugging formulations. Most models have strong repetitive structure. Incorrectly entered sections of
the model will stand out in a model’s matrix picture. Secondly, a nonzero picture can be helpful when
you are attempting to identify special structure in your model. As an example, if your model displays
strong block angular structure, then algorithms that decompose the model into smaller fragments might
prove fruitful.
As an example, we loaded the DNRISK.LG4 model from LINGO’s sample model set. Issuing the
Picture command, we see the following:
Positive coefficients are represented with blue tiles, negatives with red, and variables that appear in a
row nonlinearly show up as black tiles.
200 CHAPTER 5
You can zoom in on a selected range in the matrix for closer viewing. To do this, place the cursor on
the upper left corner of the range you wish to view, press and hold down the left mouse button. Next,
drag the mouse to the lower right-hand corner of the desired range. Now, release the left mouse button
and LINGO will zoom in on the selected range. As an example, here is a view of the matrix after
zooming in on a 4x4 range:
Note, we have zoomed in far enough to be able see the actual coefficient values, row names, and
variable names. Scroll bars have also appeared to allow scrolling through the matrix.
The matrix picture window supports several additional interactive features. To access these features,
place the cursor over the matrix picture and press and hold the right mouse button. This will bring up
the following menu:
LINGO|Debug
In the ideal world, all models would return an optimal solution. Unfortunately, this is not the case.
Sooner or later, you are bound to run across either an infeasible or unbounded model. This is
particularly true in the development phase of a project when the model will tend to suffer from
typographical errors.
Tracking down an error in a large model can prove to be a daunting task. The Debug command is
useful in narrowing the search for problems in both infeasible and unbounded linear programs. A small
portion of the original model is isolated as the source of the problem. This allows you to focus your
attention on a subsection of the model in search of formulation or data entry errors.
The Debug command identifies two types of sets: sufficient and necessary. Removing any sufficient
set object from the model is sufficient to fix the entire model. Not all models will have a sufficient set.
In which case, they will have a necessary set with the property that removing any object from this set
fixes the remaining objects within that set.
As an example, suppose you have an infeasible model. If the complete model would be feasible except
for a bug in a single row, that row will be listed as part of the sufficient set. If the model has a
necessary set, then, as long as all of them are present, the model will remain infeasible.
The following example illustrates. The coefficient .55 in row 4 should have been 5.5:
202 CHAPTER 5
When we attempt to solve this formulation, we get the following error:
Next, if we run the LINGO|Debug command, we are presented with the following report:
The Debug command has correctly identified that the erroneous ROW4, when eliminated, is sufficient
to make the entire model feasible.
Debug operates in a similar manner for unbounded models. In the following example, we introduced
an error by placing a minus sign instead of a plus sign in front of variable Z3 in ROW3. A look at
ROW3 reveals that Z3 can be increased indefinitely, leading to an unbounded objective.
WINDOWS COMMANDS 203
The resulting model is unbounded and, when issuing the LINGO|Solve command, we receive the
unbounded error message:
The Debug command has successfully determined that bounding Z3 is sufficient to bound the entire
model.
Typically, the Debug command helps to substantially reduce the search effort. The first version of this
feature was implemented in response to a user who had an infeasible model. The user had spent a day
searching for a bug in a model with 400 constraints. The debug feature quickly found a necessary set
with 55 constraints, as well as one sufficient set constraint. The user immediately noticed that the
right-hand side of the sufficient set constraint was incorrect.
Note: Prior to release 10.0 of LINGO, the debugger was only capable of processing linear models.
Starting with release 10.0, all classes of models (LP, QP, IP and NLP) may now be debugged.
LINGO|Model Statistics
The Model Statistics command lists summary statistics for your model. The statistics vary slightly
depending on whether the model you’re working with is linear or nonlinear.
204 CHAPTER 5
In the following example, we open the linear transportation model, TRAN.LG4, issue the Model
Statistics command, and then discuss some of the details of the report. Here is the output generated by
Model Statistics for TRAN.LG4:
LINGO|Look... Ctrl+L
Use the Look command to generate a report containing your model’s formulation. The Look
command’s dialog box, pictured below, lets you choose All or Selected rows for viewing from the
Rows to View:
When you choose Selected rows, the Beginning Row and Ending Row text boxes are available for entry
in the Selected Rows box. You must enter the indices of the range of rows you wish displayed. LINGO
will display the requested lines with line numbers in a new window.
4. Window Menu
The Window menu, pictured at left,
contains commands that generally
pertain to managing open windows.
206 CHAPTER 5
In general, you will probably prefer to use the pull down menus and toolbar when using LINGO
interactively. The command window interface is primarily provided for users wishing to interactively
test command scripts.
208 CHAPTER 5
This window allows you to monitor the progress of the solver. You can close the status window at any
time. If you close the status window, it may be reopened with the Window|Status Window command.
If you would like to prevent LINGO from opening a status window, see the LINGO|Options command
above. For more information on the interpretation and use of the status window, see page 8.
Window|Tile Ctrl+4
The Window|Tile command arranges all the open windows in a tiled pattern. Each window is resized,
so all windows appear on the screen and are of roughly the same size.
When you issue the Window|Tile command, you will see the dialog box:
You have the choice of tiling the windows horizontally or vertically. If you tile Horizontally (or
Vertically), LINGO will maximize the horizontal (or vertical) dimension of each window.
If there are more than three open windows, LINGO will tile the windows, but the choice of horizontal
or vertical will no longer make a difference.
Window|Cascade Ctrl+5
The Window|Cascade command arranges all open windows in a cascade pattern starting in the upper
left corner of the mainframe window. The currently active window remains on top.
5. Help Menu
The Help menu, pictured at left, contains
commands that generally pertain to
LINGO’s Help system, copyright notice,
and version specific information.
210 CHAPTER 5
Help|Help Topics
A portion of the dialog box displayed by the Help Topics command is displayed below:
Select the Contents tab to display a table of contents for the Help system. You can select any of the
topics that are of interest by double clicking on them.
Select the Index tab to display an index of topics for the Help system. Select an item for viewing by
double clicking on it.
Go to the Find tab to search the Help system for a particular item.
WINDOWS COMMANDS 211
Help|Register
Use the Help|Register command to register your version of LINGO online. You will need a
connection to the Internet open for this command to work. When you issue the Register command,
you will be presented with the following dialog box:
Enter your personal information and select the Register button. Your information will be sent directly
to LINDO Systems via the Internet.
212 CHAPTER 5
Once your registration is complete, the following dialog box will appear on your screen:
Help|AutoUpdate
Turn the Help|AutoUpdate command on to have LINGO automatically check every time you start the
LINGO software whether there is a more recent version of LINGO available for download on the
LINDO Systems website. You will need a connection to the internet open for this command to work.
When you issue the AutoUpdate command or start a version of LINGO with AutoUpdate enabled,
LINGO will search the Internet to see if an updated version of the LINGO software is available for
download. If you currently have the most recent version, then you will be returned to the main LINGO
environment. If you have an outdated version of the software, you will be presented with the following
dialog box:
at which point, you may wish to go to the LINDO Systems Web site, www.lindo.com, to download the
latest build of the software.
If you want to disable the AutoUpdate feature, then select the Disable AutoUpdate button from the
AutoUpdate dialog box
The AutoUpdate feature is disabled by default.
WINDOWS COMMANDS 213
Help|About LINGO
When you issue the About LINGO command, you will be presented with a dialog box resembling the
following:
214 CHAPTER 5
The first box lists size and release information about your copy of LINGO.
The second box tells you where you can get in touch with LINDO Systems.
The third box, titled Limits for this Installation, lists various capacity limits of your version and the
current number of bytes allocated to LINGO's model generator. The maximum sized problem your
LINGO software can handle depends on the version you have. The current limits for the various
versions are:
Total Integer Nonlinear Global
Version Variables Variables Variables Variables
Demo/Web 300 30 30 5
Solver Suite 500 50 50 5
Super 2,000 200 200 10
Hyper 8,000 800 800 20
Industrial 32,000 3,200 3,200 50
Extended Unlimited Unlimited Unlimited Unlimited
For more information on the definitions of these limits see section Maximum Problem Dimensions. In
addition to the maximum problem limits, this box also lists the amount of memory allocated to
LINGO’s model generator. You can adjust the size of generator memory allocation on the General
Solver tab of the LINGO|Options dialog box.
The fourth box titled License Expiration lists the date at which your license expires. If your license
does not have an expiration date, this field will display Perpetual.
The box labeled License Usage lists whether your license is for commercial or educational use.
Educational licenses are restricted to use by faculty, staff, and students of educational institutions for
instructional or research purposes. Commercial licenses are not restricted to any particular use.
The box titled Licenses lists the number of users licensed to use your copy of LINGO.
The API Version box lists the version number of the LINDO API in use by your copy of LINGO. The
LINDO API is the library of solver tools used by LINGO to optimize your models.
The License Location box displays the location of the license file in effect for the current LINGO
session.
The final box, Additional License Information, contains information relevant to your particular license.
In most cases, your LINGO serial number can be found in this field. Scrolling through this field, you
will also find information as to the optional features included with your license (e.g., the barrier,
nonlinear and global solvers.)
Help|Pointer
Press this button to switch the cursor into Help mode. Once the cursor is in Help mode, you can select
a menu command or a toolbar button and LINGO will display help information on the selected item.
6 Command-Line
Commands
This chapter discusses all of the command-line commands available to the LINGO user. On platforms
other than Windows based PC’s, the user interfaces with LINGO entirely through text commands
issued to LINGO’s command-line colon prompt.
If you are using a Windows version of LINGO, you will primarily be interested in the previous
chapter, Windows Commands, which details the commands available in the pull down menus of
LINGO’s Windows version. However, in Windows versions, LINGO command-line commands may
be entered using the command window (see the Window|Command Window section in Chapter 5,
Windows Commands) and may also be used to build command scripts. Command scripts may be run
automatically at startup or whenever the user desires. Command scripts are useful to both the Windows
user and users on other platforms. Thus, the Windows user may find this chapter of interest, too.
We will begin by briefly listing all the command-line commands according to their general function.
This will be followed up by an in-depth explanation of the commands.
2. Input
FRMPS retrieves a model in free MPS format
MODEL begins input of a new model
RMPS retrieves a model in fixed MPS format
TAKE runs a command script from an external file
215
216 CHAPTER 6
3. Display
GEN generates the algebraic formulation for the model
HIDE password protects the current model
LOOK displays the current model
PICTURE displays a picture of the model’s nonzero structure
STATS gives summary statistics about the properties of a generated model
4. File Output
DIVERT opens a file for receiving output
RVRT closes a file previously opened with DIVERT
SAVE saves the current model to disk
SMPI exports a model in MPI format
SMPS sends a copy of the current model to a file in MPS format
5. Solution
DEBUG tracks down formulation errors in infeasible and unbounded
models
GO solves the current model
NONZ generates a nonzeros only solution report
RANGE generates a range analysis report
SOLU generates a solution report
6. Problem Editing
ALTER edits the contents of the model
DELETE deletes a selected row from the model
EXTEND adds rows to the end of the current model
7. Conversational Parameters
PAGE sets the page/screen length
PAUSE pauses for keyboard input
TERSE output level
VERBOSE switches to verbose output mode
WIDTH sets terminal display and input width
COMMAND-LINE COMMANDS 217
8. Tolerances
APISET allows access to advanced parameters in the LINDO API, which is
the solver library used by LINGO
DBPWD sets the password for database access via @ODBC
DBUID sets your user id for database access via @ODBC
FREEZE saves current tolerance settings to disk
SET overrides a number of LINGO defaults and tolerances
9. Miscellaneous
! inserts a comment
QUIT exits LINGO
TIME displays current elapsed time since start of session
Note: User input in the examples below is indicated through the use of bold typeface.
1. Information
The Information category contains commands related to on-line information.
CAT
The CAT command displays the nine categories of commands available in LINGO. You will be
prompted to input a number corresponding to one of the categories. If you input a number, LINGO
will display the commands available under the corresponding category. To exit out of the command,
input a blank line.
COM
The COM command lists all the command-line commands available in LINGO by category.
HELP
The HELP command combined with another LINGO command gives you information on the
command specified. The information is usually quite brief, but is often all that is needed.
The HELP command without an argument will give you general information about your version of
LINGO, along with the maximum number of constraints and variables that your version of LINGO can
handle.
218 CHAPTER 6
MEM
The MEM command displays statistics about the model generator's memory usage. The following is
some sample output from the MEM command:
: MEM
Total generator memory 5242880
Peak generator memory usage 12048
Current generator memory usage 1312
Total handles 96
Peak handle usage 9
Current handle usage 5
Total bytes moved 1552
Total blocks moved 6
Total heap compacts 0
Fragmentation ratio 0.002
:
The Total generator memory figure is the amount of memory LINGO has allocated for a working
memory heap for model generation. You can control the size of the heap using the SET command.
Peak generator memory usage refers to the maximum amount of memory the model generator used
during the current session. Current memory usage lists the amount of working memory currently in use
by the model generator.
Total handles is the maximum number of memory blocks LINGO can allocate. Peak handle usage lists
the maximum number of memory blocks LINGO allocated at any one time during this session. Current
handle usage represents the number of memory blocks currently in use by the model generator.
Total bytes moved lists the number of memory bytes the generator has had to move so far in order to
reallocate memory. Total blocks moved lists the number of memory blocks moved due to reallocation.
Total heap compacts lists the number of times the generator has had to compact the heap to make room
for growing memory needs. If the number of heap compacts is abnormally large, you should allocate
more working memory using the SET command.
The Fragmentation ratio is a statistic measuring how fragmented the memory heap is. A value of 1
would indicate high fragmentation, whereas a value of 0 indicates no fragmentation
2. Input
The Input category contains commands that initiate input into LINGO
FRMPS / RMPS
The FRMPS and RMPS commands are used to read MPS formatted models. The MPS file format is an
industry standard format developed by IBM and is useful for passing models from one solver or
platform to another.
FRMPS and RMPS don’t presently support quadratic MPS files, so the models read by these
commands must be either linear or mixed integer linear. FRMPS reads an MPS file in free format,
while RMPS reads fixed format MPS files.
COMMAND-LINE COMMANDS 219
When LINGO reads an MPS file, it converts the formulation to an equivalent LINGO model. As an
example, consider the following, simple model:
ObjRow) Maximize 20X + 30Y
Subject To:
Row1) X < 50
Row2) Y < 60
Row3) X + 2Y < 120
As an aside, one thing to notice about the MPS representation is that it is not a very compact method
for storing a model.
In the following session, we read this MPS file into LINGO and then display the model with the
LOOK command. Note how the model is automatically converted from MPS format to LINGO format:
: rmps c:\sample.mps
: look all
1] TITLE SAMPLE;
2] [ OBJROW] MAX = 20 * X + 30 * Y;
3] [ ROW1] X <= 50;
4] [ ROW2] Y <= 60;
5] [ ROW3] X + 2 * Y <= 120;
:
Should you wish to save the file again using MPS format rather than LINGO format, you may use the
SMPS command (shown in the File Output section below).
220 CHAPTER 6
When it comes to acceptable constraint and variable names, MPS format is less restrictive than
LINGO. MPS allows for embedded blanks and other additional characters in names. To compensate
for this fact, LINGO attempts to patch names when reading an MPS file, so all the incoming names are
compatible with its syntax. LINGO does this by substituting an underscore for any character in a name
that is not admissible. In most cases, this will work out OK. However, there is a chance for name
collisions where two or more names get mapped into one. For instance, the variable names X.1 and
X%1 would both get mapped into the single LINGO name X_1. Of course, situations such as this
entirely alter the structure of the model rendering it incorrect.
You will be warned whenever LINGO has to patch a name with the following error message:
[Error Code: 179]
This message displays the number of variable and row names that were patched to get them to conform
to LINGO syntax.
If name collisions are a problem, then LINGO has an option that will ensure all names remain unique.
This option involves using RC format for names encountered during MPS I/O. RC format involves
renaming each row (constraint) in a model to be Rn, where n is the row’s index. Similarly, each
column (variable) is renamed to Cn. In addition, LINGO renames the objective row to be ROBJ. To
switch to RC format for MPS names, you will need to use the SET command as follows:
: SET RCMPSN 1
This will cause LINGO to use RC naming conventions for all MPS reads and saves. To cancel the use
of RC names, type:
: SET RCMPSN 0
As an example, we will once again read the same MPS format model we read above, but this time we
will switch to RC naming conventions:
: set rcmpsn 1
Parameter Old Value New Value
RCMPSN 0 1
: rmps c:\sample.mps
: look all
1] TITLE SAMPLE;
2] [ ROBJ] MAX = 20 * C1 + 30 * C2;
3] [ R1] C1 <= 50;
4] [ R2] C2 <= 60;
5] [ R3] C1 + 2 * C2 <= 120;
Notice how the variable names now use RC format, guaranteeing that name collisions will not occur.
COMMAND-LINE COMMANDS 221
Another potential conflict is that MPS allows variable names to be duplicated as constraint names and
vice versa. LINGO does not allow for this. When you go to solve the model, you will either receive
error message 28 (Invalid use of a row name), or error message 37 (Name already in use). However,
once again, you can switch to using RC format for names to avoid this conflict.
MODEL
Use the MODEL command to begin inputting a new model into LINGO. LINGO prompts for each
new line of the model with a question mark. When you are through entering the model, enter END on a
single line by itself. LINGO will then return to normal command mode (indicated by the colon
prompt).
In the following example, we enter a small model with the MODEL command, display it with the
LOOK command, and then solve it with the GO command:
: MODEL
? !How many years does it take
? to double an investment growing
? 10% per year?;
? 1.1 ^ YEARS = 2;
? END
: LOOK ALL
1]!How many years does it take
2]to double an investment growing
3]10% per year?;
4]1.1 ^ YEARS = 2;
: GO
Feasible solution found at step: 0
Variable Value
YEARS 7.272541
Row Slack or Surplus
1 0.000000
:
TAKE
The TAKE command is used to 1) read models saved to disk using the SAVE command, and 2) execute
command scripts contained in external files. The syntax for the TAKE command is:
TAKE [filename]
If you omit a filename, LINGO will prompt you for one.
As an example, suppose you used the SAVE command to save a model to the file
C:\LINGOMOD\MYMODEL.LNG. You can read it back into LINGO by giving the command:
: TAKE C:\LINGOMOD\MYMODEL.LNG
222 CHAPTER 6
As a second example, we will use the TAKE command to execute a LINGO command script. A
command script is simply a text file that contains a series of LINGO commands. Suppose we have
built the following command script in an editor and have saved it in the text file
D:\LNG\MYSCRIPT.LTF:
MODEL:
!For a given probability P, this
model returns the value X such
that the probability that a unit
normal random variable is less
than or equal to X is P;
! Here is the probability;
P = .95;
! Solve for X;
P = @PSN(X);
END
!Terse output mode;
TERSE
!Solve the model;
GO
!Report X;
SOLU X
3. Display
This category contains commands that display information.
GEN
Once you remove all the syntax errors from your LINGO model, there is still one very important step
required: model verification. LINGO’s set-based modeling capabilities are very powerful, and they
allow you to generate large, complex models quickly and easily. However, when you first develop a
model you will need to verify that the model being generated matches up to the model you actually
intended to generate. Many set-based models can be quite complex, and it is highly likely that logic
errors may creep into one or more expressions, thereby causing your generated model to be flawed.
The GEN (short for generate) command is very useful for debugging such errors. It expands all of the
model's compact set-based expressions and then writes out the full scalar-based equivalent of the
LINGO model. The expanded model report explicitly lists all the generated constraints and variables in
your model. You will find that the Generate report can be an invaluable tool in tracking down errors.
COMMAND-LINE COMMANDS 223
As an example of the output from the generate command, consider the transportation model developed
in Chapter 1:
MODEL:
! A 6 Warehouse 8 Vendor Transportation Problem;
SETS:
WAREHOUSES: CAPACITY;
VENDORS: DEMAND;
LINKS( WAREHOUSES, VENDORS): COST, VOLUME;
ENDSETS
DATA:
!set members;
WAREHOUSES = WH1 WH2 WH3 WH4 WH5 WH6;
VENDORS = V1 V2 V3 V4 V5 V6 V7 V8;
!attribute values;
CAPACITY = 60 55 51 43 41 52;
DEMAND = 35 37 22 32 41 32 43 38;
COST = 6 2 6 7 4 2 5 9
4 9 5 3 8 5 8 2
5 2 1 9 7 4 3 3
7 6 7 3 9 2 7 1
2 3 9 5 7 2 6 5
5 5 2 2 8 1 4 3;
ENDDATA
! The objective;
[OBJECTIVE] MIN = @SUM( LINKS( I, J):
COST( I, J) * VOLUME( I, J));
! The demand constraints;
@FOR( VENDORS( J): [DEMAND_ROW]
@SUM( WAREHOUSES( I): VOLUME( I, J)) =
DEMAND( J));
! The capacity constraints;
@FOR( WAREHOUSES( I): [CAPACITY_ROW]
@SUM( VENDORS( J): VOLUME( I, J)) <=
CAPACITY( I));
END
Model: WIDGETS
224 CHAPTER 6
The objective will generate one expression; there should be one demand constraint generated for each
of the eight vendors and one supply constraint generated for each of the six warehouses, for a grand
total of 15 rows in the expanded model. Running the generate command to verify this reveals the
following report:
MODEL:
[OBJECTIVE] MIN= 6 * VOLUME_WH1_V1 + 2 * VOLUME_WH1_V2 + 6 *
VOLUME_WH1_V3 + 7 * VOLUME_WH1_V4 + 4 * VOLUME_WH1_V5 + 2 *
VOLUME_WH1_V6 + 5 * VOLUME_WH1_V7 + 9 * VOLUME_WH1_V8 + 4 *
VOLUME_WH2_V1 + 9 * VOLUME_WH2_V2 + 5 * VOLUME_WH2_V3 + 3 *
VOLUME_WH2_V4 + 8 * VOLUME_WH2_V5 + 5 * VOLUME_WH2_V6 + 8 *
VOLUME_WH2_V7 + 2 * VOLUME_WH2_V8 + 5 * VOLUME_WH3_V1 + 2 *
VOLUME_WH3_V2 + VOLUME_WH3_V3 + 9 * VOLUME_WH3_V4 + 7 *
VOLUME_WH3_V5 + 4 * VOLUME_WH3_V6 + 3 * VOLUME_WH3_V7 + 3 *
VOLUME_WH3_V8 + 7 * VOLUME_WH4_V1 + 6 * VOLUME_WH4_V2 + 7 *
VOLUME_WH4_V3 + 3 * VOLUME_WH4_V4 + 9 * VOLUME_WH4_V5 + 2 *
VOLUME_WH4_V6 + 7 * VOLUME_WH4_V7 + VOLUME_WH4_V8 + 2 *
VOLUME_WH5_V1 + 3 * VOLUME_WH5_V2 + 9 * VOLUME_WH5_V3 + 5 *
VOLUME_WH5_V4 + 7 * VOLUME_WH5_V5 + 2 * VOLUME_WH5_V6 + 6 *
VOLUME_WH5_V7 + 5 * VOLUME_WH5_V8 + 5 * VOLUME_WH6_V1 + 5 *
VOLUME_WH6_V2 + 2 * VOLUME_WH6_V3 + 2 * VOLUME_WH6_V4 + 8 *
VOLUME_WH6_V5 + VOLUME_WH6_V6 + 4 * VOLUME_WH6_V7 + 3 *
VOLUME_WH6_V8 ;
[DEMAND_ROW_V1] VOLUME_WH1_V1 + VOLUME_WH2_V1 +
VOLUME_WH3_V1 + VOLUME_WH4_V1 + VOLUME_WH5_V1 +
VOLUME_WH6_V1 = 35 ;
[DEMAND_ROW_V2] VOLUME_WH1_V2 + VOLUME_WH2_V2 +
VOLUME_WH3_V2 + VOLUME_WH4_V2 + VOLUME_WH5_V2 +
VOLUME_WH6_V2 = 37 ;
[DEMAND_ROW_V3] VOLUME_WH1_V3 + VOLUME_WH2_V3 +
VOLUME_WH3_V3 + VOLUME_WH4_V3 + VOLUME_WH5_V3 +
VOLUME_WH6_V3 = 22 ;
[DEMAND_ROW_V4] VOLUME_WH1_V4 + VOLUME_WH2_V4 +
VOLUME_WH3_V4 + VOLUME_WH4_V4 + VOLUME_WH5_V4 +
VOLUME_WH6_V4 = 32 ;
[DEMAND_ROW_V5] VOLUME_WH1_V5 + VOLUME_WH2_V5 +
VOLUME_WH3_V5 + VOLUME_WH4_V5 + VOLUME_WH5_V5 +
VOLUME_WH6_V5 = 41 ;
[DEMAND_ROW_V6] VOLUME_WH1_V6 + VOLUME_WH2_V6 +
VOLUME_WH3_V6 + VOLUME_WH4_V6 + VOLUME_WH5_V6 +
VOLUME_WH6_V6 = 32 ;
[DEMAND_ROW_V7] VOLUME_WH1_V7 + VOLUME_WH2_V7 +
VOLUME_WH3_V7 + VOLUME_WH4_V7 + VOLUME_WH5_V7 +
VOLUME_WH6_V7 = 43 ;
[DEMAND_ROW_V8] VOLUME_WH1_V8 + VOLUME_WH2_V8 +
VOLUME_WH3_V8 + VOLUME_WH4_V8 + VOLUME_WH5_V8 +
VOLUME_WH6_V8 = 38 ;
[CAPACITY_ROW_WH1] VOLUME_WH1_V1 + VOLUME_WH1_V2 +
VOLUME_WH1_V3 + VOLUME_WH1_V4 + VOLUME_WH1_V5 +
VOLUME_WH1_V6 + VOLUME_WH1_V7 + VOLUME_WH1_V8 <= 60 ;
[CAPACITY_ROW_WH2] VOLUME_WH2_V1 + VOLUME_WH2_V2 +
VOLUME_WH2_V3 + VOLUME_WH2_V4 + VOLUME_WH2_V5 +
COMMAND-LINE COMMANDS 225
VOLUME_WH2_V6 + VOLUME_WH2_V7 + VOLUME_WH2_V8 <= 55 ;
[CAPACITY_ROW_WH3] VOLUME_WH3_V1 + VOLUME_WH3_V2 +
VOLUME_WH3_V3 + VOLUME_WH3_V4 + VOLUME_WH3_V5 +
VOLUME_WH3_V6 + VOLUME_WH3_V7 + VOLUME_WH3_V8 <= 51 ;
[CAPACITY_ROW_WH4] VOLUME_WH4_V1 + VOLUME_WH4_V2 +
VOLUME_WH4_V3 + VOLUME_WH4_V4 + VOLUME_WH4_V5 +
VOLUME_WH4_V6 + VOLUME_WH4_V7 + VOLUME_WH4_V8 <= 43 ;
[CAPACITY_ROW_WH5] VOLUME_WH5_V1 + VOLUME_WH5_V2 +
VOLUME_WH5_V3 + VOLUME_WH5_V4 + VOLUME_WH5_V5 +
VOLUME_WH5_V6 + VOLUME_WH5_V7 + VOLUME_WH5_V8 <= 41 ;
[CAPACITY_ROW_WH6] VOLUME_WH6_V1 + VOLUME_WH6_V2 +
VOLUME_WH6_V3 + VOLUME_WH6_V4 + VOLUME_WH6_V5 +
VOLUME_WH6_V6 + VOLUME_WH6_V7 + VOLUME_WH6_V8 <= 52 ;
END
Model: WIDGETS
As expected, there are 15 rows in the generated model: [OBJECTIVE], [DEMAND_ROW_V1]
through [DEMAND_ROW_V8], and [CAPACITY_ROW_WH1] through
[CAPACITY_ROW_WH6].
As a side note, it’s interesting to compare the generated model to the original, set-based model. We
think most would agree that the set-based model is much easier to comprehend, thereby illustrating one
of the primary benefits of modern algebraic languages over more traditional, scalar-based languages.
In addition to verifying that the correct number of rows is being generated, you should also examine
each of the rows to determine that the correct variables are appearing in each row along with their
correct coefficients.
Note: The reports generated by the GEN command are valid LINGO models. You may load
Generate reports into LINGO and solve them as you would any other model.
One thing to keep in mind when examining generated model reports is that the LINGO model
generator performs fixed variable reduction. This means that any variables that are fixed in value are
substituted out of the generated model. For example, consider the simple model:
MODEL:
MAX = 200 * WS + 300 * NC;
WS = 60;
NC <= 40;
WS + 2 * NC <= 120;
END
At first glance, it seems as if both the first constraint and the variable WS are missing from the
generated model. Note that by the first constraint in the original model (WS = 60), WS is fixed at a
226 CHAPTER 6
value of 60. The LINGO model generator exploits this fact to reduce the size of the generated model
by substituting WS out of the formulation. The final solution report will still contain the values for all
the fixed variables; however, the fixed variables will not appear in the generated model report. If you
would like to suppress fixed variable reduction so that all variables appear in your generated model,
you may do so via the Fixed Var Reduction option.
Note: To capture the results of the GEN command in a file, use the DIVERT command to open an
output file before issuing the GEN command.
HIDE
The HIDE command hides the text of a model from viewing by the user. This may be useful if you are
trying to protect proprietary ideas contained in your model.
When you enter the HIDE command, you’ll be prompted for a password. You may enter any password
with up to eight characters. LINGO will prompt you for this password once more for verification.
LINGO is sensitive to the case of the alphabetic characters in the password.
Once a model is hidden, commands allowing the user to view the model text (GEN, GENL, LOOK,
SMPS) are disabled. All other commands, however, will function as normal with the exception of
ALTER. If a model is hidden, ALTER will perform modifications, but they will not be echoed to the
screen.
When a hidden model is saved to disk, its text will be encrypted. This prevents the user from viewing
the model from outside of LINGO as well. You will want to distribute the encrypted version of the
model to those using your application. However, you should always keep an unhidden version of the
model at your site for safekeeping in the event you forget the password.
A hidden model may be returned to the normal unhidden state by once again issuing the HIDE
command with the correct password.
COMMAND-LINE COMMANDS 227
A sample session illustrating the use of the HIDE command follows:
: TAKE TRAN.LNG !Read in a model
: LOOK 4 6 !Display some rows
4] SUPPLY / WH1, WH2, WH3/ : CAP;
5] DEST / C1, C2, C3, C4/ : DEM;
6] LINKS(SUPPLY, DEST) : COST, VOL;
: HIDE !Now hide the model
Password?
TIGER
Please reenter password to verify:
TIGER
Model is now hidden.
: ! Model is hidden so LOOK will fail
: LOOK ALL
[Error Code: 111]
Command not available when model is hidden.
: ! We can still solve it though
: TERSE
: GO
Global optimal solution found at step: 6
Objective value: 161.0000
: !And get a solution report
: NONZ VOL
Variable Value Reduced Cost
VOL(WH1, C1) 2.000000 0.000000
VOL(WH1, C2) 17.00000 0.000000
VOL(WH1, C3) 1.000000 0.000000
VOL(WH2, C1) 13.00000 0.000000
VOL(WH2, C4) 12.00000 0.000000
VOL(WH3, C3) 21.00000 0.000000
: !Now, unhide the model
: HIDE
Password?
TIGER
Model is no longer hidden.
: !Once again, we can view the model
: LOOK 4 6
4] SUPPLY / WH1, WH2, WH3/ : CAP;
5] DEST / C1, C2, C3, C4/ : DEM;
6] LINKS(SUPPLY, DEST) : COST, VOL;
:
LOOK
The LOOK command displays all or part of the current model. The syntax of the LOOK command is:
LOOK row_index|beg_row_index end_row_index|ALL
Thus, you can specify the index of a single row to view, a range of rows, or ALL to view the entire
model.
228 CHAPTER 6
In this next example, we use several forms of the LOOK command to view the current model:
: LOOK ALL
1]!For a given probability P, this
2] model returns the value X such
3] that the probability that a unit
4] normal random variable is less
5] than or equal to X is P;
6]
7]! Here is the probability;
8] P = .95;
9]
10]! Solve for X;
11]P = @PSN(X);
12]
: LOOK 8
8] P = .95;
: LOOK 10 11
10]! Solve for X;
11]P = @PSN(X);
:
COMMAND-LINE COMMANDS 229
PICTURE
The PICTURE command displays the model in matrix form. For small to medium sized models, the
PICTURE command is a useful way to obtain a visual impression of the model and to hunt for
formulation errors.
The following letter codes are used to represent the linear coefficients in the PICTURE output:
Letter Code Coefficient Range
Z (.000000, .000001)
Y (.000001, .00001)
X (.00001, .0001)
W (.0001, .001)
V (.001, .01)
U (.01, .1)
T (.1, 1 )
A (1, 10)
B (10, 100)
C (100, 1000)
D (1000, 10000)
E (10000, 100000)
F (100000, 1000000)
G > 1000000
Single digit integers are shown explicitly rather than being displayed as a code. This is especially
handy, because many models have a large number of coefficients of positive or negative 1, which can
affect the solution procedure. If a variable appears nonlinearly in a row, then the PICTURE command
will represent its coefficient with a question mark.
230 CHAPTER 6
In this example, we read in a copy of the small transportation model supplied with LINGO and use the
PICTURE command to view the logical structure of the model:
: take \lingo\samples\tran.lng
: pic
V V V V V V V V V V V V
O O O O O O O O O O O O
L L L L L L L L L L L L
U U U U U U U U U U U U
M M M M M M M M M M M M
E E E E E E E E E E E E
( ( ( ( ( ( ( ( ( ( ( (
W W W W W W W W W W W W
H H H H H H H H H H H H
1 1 1 1 2 2 2 2 3 3 3 3
, , , , , , , , , , , ,
C C C C C C C C C C C C
1 2 3 4 1 2 3 4 1 2 3 4
) ) ) ) ) ) ) ) ) ) ) )
OBJ: 6 2 6 7 4 9 5 3 8 8 1 5 MIN
DEM(C1): 1 ' 1 ' 1 ' > B
DEM(C2): ' 1' ' '1 ' ' 1 ' > B
DEM(C3): ' 1 ' 1 ' 1 > B
DEM(C4): ' 1 ' 1 ' 1 > B
SUP(WH1): 1 1'1 1 ' ' ' ' ' < B
SUP(WH2): ' ' 1 1 1 1 ' < B
SUP(WH3): ' ' ' 1 1 1 1 < B
In this model, all the right-hand side values are in the range [12, 30]. Thus, they are all represented
using the letter B. Row names are displayed running down the left-hand side of the matrix, while
variable names are displayed along the top. The sense of the objective row and each of the constraints
are shown. Spaces stand in for zero coefficients, and single quote marks are inserted to give a grid-like
background.
Note: The PICTURE command is best used on small models. The amount of output generated for
large models can be cumbersome. For larger models, the LINGO|Picture command in
Windows versions of LINGO can compress the matrix picture of large models into a single
screen for easier viewing.
STATS
The STATS command lists summary statistics for your model. The statistics vary slightly depending on
whether the model you’re working with is linear or nonlinear. In this next example, we will read in a
linear transportation model, run the STATS command, and explain some of the details of the report.
: take \lingo\samples\tran.lng
: stats
Rows= 8 Vars= 12 No. integer vars= 0 (all are linear)
Nonzeros= 43 Constraint nonz= 24( 24 are +- 1) Density=0.413
Smallest and largest elements in abs value= 1.00000 30.0000
No. < : 3 No. =: 0 No. > : 4, Obj=MIN, GUBs <= 4
Single cols= 0
Line two of the report gives a count of the number of nonzero coefficients appearing in the model. The
first count is the number of nonzero coefficients in the entire model. The Constraint nonz count is the
number of coefficients on the left-hand sides of all the constraints, excluding the nonzero objective and
right-hand side coefficients. Next, STATS gives a count of the number of constraint coefficients that
are plus or minus one. In general, a linear programming model is easier to solve when the number of
unity coefficients increases. Finally, STATS reports a Density figure, defined as:
The nonlinear STATS report drops information about the range of coefficient values, the number of
+/-1 coefficients, and the GUB upper bound. A count of the number of nonlinear rows and variables is
added in line two. The nonlinear rows count includes the objective, while the nonlinear constraint
count does not.
232 CHAPTER 6
4. File Output
The File Output category contains commands that output model and session information to a file.
DIVERT
The DIVERT command opens a file and causes LINGO to route all subsequent reports (e.g.,
SOLUTION, RANGE, and LOOK commands) from the screen to the file. This command captures the
reports in text format in the file you specify. Since the files created by the DIVERT command are in
text format, they may be read into other programs, such as word processors and spreadsheets, or they
may be queued to your printer.
The syntax for the DIVERT command is:
DIVERT filename
where filename is the name of the file you wish to create.
The RVRT command reverses a DIVERT command by closing the DIVERT file and then rerouting
output back to the screen.
In the following example, we create a small model with the MODEL command, solve it with the GO
command, and then use the DIVERT command to create a file containing the formulation and solution:
: !Enter a small model
: MODEL
? MAX = 20*X + 30*Y;
? X <= 50;
? Y <= 60;
? X + 2*Y <= 120;
? END
: !Solve the model
: TERSE
: GO
Global optimal solution found at step: 1
Objective value: 2050.000
: !Create a DIVERT file with
: !the formulation & solution
: DIVERT MYFILE.TXT !Opens the file
: LOOK ALL !Sends model to file
: SOLU !Sends solution to file
: RVRT !Closes DIVERT file
:
COMMAND-LINE COMMANDS 233
Opening the DIVERT file created in this example, we find the following file with the formulation and
solution:
1]MAX = 20*X + 30*Y;
2]X <= 50;
3]Y <= 60;
4]X + 2*Y <= 120;
Variable Value Reduced Cost
X 50.00000 0.000000
Y 35.00000 0.000000
Row Slack or Surplus Dual Price
1 2050.000 1.000000
2 0.000000 5.000000
3 25.00000 0.000000
4 0.000000 15.00000
Note 1: Keep in mind that, when a DIVERT command is in effect, you will see little or no output on
your screen. This is because the majority of output is being routed to the DIVERT file rather
than to the screen.
Note 2: Also, be sure you choose a DIVERT filename different from your model filename. If not, you
will overwrite your model file and will be unable to retrieve it!
RVRT
The RVRT command closes an output file opened with the DIVERT command. For an example of its
use, see the DIVERT command immediately above.
SAVE
The SAVE command saves the current model to a file. The syntax is:
SAVE filename
where filename is the name of the file to save your model in. LINGO saves the model in text format.
You can read the model back into LINGO with the TAKE command. We recommend you use an
extension of .LNG on your model files, so you can readily identify them.
You may want to use your own text editor to modify your model. If you do, be sure to save the LINGO
model in text (ASCII) format. Use the TAKE command to reopen the model in LINGO when you are
through editing it.
234 CHAPTER 6
In the following example, we input a small model and save it in the file titled MYMODEL.LNG:
: !Enter a small model
: MODEL
? MAX = 20*X + 30*Y;
? X <= 50;
? Y <= 60;
? X + 2*Y <= 120;
? END
: !Save model to a file
: SAVE MYMODEL.LNG
:
If you open the model file, MYMODEL.LNG, in a text editor, you should see the following:
MODEL:
1]MAX = 20*X + 30*Y;
2]X <= 50;
3]Y <= 60;
4]X + 2*Y <= 120;
END
SMPI
The SMPI command saves your model in a special format called Mathematical Programming
Interface (MPI). MPI is a special format developed by LINDO Systems for representing all classes of
mathematical programs − linear, integer, and nonlinear. This format is not intended for permanent
storage of your models. LINDO API users may be interested in this format for exporting models to the
LINDO API.
Note: At present, LINGO does not read MPI format files. Thus, it is important that you do not use
this format for permanent storage. Use the SAVE command, discussed above, to permanently
save your files for later retrieval.
SMPS
The SMPS command generates the underlying algebraic formulation for the current model and then
writes it to a disk file in MPS format. MPS format is a common format for representing linear
programming models. MPS files can be ported to any solver that reads MPS files—this includes most
commercial linear programming packages.
The syntax for the SMPS command is:
SMPS filename
where filename is the name of the file you wish to save the MPS representation of the model under.
COMMAND-LINE COMMANDS 235
In the following example, we input a small model and then save it in an MPS file:
: !Enter a small model
: MODEL
? MAX = 20*X + 30*Y;
? X <= 50;
? Y <= 60;
? X + 2*Y <= 120;
? END
: !Save model to an MPS file
: SMPS MYMODEL.MPS
:
If you open the MPS file created in a text editor, you should find:
NAME LINGO GENERATED MPS FILE(MAX)
ROWS
N 1
L 2
L 3
L 4
COLUMNS
Y 1 30.0000000
Y 3 1.0000000
Y 4 2.0000000
X 1 20.0000000
X 2 1.0000000
X 4 1.0000000
RHS
RHS 2 50.0000000
RHS 3 60.0000000
RHS 4 120.0000000
ENDATA
Note 1: Your model must be entirely linear to be able to successfully export it using SMPS. If a model
is nonlinear, the MPS file will contain question marks in place of numbers for coefficients of
nonlinear variables.
Note 2: SMPS truncates all variable names to 8 characters. For instance, the two distinct LINGO
names SHIP(WH1, C1) and SHIP(WH1, C2) would both be truncated to the single 8 character
name SHIPWH1C under SMPS . Either choose names to avoid collisions of truncated names
or enable the RCMPSN option for converting names to RC format when doing MPS I/O.
LINGO will print an error message if potential collisions exist.
Note 3: The MPS file format is intended primarily for exporting models to other applications or
platforms. The MPS format is purely scalar in nature—all set-based information is lost upon
converting a LINGO model to MPS format. Thus, when saving copies of a model on your
own machine, you should always use the SAVE command instead of the SMPS command.
236 CHAPTER 6
5. Solution
The Solution category contains commands for viewing a model’s solution.
DEBUG
In the ideal world, all models would return an optimal solution. Unfortunately, this is not the case.
Sooner or later, you are bound to run across either an infeasible or unbounded model. This is
particularly true in the development phase of a project when the model will tend to suffer from
typographical errors.
Tracking down an error in a large model can prove to be a daunting task. The DEBUG command is
useful in narrowing the search for problems in both infeasible and unbounded linear programs. A small
portion of the original model is isolated as the source of the problem. This allows you to focus your
attention on a subsection of the model in search of formulation or data entry errors.
The DEBUG command identifies two types of sets: sufficient and necessary. Removing any sufficient
set object from the model is sufficient to fix the entire model. Not all models will have a sufficient set.
In which case, they will have a necessary set with the property that removing any object from this set
fixes the remaining objects within that set.
As an example, suppose you have an infeasible model. If the complete model would be feasible except
for a bug in a single row, that row will be listed as part of the sufficient set. If the model has a
necessary set, then, as long as all of them are present, the model will remain infeasible.
The following example illustrates. The coefficient .55 in ROW4 should have been 5.5:
: look all
MODEL:
1][ROW1] Max = 3*X + 7*Y;
2][ROW2] X + 2*Y <= 3;
3][ROW3] 2*X + Y <= 2;
4][ROW4] 0.55*X + Y >=4;
END
COMMAND-LINE COMMANDS 237
When we attempt to solve this formulation, we get the following error:
: go
[Error Code: 81]
No feasible solution found.
Variable Value Reduced Cost
X 50.00000 0.000000
Y -23.50000 0.000000
Row Slack or Surplus Dual Price
ROW1 0.000000 -1.000000
ROW2 0.000000 8.500000
ROW3 -74.50000 0.000000
ROW4 0.000000 -10.00000
Next, if we run the DEBUG command, we are presented with the following report:
: debug
Sufficient Rows:
ROW4] .55 X + Y >= 4
Necessary Rows:
ROW2] X + 2 Y <= 3
Necessary Variable Bounds:
Y >= 0
The DEBUG command has correctly identified that the erroneous ROW4, when eliminated, is
sufficient to make the entire model feasible.
The debug feature operates in a similar manner for unbounded models. In the following example, we
introduced an error by placing a minus sign instead of a plus sign in front of variable Z3 in ROW3. A
look at ROW3 reveals that Z3 can be increased indefinitely, leading to an unbounded objective.
: look all
MODEL:
1][ROW1] Max = 12*X1 + 13*X2 + 22*Y1 + 23*Z1 +
2] 28*Z2 + X3 + Y3 + Z3;
3][ROW2] X1 + X2 + X3 <= 400;
4][ROW3] Y1 + Y2 + Y3 - Z3 <= 500;
5][ROW4] Z1 + Z2 <= 500;
END
238 CHAPTER 6
The resulting model is unbounded and, when issuing the LINGO|Solve command, we receive the
unbounded error message:
: go
[Error Code: 82]
Unbounded solution.
The DEBUG command has successfully determined that bounding Z3 is sufficient to bound the entire
model.
Typically, the DEBUG command helps to substantially reduce the search effort. The first version of
this feature was implemented in response to a user who had an infeasible model. The user had spent a
day searching for a bug in a model with 400 constraints. The debug feature quickly found a necessary
set with 55 constraints, as well as one sufficient set constraint. The user immediately noticed that the
right-hand side of the sufficient set constraint was incorrect.
GO
The GO command compiles and then solves the current model. When LINGO compiles the model, it
produces an internally executable version of the model and then runs it to produce the solution.
When LINGO finishes solving the model, it displays a full solution report on your screen. To suppress
the full solution report, issue the TERSE command before the GO command.
To capture the solution report generated by the GO command in a file, use the DIVERT command
before the GO command.
To set various parameters pertaining to the operation of LINGO’s solver, see the SET command later
in this chapter.
NONZ
The NONZ, or NONZEROS, command displays an abbreviated version of the solution for the current
model. NONZ is identical to the SOLUTION command with the exception that NONZ displays
information only about nonzero variables and binding rows (i.e., the slack or surplus is 0).
The syntax of the NONZ command is:
NONZ [‘header_text’] [var_or_row_name]
For a standard NONZ solution report, omit the two optional arguments and enter the NONZ command
by itself. LINGO will print primal and dual values for all nonzero variables and binding rows. LINGO
will label all the columns in the report.
COMMAND-LINE COMMANDS 239
The first optional field, header_text, will be displayed as a title header in the solution report. If the
header_text argument is included, LINGO prints primal values only, omitting all labels in the report.
The second optional field, var_or_row_name, is a variable or row name that, if included, will limit the
report to the given variable or row name.
As an example, in the following session, we load the Chess Snackfoods example from Chapter 2,
Using Sets, and then generate several solution reports using NONZ:
: TAKE CHESS.LNG
: TERSE
: GO
Global optimal solution found at step: 0
Objective value: 2692.308
: !Generate a standard NONZ report
: NONZ
Variable Value Reduced Cost
SUPPLY(PEANUTS) 750.0000 0.000000
SUPPLY(CASHEWS) 250.0000 0.000000
PRICE(PAWN) 2.000000 0.000000
PRICE(KNIGHT) 3.000000 0.000000
PRICE(BISHOP) 4.000000 0.000000
PRICE(KING) 5.000000 0.000000
PRODUCE(PAWN) 769.2308 0.000000
PRODUCE(KING) 230.7692 0.000000
FORMULA(PEANUTS, PAWN) 15.00000 0.000000
FORMULA(PEANUTS, KNIGHT) 10.00000 0.000000
FORMULA(PEANUTS, BISHOP) 6.000000 0.000000
FORMULA(PEANUTS, KING) 2.000000 0.000000
FORMULA(CASHEWS, PAWN) 1.000000 0.000000
FORMULA(CASHEWS, KNIGHT) 6.000000 0.000000
FORMULA(CASHEWS, BISHOP) 10.00000 0.000000
FORMULA(CASHEWS, KING) 14.00000 0.000000
Row Slack or Surplus Dual Price
1 2692.308 1.000000
2 0.000000 1.769231
3 0.000000 5.461538
: !Generate a NONZ report for PRODUCE
: NONZ PRODUCE
Variable Value Reduced Cost
PRODUCE(PAWN) 769.2308 0.000000
PRODUCE(KING) 230.7692 0.000000
: !Now add a header
: NONZ 'NONZERO PRODUCTION VALUES:' PRODUCE
NONZERO PRODUCTION VALUES:
769.2308
230.7692
If you would like to capture the solution report in a file, use the DIVERT command before the NONZ
command.
For more information on the interpretation of the various fields in the NONZ report, see Chapter 1,
Getting Started with LINGO.
240 CHAPTER 6
Note: If the solution report is scrolling off the screen, you can use the PAGE command to set the
page length to n lines, so LINGO will pause every time n lines are printed and wait until you
are ready to proceed with the next page.
RANGE
Use the RANGE command to generate a range report for the model in the active window. A range
report shows over what ranges you can: 1) change a coefficient in the objective without causing any of
the optimal values of the decision variables to change, or 2) change a row’s constant term (also
referred to as the right-hand side coefficient) without causing any of the optimal values of the dual
prices or reduced costs to change.
Note: The solver computes range values when you solve a model. Range computations must be
enabled in order for the solver to compute range values. Range computations are not enabled
by default, so you will need to switch them on with the command:
SET DUALCO 2
Range computations can take a fair amount of computation time. If speed is a concern, you
don’t want to enable range computations unnecessarily.
The example model below, when solved, yields the range report that follows:
[OBJECTIVE] MAX = 20 * A + 30 * C;
[ALIM] A <= 60;
[CLIM] C <= 50;
[JOINT] A + 2 * C <= 120;
The first section of the report is titled Objective Coefficient Ranges. In the first column, Variable, all
the optimizable variables are listed by name. The next column, Current Coefficient, lists the current
coefficient of the variable in the objective row. The third column, Allowable Increase, tells us the
amount that we could increase the objective coefficient without changing the optimal values for the
variables. The final column, Allowable Decrease, lists the amount that the objective coefficient of the
variable could decrease before the optimal values of the variables would change. Information on the
allowable increases and decreases on objective coefficients can be useful when you need answers to
questions like, “How much more (less) profitable must this activity be before we should be willing to
do more (less) of it?”
COMMAND-LINE COMMANDS 241
Referring to the Objective Coefficient Ranges report for our example, we can say, as long as the
objective coefficient of A is greater-than-or-equal-to 15, the optimal values of the variables will not
change. The same may be said for the objective coefficient of variable C, as long as it falls within the
range of [0-40].
Note: Ranges are valid only if you are planning to alter a single objective or right-hand side
coefficient. The range information provided by LINGO cannot be applied in situations where
one is simultaneously varying two or more coefficients. Furthermore, ranges are only lower
bounds on the amount of change required in a coefficient to actually force a change in the
optimal solution. You can change a coefficient by any amount up to the amount that is
indicated in the range report without causing a change in the optimal solution. Whether the
optimal solution will actually change if you exceed the allowable limit is not certain.
The second section of the range report is titled Right-hand side Ranges. The first column, Row, lists
the names of all the optimizable rows, or constraints, in the model. The second column, Current RHS,
gives the constant term, or right-hand side value, for the row. The next two columns, Allowable
Increase and Allowable Decrease, tell us how far we can either increase or decrease the right-hand
side coefficient of the row without causing a change in the optimal values of the dual prices or reduced
costs. If you recall, the dual prices on rows are, effectively, shadow prices, which tell us at what price
we should be willing to buy (or sell) our resources for. The dual prices do not, however, tell us what
quantity we should be willing to buy (or sell) at the dual price. This information is obtained from the
allowable increases and decreases on the right-hand side coefficients for the row. So, for our example,
the dual prices and reduced costs will remain constant as long as the right-hand side of row ALIM falls
within the range [20-120], the right-hand side of CLIM is greater-than-or-equal-to 30, and the
right-hand side of JOINT is in [60-160].
Note: We preceded all the rows in our model with a name enclosed in square brackets. This is an
important practice if you wish to generate range reports. If you do not name your rows,
LINGO assigns them a name that corresponds to the internal index of the row. This internal
index will not always correspond to the order of the row in the text of the original model. To
make the Right-hand side Ranges section of range reports meaningful, be sure to name all
your rows. For details on assigning names to rows, see page 29.
If a variable is nonlinear in the objective, its value in the Current Coefficient column will be displayed
as NONLINEAR. Similarly, if a row is nonlinear, the value in the Current RHS column will be
displayed as NONLINEAR.
Coefficients that can be increased or decreased indefinitely will display a range of INFINITY.
Fixed variables are substituted out of a model and will not appear in a range report. Rows that contain
only fixed variables are also substituted out of models and will not appear in range reports. As an
example, suppose we changed the following inequality in our sample model from:
[ALIM] A <= 60;
to the equality:
[ALIM] A = 60;
242 CHAPTER 6
LINGO can now solve directly for the value of A. The variable A is considered fixed; as is the row
ALIM (since it contains no optimizable variables). Given this, the variable A will no longer appear in
the Objective Coefficient Ranges section of the range report, and the row ALIM will not appear in the
Right-hand Side Ranges section. We can verify this by examining the updated range report:
Ranges in which the basis is unchanged:
Objective Coefficient Ranges
Current Allowable Allowable
Variable Coefficient Increase Decrease
C 30.00000 INFINITY 30.00000
Right-hand Side Ranges
Row Current Allowable Allowable
RHS Increase Decrease
CLIM 50.00000 INFINITY 20.00000
JOINT 60.00000 40.00000 60.00000
As a final note, if the range report is scrolling off the screen, you can use the PAGE n command to set
the page length to n lines, so LINGO will pause every time n lines are printed and wait until you are
ready to proceed with the next page. In addition, if you would like to capture the solution report in a
file, use the DIVERT command before the SOLU command.
SOLU
The SOLU, or SOLUTION, command displays a solution report for the current model. The syntax of
the SOLU command is:
SOLU [‘header_text’] [ var_or_row_name]
For a standard solution report, omit the two optional arguments, and enter the SOLU command by
itself. LINGO will print primal and dual values for all the variables and rows in the model. LINGO
will label all the columns in the report.
The first optional field, header_text, will be displayed as a title header in the solution report. If the
header_text argument is included, LINGO prints primal values only, omitting all labels in the report.
The second optional field, var_or_row_name, is a variable or row name that, if included, will limit the
report to the given variable or row name.
COMMAND-LINE COMMANDS 243
As an example, in the following session, we load the Chess Snackfoods example from Chapter 2,
Using Sets, and then generate several solution reports using SOLU:
: TAKE CHESS.LNG
: TERSE
: GO
Global optimal solution found at step: 0
Objective value: 2692.308
: !Generate a standard SOLU report
: SOLU
Variable Value Reduced Cost
SUPPLY(PEANUTS) 750.0000 0.0000000
SUPPLY(CASHEWS) 250.0000 0.0000000
PRICE(PAWN) 2.000000 0.0000000
PRICE(KNIGHT) 3.000000 0.0000000
PRICE(BISHOP) 4.000000 0.0000000
PRICE(KING) 5.000000 0.0000000
PRODUCE(PAWN) 769.2308 0.0000000
PRODUCE(KNIGHT) 0.000000 0.1538461
PRODUCE(BISHOP) 0.000000 0.7692297E-01
PRODUCE(KING) 230.7692 0.0000000
FORMULA(PEANUTS, PAWN) 15.00000 0.0000000
FORMULA(PEANUTS, KNIGHT) 10.00000 0.0000000
FORMULA(PEANUTS, BISHOP) 6.000000 0.0000000
FORMULA(PEANUTS, KING) 2.000000 0.0000000
FORMULA(CASHEWS, PAWN) 1.000000 0.0000000
FORMULA(CASHEWS, KNIGHT) 6.000000 0.0000000
FORMULA(CASHEWS, BISHOP) 10.00000 0.0000000
FORMULA(CASHEWS, KING) 14.00000 0.0000000
Row Slack or Surplus Dual Price
1 2692.308 1.000000
2 0.000000 1.769231
3 0.000000 5.461538
: !Generate a SOLU report for PRODUCE
: SOLU PRODUCE
Variable Value Reduced Cost
PRODUCE(PAWN) 769.2308 0.0000000
PRODUCE(KNIGHT) 0.000000 0.1538461
PRODUCE(BISHOP) 0.000000 0.7692297E-01
PRODUCE(KING) 230.7692 0.0000000
: !Now add a header
: SOLU 'PRODUCTION QUANTITIES' PRODUCE
PRODUCTION QUANTITIES
769.2308
0.000000
0.000000
230.7692
If you would like to capture the solution report in a file, use the DIVERT command before the SOLU
command.
For more information on the interpretation of the various fields in the solution report, see page 16.
If the solution report is scrolling off the screen, you can use the PAGE command to set the page length
to n lines, so LINGO will pause every time n lines are printed and wait until you are ready to proceed
with the next page.
244 CHAPTER 6
6. Problem Editing
The Problem Editing category contains commands used in editing and modifying models.
ALTER
The ALTER command is used to edit the current model. The syntax of ALTER is:
ALTER [line_number|line_range|ALL] 'old_string'new_string'
where,
line_number is the index of a single line to edit,
line_range is a range of lines to edit,
ALL means edit all the lines in the model,
old_string is the old string to search for and replace, and
new_string is the string to replace all occurrences of old_string with in the
specified line range.
COMMAND-LINE COMMANDS 245
In the following sample session, we read in a small knapsack model and perform two ALTER
commands to modify the model:
: TAKE ALTER.LNG
: LOOK ALL
1]SETS:
2] THINGS /1..4/: VALUE, WEIGHT, X;
3]ENDSETS
4]DATA:
5] VALUE = 8 6 4 3;
6] WEIGHT = 66 44 35 24;
7]ENDDATA
8] MAX = @SUM(THINGS: VALUE * X);
9] @SUM(THINGS: WEIGHT * X) >= 100;
10] @FOR(THINGS: @BIN(X));
: !Change the direction of the constraint
: ALTER 9 '>='<='
9] @SUM(THINGS: WEIGHT * X) <= 100;
: !Change 'THINGS' to 'ITEMS' in ALL rows
: ALTER ALL 'THINGS'ITEMS'
2] ITEMS /1..4/: VALUE, WEIGHT, X;
8] MAX = @SUM(ITEMS: VALUE * X);
9] @SUM(ITEMS: WEIGHT * X) <= 100;
10] @FOR(ITEMS: @BIN(X));
: LOOK ALL
1]SETS:
2] ITEMS /1..4/: VALUE, WEIGHT, X;
3]ENDSETS
4]DATA:
5] VALUE = 8 6 4 3;
6] WEIGHT = 66 44 35 24;
7]ENDDATA
8] MAX = @SUM(ITEMS: VALUE * X);
9] @SUM(ITEMS: WEIGHT * X) <= 100;
10] @FOR(ITEMS: @BIN(X));
:
Note: In addition to the single quote character ('), LINGO also allows the use of the double quote
character (") for delimiting the text fields of the ALTER command.
DELETE
The DELETE command is used to delete one or more lines of text from the current model. The syntax
of DELETE is:
DELETE [line_number|line_range|ALL]
where,
line_number is the index of a single line to delete,
line_range is a range of lines to delete, and
ALL means delete the entire model.
246 CHAPTER 6
Some examples of the DELETE command follow:
Example 1: DELETE 3
deletes line 3 of the model,
Example 2: DEL 2 10
deletes lines 2 through 10 of the model, and
Example 3: DEL ALL
deletes the entire model.
EXTEND
The EXTEND command allows you to append lines to the current model. It puts LINGO in model
input mode just after the last line of the current model. When you use the EXTEND command, you’ll
see LINGO’s question mark prompt. Start entering your new lines of model text. When you’re done,
enter END at the prompt.
In the following sample session, we use the EXTEND command to append an additional constraint to a
small model:
: LOOK ALL
1]MAX 20*X + 30*Y;
2]X <= 50;
3]Y <= 60;
4]X + 2*Y <=120;
: ! Use EXTEND to add another line
: EXTEND
? X >= 30;
? END
: LOOK ALL
1]MAX 20*X + 30*Y;
2]X <= 50;
3]Y <= 60;
4]X + 2*Y <=120;
5]X >= 30;
:
7. Conversational Parameters
The Conversational Parameters category contains commands that control how information is
displayed.
PAGE
The PAGE command sets the length of the page or screen size in lines. The syntax for PAGE is:
PAGE n
where n is the desired number of lines per page of output. For instance, PAGE 25 will cause the
display to pause after 25 lines and await a carriage return before displaying the next 25 lines. The
PAGE command is convenient when you wish to page through long reports and not have them scroll
off the top of the screen.
COMMAND-LINE COMMANDS 247
When 0 is entered as the argument to PAGE, paging is turned off entirely. LINGO will no longer stop
output to wait for a carriage return. Entering PAGE 0 at the top of any command script is helpful in
that you generally want command scripts to run uninterrupted.
The PAGE command is equivalent to the SET LENPAG command and is maintained for backward
compatibility
PAUSE
The PAUSE command causes screen display to pause until a carriage return is typed. If you enter text
on the same line as the PAUSE command, the text will be displayed. The PAUSE command is useful in
command scripts for conveying information to the user.
TERSE
The TERSE command causes LINGO to suppress the automatic display of a solution report after a
model is solved with the GO command. When TERSE is enabled, you will need to use the NONZ or
SOLU commands to view the solution.
When LINGO is in terse output mode, export summary reports are also suppressed. Export summary
reports are normally generated each time you export solutions to spreadsheets or databases.
Once you enter the TERSE command, LINGO stays in terse output mode until you enter the VERBOSE
command (see below).
The TERSE command is equivalent to the SET TERSEO 1 command and is maintained for backward
compatibility.
VERBOSE
The VERBOSE command undoes the effects of the TERSE command, and places LINGO in verbose
output mode. Verbose output mode is the default mode. It results in the automatic display of solution
reports after solving a model. Verbose output mode also results in the automatic display of export
summary reports whenever export operations are performed to spreadsheets and databases.
The VERBOSE command is equivalent to the SET TERSEO 0 command and is maintained for
backward compatibility.
WIDTH
Use the WIDTH command to set the terminal width for input and output. The syntax of the WIDTH
command is:
WIDTH n
where n is the desired terminal width. You may set the width between 64 and 200. The default is 76.
When LINGO generates reports, it limits output lines to the terminal width length. In some reports,
lines will be wrapped, so they fall within the line limit. In other reports, lines may be truncated. Since
LINGO concatenates variable names in performing set operations, a variable name, such as
SHIPMENTS(WAREHOUSE1, CUSTOMER2), may result, which may be truncated in a solution report
if too narrow a terminal width is used.
248 CHAPTER 6
The WIDTH command is equivalent to the SET LINLEN command and is maintained for backward
compatibility.
8. Tolerances
The Tolerances category contains commands for setting system parameters in LINGO.
APISET
The APISET command gives you access to all the parameters in the LINDO API, which is the solver
library used by LINGO. LINGO allows access to most of the important solver parameters through the
SET command and, under Windows, via the LINGO|Options command. However, some of the more
advanced parameters may only be accessed through the APISET command. The syntax for this
command is:
APISET param_id {int|double} param_value
where param_id is the parameter’s index and param_value is the value you wish to set the parameter
to. You will also need to indicate if the parameter is an integer or double precision quantity.
Some examples of the APISET command follow:
Example 1: APISET 341 INT 10000
sets the MIP branch limit (LS_IPARAM_MIP_BRANCH_LIMIT=341) to 10000,
Example 2: HELP APISET
will cause LINGO to display all current APISET settings, and
Example 3: APISET DEFAULT
removes all custom LINDO API settings, returning to the defaults.
You will need to refer to the LINDO API documentation for a list of available parameters and their
indices. The LINDO API documentation is available at no charge as part of the LINDO API download
on the LINDO Systems Web site. The LINGO installation also comes with a macro definition file,
Lindo.h, which contains all the parameter indices for the LINDO API.
Parameter values set with the APISET command are not stored from one LINGO session to the next.
Give the HELP APISET command for a listing of parameters that are currently active. To remove all
APISET parameter settings type the command: APISET DEFAULT.
If there are some LINDO API parameters you wish to permanently set, you may place a series of
APISET commands in an AUTOLG.DAT script file that automatically gets run at the start of each
LINGO session.
COMMAND-LINE COMMANDS 249
DBPWD
The DBPWD command is used to input a password for accessing databases via the @ODBC()
function. Any password input with this command will not be permanently stored. Therefore, at the
start of each session, you will need to reenter your database password. The syntax for the command is:
DBPWD my_password
See the DBUID command below for entering any user id required by your database.
DBUID
The DBUID command is used to input a user id for accessing databases via the @ODBC() function.
Any user id input with this command will not be permanently stored. Therefore, at the start of each
session, you will need to reenter your database user id. The syntax for the command is:
DBUID my_user_id
See the DBPWD command above for entering any password required with your user id.
FREEZE
The FREEZE command saves your current configuration to LINGO’s configuration file, so it may be
automatically restored the next time LINGO starts. Any non-default features of the current
configuration are saved to the LINGO.CNF file in LINGO’s main directory. The LINGO.CNF
configuration file is a text file, and the curious user may examine it by simply opening it in a text
editor. All parameters controlled by the SET command, see below, are stored by the FREEZE
command.
Note: Be careful when saving a non-default configuration. The saved configuration will
automatically be restored next time you start LINGO. Settings of certain parameters will
affect the way models are solved, potentially leading to misleading results when used on a
different set of models. To restore the default configuration, use the following command
sequence:
: SET DEFAULT
: FREEZE
SET
The SET command allows you to override LINGO’s default tolerances and settings. All user
configurable options in LINGO are available through the SET command. The syntax for the SET
command is:
SET parameter_name|parameter_index [parameter_value]
where,
parameter_name is the name of the parameter to set,
parameter_index is the index of the parameter to set, and
parameter_value is the new value for the parameter that, if omitted, will cause
LINGO to display the current value for the specified
parameter.
250 CHAPTER 6
Use the FREEZE command, see above, to save any tolerances modified with the SET command to the
configuration file, so they will be automatically restored the next time LINGO starts. You may also
enter SET DEFAULT to return all parameters to their default values.
Some examples of the SET command follow:
Example 1: SET MXMEMB 128
FREEZE
sets the generator memory limit to 128MB and saves parameter settings to
the configuration file,
Example 2: SET 5 1.E-7
sets the relative integrality tolerance (RELINT) toe 1.e-7,
Example 3: SET DEFAULT
restores all parameters to their default values, and
Example 4: HELP SET
causes LINGO to display all parameter settings.
The parameters accessible through the SET command are:
No. Name Default Description
1 ILFTOL 0.3e-5 Initial linear feasibility tolerance
2 FLFTOL 0.1e-6 Final linear feasibility tolerance
3 INFTOL 0.1e-2 Initial nonlinear feasibility tolerance
4 FNFTOL 0.1e-5 Final nonlinear feasibility tolerance
5 RELINT 0.8e-5 Relative integrality tolerance
6 NOPTOL 0.2e-6 Nonlinear optimality tolerance
7 ITRSLW 5 Iteration limit for slow progress
8 DERCMP 0 Derivatives (0:LINGO chooses, 1:backward analytical, 2:forward
analytical, 3:central differences, 4:forward differences)
9 ITRLIM 0 Iteration limit (0: no limit)
10 TIMLIM 0 Solver time limit in seconds (0: no limit)
11 OBJCTS 1 Objective cuts (1:yes, 0:no)
12 MXMEMB 32 Memory limit in megabytes for LINGO's model generator (N/A on
some machines)
13 CUTAPP 2 Cuts application (0:root, 1:all, 2:solver chooses)
14 ABSINT .000001 Absolute integrality tolerance
15 HEURIS 3 Integer programming heuristics (0:none, 100:advanced)
16 HURDLE 0 Use an Integer Programming (IP) hurdle value (1:yes, 0:no)
17 IPTOLA .8e-7 IP absolute optimality tolerance
18 IPTOLR .5e-7 IP relative optimality tolerance
19 TIM2RL 100 Seconds before switching to IP relative optimality tolerance
20 NODESL 0 0:LINGO decides, 1:depth first, 2:worst bound, 3:best bound
21 LENPAG 0 Terminal page length limit (0:none)
COMMAND-LINE COMMANDS 251
22 LINLEN 76 Terminal page width (0:none)
23 TERSEO 0 Output level (0:verbose, 1:terse)
24 STAWIN 1 Post status window (1:yes, 0:no, Windows only)
25 SPLASH 1 Display splash screen (1:yes, 0:no, Windows only)
26 OROUTE 0 Route output to command window (1:yes, 0:no, Windows only)
27 WNLINE 800 Max command window lines (Windows only)
28 WNTRIM 400 Min command window lines (Windows only)
29 STABAR 1 Display status bar (1:yes, 0:no, Windows only)
30 FILFMT 1 File format (0:lng, 1:lg4, 2:ltx, Windows only)
31 TOOLBR 1 Display toolbar (1:yes, 0:no, Windows only)
32 CHKDUP 0 Check for duplicate model names in data (1:yes, 0:no)
33 ECHOIN 0 Echo command input to terminal (1:yes, 0:no)
34 ERRDLG 1 Route error messages to a dialog box (1:yes, 0:no, Windows only)
35 USEPNM 0 Allow for unrestricted use of primitive set names (1:yes, 0:no)
36 NSTEEP 0 Use steepest edge variable selection in nonlinear solver (1:yes, 0:no)
37 NCRASH 0 Run crash procedure to get an initial starting point in nonlinear
models (1:yes,0:no)
38 NSLPDR 1 Compute search directions in nonlinear solver using successive linear
programming (1:yes, 0:no)
39 SELCON 0 Use selective constraint evaluation (1:yes, 0:no)
40 PRBLVL 0 Specify probing level on MILPs (0:LINGO chooses, 1:none, 7:high)
41 SOLVEL 0 Specify linear solver (0:LINGO chooses, 1:primal, 2:dual, 3:barrier)
42 REDUCE 2 Perform model reduction (0:no, 1:yes, 2:LINGO chooses)
43 SCALEM 1 Scale the model (0:no, 1:yes)
44 PRIMPR 0 Select primal pricing method (0:LINGO chooses, 1:partial, 2:devex)
45 DUALPR 0 Select dual pricing method (0:LINGO chooses, 1:Dantzig,
2:steepest-edge)
46 DUALCO 1 Specify dual computations (0:none, 1:prices, 2:prices only and ranges,
3:price only on optimizable rows)
47 RCMPSN 0 Use RC format names for MPS Input/Output (0:no, 1:yes)
48 MREGEN 1 Select model regeneration (0:only on modifications to model, 1:same
as 0 plus whenever model has external references, 2:always)
49 BRANDR 0 Select branch direction (0:both, 1:up, 2:down)
50 BRANPR 0 Select branch priority (0:LINGO decides, 1:binary)
51 CUTOFF .1e-8 Cutoff solution values smaller than this
52 STRONG 10 Specify strong branch level
53 REOPTB 0 IP warm start LP (0:LINGO, 1:barrier, 2:primal, 3:dual)
54 REOPTX 0 IP cold start LP (0:LINGO, 1:barrier, 2:primal, 3:dual)
55 MAXCTP 200 Max top cut passes
56 RCTLIM .75 Relative cuts limit
57 GUBCTS 1 GUB cuts (1:yes, 0:no)
58 FLWCTS 1 Flow cuts (1:yes, 0:no)
59 LFTCTS 1 Lift cuts (1:yes, 0:no)
60 PLOCTS 1 Plant location cuts (1:yes, 0:no)
252 CHAPTER 6
61 DISCTS 1 Disaggregation cuts (1:yes, 0:no)
62 KNPCTS 1 Knapsack cover cuts (1:yes, 0:no)
63 LATCTS 1 Lattice cuts (1:yes, 0:no)
64 GOMCTS 1 Gomory cuts (1:yes, 0:no)
65 COFCTS 1 Coefficient reduction cuts (1:yes, 0:no)
66 GCDCTS 1 Greatest common divisor cuts (1:yes, 0:no)
67 SCLRLM 1000 Syntax coloring line limit (Windows only)
68 SCLRDL 0 Syntax coloring delay in seconds (Windows only)
69 PRNCLR 1 Matching parenthesis coloring (1:yes, 0:no, Windows only)
70 MULTIS 0 NLP Multistart attempts (0:LINGO, n:number of attempts)
71 USEQPR 0 Use quadratic recognition (1:yes, 0:no)
72 GLOBAL 0 Use global solver on NLPs (1:yes, 0,no)
73 LNRISE 0 Linearization (0:LINGO, 1:none, 2:low, 3:high)
74 LNBIGM 100,000 Linearization BigM coefficient
75 LNDLTA .1e-5 Linearization Delta coefficient
76 BASCTS 0 Basis cuts (1:yes, 0:no)
77 MAXCTR 2 Max tree cuts passes
78 HUMNTM 0 Minimum heuristic time limit (seconds)
79 DECOMP 0 Matrix decomposition (1:yes, 0:no)
80 GLBOPT .1e-5 Global solver optimality tolerance
81 GLBDLT .1e-6 Global solver delta tolerance
82 GLBVBD .1e+11 Global solver variable bound limit
83 GLBUBD 2 Global solver bound use (0:no, 1:all, 2:some)
84 GLBBRN 5 Global solver branch selection (see below)
85 GLBBXS 1 Global solver box (0:depth first, 1:worst bound)
86 GLBREF 3 Global solver reformulation level (0:none, 3:high)
87 SUBOUT 2 Fixed variable reduction (0:none, 1:max, 2:not when using global or
multistart solvers)
88 NLPVER 0 NLP solver version (0:LINGO, 1:1.0, 2:2.0)
89 DBGCLD 0 Debugging cold start solver (0:LINGO, 1:primal, 2:dual, 3:barrier)
90 DBGWRM 0 Debug warm start solver (0:LINGO, 1:primal, 2:dual, 3:barrier)
91 LCRASH 1 Use aggressive crashing for NLPs (0:no, 1:yes)
92 BCROSS 1 Perform a basis crossover on LPs when using barrier solver (0:no
1:yes)
93 LOWMEM 0 Opt for less memory usage (0:no, 1:yes)
94 FILOUT 0 Fill out workbook output ranges (0:no, 1:yes)
95 DBGLVL 15 Debugger output level (1:low, 15:high)
96 UNARYM 1 Unary minus priority (0:low, 1:high)
97 LINEAR 0 Assume model is linear to reduce memory consumption(0:no, 1:yes)
98 LOPTOL .1e-6 Linear optimality tolerance
99 SECORD 0 Use second order derivatives for NLPs (0:no, 1:yes)
100 NONNEF 1 Variables default to being non-negative (0:no, 1:yes)
101 BIGMVL 1.e8 BigM coefficient threshold value
COMMAND-LINE COMMANDS 253
1. ILFTOL and 2. FLFTOL
Due to the finite precision available for floating point operations on digital computers, LINGO can’t
always satisfy each constraint exactly. Given this, LINGO uses these two tolerances as limits on the
amount of violation allowed on a constraint while still considering it “satisfied”. These two tolerances
are referred to as the initial linear feasibility tolerance (ILFTOL) and the final linear feasibility
tolerance (FLFTOL). The default values for these tolerances are, respectively, 0.000003 and
0.0000001.
ILFTOL is used when the solver first begins iterating. ILFTOL should be greater than FLFTOL. In the
early stages of the solution process, being less concerned with accuracy can boost the performance of
the solver. When LINGO thinks it has an optimal solution, it switches to the more restrictive FLFTOL.
At this stage in the solution process, one wants a relatively high degree of accuracy. Thus, FLFTOL
should be smaller than ILFTOL.
One instance where these tolerances can be of use is when LINGO returns a solution that is almost, but
not quite, feasible. You can verify this by checking the values in the Slack or Surplus column in the
model’s solution report. If there are only a few rows with small, negative values in this column, then
you have a solution that is close to being feasible. Loosening (i.e., increasing the values of) ILFTOL
and FLFTOL may help you get a feasible solution. This is particularly true in a model where scaling is
poor (i.e., very large and very small coefficients are used in the same model), and the units of
measurement on some constraints are such that minor violations are insignificant. For instance,
suppose you have a budget constraint measured in millions of dollars. In this case, a violation of a few
pennies would be of no consequence. Short of the preferred method of rescaling your model, loosening
the feasibility tolerances may be the most expedient way around a problem of this nature.
5. RELINT
RELINT, the relative integrality tolerance, is used by LINGO as a test for integrality in integer
programming models. Due to round-off errors, the “integer” variables in a solution may not have
values that are precisely integral. The relative integrality tolerance specifies the relative amount of
violation from integrality that is acceptable. Specifically, if I is the closest integer value to X, X will be
considered an integer if:
|X - I| <= Relative Integrality Tolerance.
|X|
The default value for the relative integrality tolerance is .000008. Although one might be tempted to
set this tolerance to 0, doing so may result in feasible models being reported as infeasible.
254 CHAPTER 6
6. NOPTOL
While solving a model, the nonlinear solver is constantly computing a gradient. The gradient gives the
rate of improvement of the objective function for small changes in the variables. If the gradient’s rate
of improvement computation for a given variable is less-than-or-equal-to NOPTOL, the nonlinear
optimality tolerance, further adjustments to the variable’s value are not considered to be beneficial.
The default value for the nonlinear optimality tolerance is .0000002. Decreasing this tolerance towards
a limit of 0 will tend to make the solver run longer and may lead to better solutions for poorly
formulated or poorly scaled models.
7. ITRSLW
LINGO’s nonlinear solver uses the ITRSLW, slow progress iteration limit, as a means of terminating
the solution process if little or no progress is being made in the objective value. Specifically, if the
objective function’s value has not improved significantly in n iterations, where n is the value of
ITRSLW, the nonlinear solver will terminate the solution process. Increasing this tolerance’s value will
tend to force the solver to run longer and may be useful in models that have relatively “flat” objective
functions around the optimal solution. The default value for ITRSLW is 5 iterations. Refer to the
description of ITRLIM below for a definition of iterations.
8. DERCMP
Use this parameter to set the style of derivative computation. Set DERCMP to 0 (Solver Decides) to
allow LINGO to select the method, 1 for backward analytical derivatives, 2 for forward analytical
derivatives, 3 for numerical derivatives using central differences, and 4 for numerical derivatives using
forward differences.
LINGO defaults to the Solver Decides setting, which presently involves using backward analytical
derivatives. However, we suggest you try the various derivative options to see which works best for
your particular models.
9. ITRLIM
Use this tolerance to place an upper limit on the number of iterations the solver will perform. An
iteration is the fundamental operation performed by the solver. At the risk of oversimplification, it is a
process that involves forcing a variable, currently at a zero value, to become nonzero until some other
variable is driven to zero, improving the objective as we go. In general, larger models will take longer
to perform an iteration, and nonlinear models will take longer than linear models. The default iteration
limit is 0, meaning no limit is imposed on the iteration count.
If the solver hits this limit, it returns to normal command mode. If the model contains integer variables,
LINGO will restore the best integer solution found so far. You may need to be patient, however,
because the solver may have to perform a fair amount of work to reinstall the current best solution
after it hits a runtime limit.
COMMAND-LINE COMMANDS 255
Note: Some caution is required when interrupting the solver. There must be an incumbent solution
available if you hope to interrupt the solver and have it return a valid solution. You can
always tell if an incumbent solution is available by examining the Best Obj field in the
Extended Solver Status box of the solver status window. If this field is blank, then an
incumbent solution does not exist, and the solution returned after an interrupt will be invalid.
If, on the other hand, this field contains a numeric value, then you should be able to interrupt
and return to a valid, if not globally optimal, solution.
10. TIMLIM
Use this tolerance to place a limit on the number of seconds the solver runs. If the solver hits this limit,
it will stop and return with the best solution found so far. The default limit is 0, meaning no time limit
is imposed on the solver.
If the solver hits this limit, it returns to normal command mode. If the model contains integer variables,
LINGO will restore the best integer solution found so far. You may need to be patient, however,
because the solver may have to perform a fair amount of work to reinstall the current best solution
after it hits a runtime limit.
Note: Some caution is required when interrupting the solver. There must be an incumbent solution
available if you hope to interrupt the solver and have it return a valid solution. You can
always tell if an incumbent solution is available by examining the Best Obj field in the
Extended Solver Status box of the solver status window. If this field is blank, then an
incumbent solution does not exist, and the solution returned after an interrupt will be invalid.
If, on the other hand, this field contains a numeric value, then you should be able to interrupt
and return to a valid, if not globally optimal, solution.
11. OBJCTS
Please refer to the Constraint Cut Types section below for information on this parameter.
12. MXMEMB
Use this parameter to set an upper limit on the amount of memory, in megabytes, that LINGO allocates
as workspace for its model generator. When LINGO starts up, it sets aside a fixed amount of memory
to use as a generator workspace. The default workspace size is 32Mb. You can determine the size of
the current workspace and the amount of memory allotted in this workspace by issuing the MEM
command.
Large models may run out of generator memory when attempting to solve them. In this case, you will
receive the error message, “The model generator ran out of memory.” To avoid this error, increase the
value of MXMEMB and issue the FREEZE command to preserve the change. You must then restart
LINGO.
Note: Changes in LINGO’s generator memory limit are not established until you restart the
program.
256 CHAPTER 6
The model generator is distinct from the actual solver engines. Memory allocated to the generator will
not be available to the solver engines. Thus, you shouldn’t allocate any more memory to the generator
than is required.
If you set MXMEMB to 0, LINGO will allocate all available memory when it starts up. This is not a
recommended practice.
Note: Setting LINGO’s generator memory limit abnormally high can result in poor performance of
LINGO and the operating system. By setting aside excessive amounts of memory for the
model generator, both LINGO and the operating system may have to resort to swapping of
virtual memory to and from the hard drive. Accessing the hard drive for memory swaps can
slow down your machine dramatically.
13. CUTAPP
Use this parameter to control the nodes in the solution tree where the branch-and-bound solver adds
constraint cuts in linear integer models. You have the following three options:
CUTAPP Setting Cuts Application at ...
0 Root only
1 All nodes
2 Solver decides
Under the Root Only option, the solver appends cuts only at the first node, or root node, in the solution
tree. With the All Nodes option, cuts are appended at each node of the tree. Under the Solver Decides
option, the solver dynamically decides when it is best to append cuts at a node.
The default is to let the solver decide when to append cuts. In general, this will offer superior
performance. There may be instances, however, where one of the other two options prevails.
14. ABSINT
Use this parameter to specify an absolute integrality tolerance. This tolerance is used by LINGO as a
test for integrality in integer programming models. Due to round-off errors, the "integer" variables in a
solution may not have values that are precisely integer. The absolute integrality tolerance specifies the
absolute amount of violation from integrality that is acceptable. Specifically, if X is an "integer"
variable and I is the closest integer to X, then X would be accepted as being integer valued if:
|X - I| <= Absolute Integrality Tolerance.
The default value for the absolute integrality tolerance is .000001. Although one might be tempted to
set this tolerance to 0, this may result in feasible models being reported as infeasible.
15. HEURIS
Use this parameter to control the level of integer programming heuristics used by the integer solver.
These heuristics use the continuous solution at each node in the branch-and-bound tree to attempt to
quickly find a good integer solution. If an integer solution better than the incumbent is found, then it is
used to fix or tighten global and local variable bounds. Heuristics are only applied to linear models.
Requesting heuristics on nonlinear models will result in no benefits.
HEURIS may be set anywhere from 0 (none) to 100 (highest level), with 3 being the default.
COMMAND-LINE COMMANDS 257
16. HURDLE
If you know the objective value of a solution to a model, you can enter it as a hurdle tolerance. This
value is used in the branch-and-bound solver to narrow the search for the optimum. More specifically,
LINGO will only search for integer solutions where the objective is better than the hurdle value. This
comes into play when LINGO is searching for an initial integer solution. LINGO can ignore branches
in the search tree with objective values worse than the hurdle value, because a better solution exists
(i.e., the hurdle) on some alternate branch. Depending on the problem, a good hurdle value can greatly
reduce solution time. Once LINGO finds an initial integer solution, however, the Hurdle tolerance no
longer has an effect.
Note: Be sure when entering a hurdle value that a solution exists that is at least as good or better
than your hurdle. If such a solution does not exist, LINGO will not be able to find a feasible
solution to the model.
The default hurdle value is None. In other words, a hurdle value is not used by the solver. To clear an
existing hurdle value, type SET HURDLE NONE.
17. IPTOLA
Use this parameter to specify the absolute optimality tolerance. This tolerance is a positive value r,
indicating to the branch-and-bound solver that it should only search for integer solutions with objective
values at least r units better than the best integer solution found so far. In many integer programming
models, there are huge numbers of branches with roughly equivalent potential. This tolerance helps
keep the branch-and-bound solver from being distracted by branches that can’t offer a solution
significantly better than the incumbent solution.
In general, you shouldn’t have to set this tolerance. Occasionally, particularly on poorly formulated
models, you might need to increase this tolerance slightly to improve performance. In most cases, you
should experiment with the relative optimality tolerance, discussed below, rather than the absolute
optimality tolerance in order to improve performance.
The default value for the absolute optimality tolerance is 8e-8.
18. IPTOLR
Use this parameter to specify the relative optimality tolerance. This tolerance is a value r, ranging
from 0 to 1, indicating to the branch-and-bound solver that it should only search for integer solutions
with objective values at least 100*r% better than the best integer solution found so far.
The end results of modifying the search procedure in this way are twofold. First, on the positive side,
solution times can be improved tremendously. Second, on the negative side, the final solution obtained
by LINGO may not be the true optimal solution. You will, however, be guaranteed the solution is
within 100*r% of the true optimum.
Typical values for the relative optimality tolerance would be in the range .01 to .05. In other words,
you would be happy to get a solution within 1% to 5% of the true optimal value. On larger integer
models, the alternative of getting a solution within a few percentage points of the true optimum after
several minutes of runtime, as opposed to the true optimum after several days, makes the use of an
optimality tolerance quite attractive.
258 CHAPTER 6
Note: Generally speaking, the relative integrality tolerance is the tolerance that will most likely
improve runtimes on integer models. You should be sure to set this tolerance whenever
possible.
The default for the relative optimality tolerance is 5e-8.
19. TIM2RL
If an integer programming model is relatively easy to solve, then we would like to have the solver
press on to the true optimal solution without immediately resorting to a relative optimality tolerance
(discussed above). On the other hand, if, after running for a while, it becomes apparent that the optimal
solution won’t be immediately forthcoming, then you might want the solver to switch to using a
relative optimality tolerance. TIM2RL, the time to relative tolerance, can be used in this manner. This
tolerance is the number of seconds before the branch-and-bound solver begins using the relative
optimality tolerance. For the first n seconds, where n is the value of the time to relative tolerance, the
branch-and-bound solver will not use the relative optimality tolerance and will attempt to find the true
optimal solution to the model. Thereafter, the solver will use the relative optimality tolerance in its
search.
The default value for the time to relative tolerance is 100 seconds.
20. NODESL
The branch-and-bound solver has a great deal of freedom in deciding how to span the
branch-and-bound solution tree. NODESL, the node selection option, allows you to control the order in
which the solver selects branch nodes in the tree.
The four choices available for NODESL are as follows:
NODESL Setting Branch Selection
0 LINGO Decides –This is the default option. LINGO makes
an educated guess as to the best node to branch on.
1 Depth First – LINGO spans the branch-and-bound tree
using a depth first strategy.
2 Worst Bound – LINGO picks the node with the worst
bound.
3 Best Bound – LINGO picks the node with the best bound.
In general, LINGO Decides will offer the best results. Experimentation with the other options may be
beneficial with some classes of models.
21. LENPAG
The LENPAG parameter sets the length of the page or screen size in lines. For instance, setting
LENPAG to 25 will cause the display to pause after 25 lines and await a carriage return before
displaying the next 25 lines. This is convenient when you wish to page through long reports and not
have them scroll off the top of the screen.
When LENPAG is set to 0, paging is turned off entirely. LINGO will no longer stop output to wait for
a carriage return. Entering SET LENPAGE 0 at the top of any command script is helpful in that you
generally want command scripts to run uninterrupted.
COMMAND-LINE COMMANDS 259
22. LINLEN
When LINGO generates reports, it limits output lines to a certain width. In some reports, lines will be
wrapped so that they fall within the line length limit. In other reports, lines may be truncated. Since
LINGO concatenates variable names in performing set operations, a variable name such as
SHIPMENTS( WAREHOUSE1, CUSTOMER2) may result, which may be truncated in a solution report
if too narrow an output width is used. You can control this line width limit through the LINLEN
parameter. You may set it anywhere between 64 and 200, with the default being 76.
23. TERSEO
You can use the TERSEO parameter to control the amount of output LINGO generates. There are four
settings available:
TERSEO Description
1 Verbose—Causes LINGO to display the maximum amount
of output, including full solution reports.
2 Terse—Less output than Verbose, with full solution reports
suppressed. This is a good output level if you tend to solve
large models. LINGO also suppresses Export Summary
Reports generated when exporting data to spreadsheets or
databases.
3 Errors Only—All output is suppressed, with the exception
of error messages
4 Nothing—LINGO suppresses all output. This level may be
useful when taking advantage of the programming
capabilities in LINGO, in which case, you will add
statements to your model to generate all required output.
The default setting for TERSEO is 1, or verbose mode.
260 CHAPTER 6
24. STAWIN (Windows Only)
If the STAWIN parameter is set to 1, LINGO displays a solver status window whenever you issue the
GO command. This window resembles the following:
The solver status window is useful for monitoring the progress of the solver and the dimensions of
your model. It is updated every n seconds, where n is the value in the Update interval field in the lower
right corner of the window. LINGO defaults to displaying the solver status window.
This option applies only to Windows versions of LINGO.
For a detailed description of the various fields in the solver status window, see Chapter 1, Getting
Started with LINGO.
The LG4 format is the default file format for Windows versions of LINGO. This is a binary format
that is readable only by LINGO. This format enables you to have custom formatting and fonts in your
262 CHAPTER 6
models, and allows you to use LINGO as an OLE server and container. Files written in LG4 format are
useful only on Windows hardware.
The LNG and LTX formats are text based. Given this, LNG and LTX files may be read into other
applications. However, these formats don’t support custom formatting and embedded objects. In
general, LNG files use LINGO syntax, while LTX files use LINDO syntax.
This option applies only to Windows versions of LINGO.
32. CHKDUP
Prior to release 4.0, LINGO allowed you to use primitive set names in the equations of a model.
Primitive set names in a model’s equations returned the index of the set member. Starting with release
4.0, LINGO required you to use the @INDEX function (see Chapter 7, LINGO's Operators and
Functions) to get the index of a primitive set member. If you would like to test your LINGO models
from releases prior to 4.0 for instances where primitive set members appear in the model’s equations,
set CHKDUP to 1. Whenever you run a model, LINGO will issue an error message if duplicate names
appear as set members and as variables in the model.
33. ECHOIN
When you run a LINGO command script with the TAKE command, the commands LINGO processes
are normally not displayed. If you would like the commands echoed to your screen, set the ECHOIN
parameter to 1. This can be a useful feature when you are trying to develop and debug a LINGO
command script.
If you want the index of FR in the DAYS set, you should use the @INDEX function (see Chapter 7,
LINGO's Operators and Functions):
INDEX_OF_FRIDAY = @INDEX(DAYS, FR);
If you are unable to update your models for some reason and you would like to allow for the direct use
of primitive set names, you can enable the USEPNM parameter by setting it to 1. The default is for
LINGO to disable USEPNM.
36. NSTEEP
Setting the NSTEEP parameter to 1 causes LINGO’s nonlinear solver to use steepest-edge variable
selection. When LINGO is not in steepest-edge mode, the nonlinear solver will tend to select variables
that offer the highest absolute rate of improvement to the objective, regardless of how far other
variables may have to move per unit of movement in the newly introduced variable. The problem with
this strategy is that other variables may quickly hit a bound, resulting in little gain to the objective.
With the steepest-edge option, the nonlinear solver spends a little more time in selecting variables by
looking at what rate the objective will improve relative to movements in the other nonzero variables.
Thus, on average, each iteration will lead to larger gains in the objective. In general, the steepest-edge
option will result in fewer iterations. However, each iteration will take longer. LINGO defaults to not
using the steepest-edge option.
37. NCRASH
If you set NCRASH to 1, LINGO’s nonlinear solver will invoke a heuristic for generating a “good”
starting point when you solve a model. If this initial point is relatively good, subsequent solver
iterations should be reduced along with overall runtimes. LINGO defaults to not crashing an initial
solution.
264 CHAPTER 6
38. NSLPDR
If you set NSLPDR to 1, LINGO’s nonlinear solver will use successive linear programming (SLP) to
compute new search directions. This technique uses a linear approximation in search computations in
order to speed iteration times. In general, the number of total iterations will tend to rise when SLP
directions are used, but on some models overall runtimes will improve. LINGO defaults to using SLP
directions.
39. SELCON
If you set SELCON to 1, LINGO’s nonlinear solver will only evaluate constraints on an as needed
basis. Thus, not every constraint will be evaluated at each iteration. This generally leads to faster
solution times, but can also lead to problems in models with undefined functions in certain regions.
LINGO may not evaluate a constraint for many iterations only to find that it has moved into a region
where the constraint is no longer defined. In this case, there may not be a valid point for the solver to
retreat to and the solution process terminates with an error. Turning off selective constraint evaluation
eliminates these errors. LINGO defaults to not using selective constraint evaluation.
40. PRBLVL
On a mixed-integer linear program, LINGO can perform an operation known as probing. Probing
involves taking a close look at the integer variables in a model and deducing tighter variable bounds
and right-hand side values. In many cases, probing can tighten an integer model sufficiently, thereby
speed overall solution times. In other cases, however, probing may not be able to do much tightening
and the overall solution time will increase due to the extra time spent probing. You can choose from
seven successive levels of probing ranging from 1 to 7. Level 1 disables probing completely, while
level 7 involves the highest degree of probing. Setting this option to 0 lets LINGO select the level of
probing. LINGO defaults to 0.
41. SOLVEL
This option allows you to choose the type of algorithm invoked by LINGO’s linear solver. At present,
LINGO offers the following four options:
SOLVEL Linear Solver
Value Algorithm
0 LINGO chooses
1 Primal simplex
2 Dual simplex
3 Barrier (only available as an option)
In general, it is difficult to say what algorithm will be fastest for a particular model. A rough guideline
is that primal simplex tends to do better on sparse models with fewer rows than columns; the dual does
well on sparse models with fewer columns than rows; and the barrier works best on densely structured
models or very large models.
The barrier solver is available only as an additional option to the LINGO package.
LINGO defaults to 0, LINGO chooses.
COMMAND-LINE COMMANDS 265
42. REDUCE
When this parameter is set to 1, LINGO’s linear solver tries to identify and remove extraneous
variables and constraints from the formulation before solving. In certain cases, this can greatly reduce
the size of the final model to be solved. Setting REDUCE to 1 enables reduction, while 0 disables it.
Setting REDUCE to 2 allows LINGO to choose whether or not to enable reduction. LINGO defaults to
this last option.
43. SCALEM
Setting SCALEM to 1 enables the scaling option in LINGO’s linear solver. This option rescales the
coefficients in the model’s matrix, causing the ratio of the largest to smallest coefficients to be
reduced. By doing this, LINGO reduces the chances of round-off error, which leads to greater
numerical stability and accuracy in the linear solver.
LINGO defaults to using scaling.
44. PRIMPR
Setting this parameter to 2 causes LINGO’s primal simplex linear solver to use devex pricing
techniques. If this parameter is set to 1, the primal simplex solver will use partial pricing. If this
parameter is set to 0, LINGO chooses the primal simplex pricing method.
LINGO defaults to choosing the primal pricing method.
45. DUALPR
If DUALPR is set to 2, LINGO’s dual simplex solver will use steepest edge pricing. If DUALPR is 1,
the dual solver will use Dantzig pricing methods. If DUALPR is 0, LINGO chooses the most
appropriate pricing method.
In Dantzig pricing mode, the dual simplex solver will tend to select variables that offer the highest
absolute rate of improvement to the objective, regardless of how far other variables may have to move
per unit of movement in the newly introduced variable. The problem with this strategy is that other
variables may quickly hit a bound, resulting in little gain to the objective. With the steepest-edge
option, the solver spends a little more time selecting variables by looking at the total improvement in
the objective by adjusting a particular variable. Thus, on average, each iteration will lead to larger
gains in the objective. In general, the steepest-edge option will result in fewer iterations. However,
each iteration will take longer.
LINGO defaults to choosing the pricing method for the dual solver.
46. DUALCO
The DUALCO parameter is used to set the level of dual computations performed by the solver. Setting
DUALCO to 0 will cause LINGO to not compute dual values and ranges. This is the fastest option, but
is suitable only if you don’t need this information. In fact, the RANGE command will not execute
when DUALCO is 0. When DUALCO is 1, LINGO will compute dual values, but not ranges. When
DUALCO is 2, LINGO computes both dual prices and ranges. Setting DUALCO to 3 causes LINGO to
compute the dual values on optimizable rows only (i.e., fixed rows are excluded) and forgo range
computations, LINGO defaults to a DUALCO value of 1.
266 CHAPTER 6
Note: Range computations can take some time, so, if speed is a concern, you don’t want to enable
range computations unnecessarily.
47. RCMPSN
Setting RCMPSN to 1 causes LINGO to convert all variable and row names to RC notation when
performing MPS file format I/O. Refer to the RMPS command on page 218 for a discussion of why
this option is useful. By default, LINGO disables the use of RC format names.
48. MREGEN
The MREGEN parameter controls the frequency with which LINGO regenerates a model. With
MREGEN set to 0, LINGO regenerates a model only when a change has been made to the model’s text
since the last generation took place. When MREGEN is 1, LINGO regenerates whenever a change is
made to the model text or if it contains references to external data sources (e.g., text files, databases, or
spreadsheets). If MREGEN is 2, then LINGO always regenerates the model each time information
regarding the generated model is needed. Commands that will trigger a model generation are GO,
GEN, GENL, STATS, RMPS, FRMPS, SMPS, and PICTURE. LINGO defaults to a MREGEN value
of 1.
49. BRANDR
LINGO uses a branch-and-bound solution procedure when solving integer programming models. One
of the fundamental operations involved in the branch-and-bound algorithm is branching on variables.
Branching involves forcing an integer variable that is currently fractional to either the next greatest
integer value or to the next lowest integer value. As an example, suppose there is a general integer
variable that currently has a value of 5.6. If LINGO were to branch on this variable, it would have to
choose whether to set the variable first to 6 or 5. The BRANDR parameter controls how LINGO makes
this branching decision.
There are three possible settings for BRANDR:
BRANDR Preferred Branching
Value Direction
0 Both up and down
1 Up
2 Down
The default option, Both up and down, involves LINGO making an intelligent guess as to whether it
should branch up or down first on each individual variable. If the Up option is selected, LINGO will
always branch up to the next highest integer first. If Down is selected, LINGO will always branch
down first. In most cases, the Both up and down option will result in the best performance.
Occasionally, models will benefit from use of one of the other two options.
COMMAND-LINE COMMANDS 267
50. BRANPR
When branching on variables, the branch-and-bound procedure can give priority to branching on the
binary variables first, or it can make an intelligent guess as to the next best variable to branch on,
regardless of whether it is binary or general.
There are two possible settings for BRANPR:
BRANPR Branching
Value Priority
0 LINGO decides
1 Binary variables first
Select the Binary variables first option to have LINGO give branching priority to the binary variables.
Select LINGO Decides to have LINGO select the next integer variable for branching based on an
intelligent guess regardless of whether it is binary or general. The default for this option is LINGO
Decides, which should generally give the best results. However, on occasion, the Binary option may
prevail.
51. CUTOFF
On occasion, due to round-off error, some of the values returned by LINGO’s solver will be very small
(less than 1e-10). In reality, the true values of these variables are either zero or so small as to be of no
consequence. These tiny values can be distracting when interpreting a solution report. The CUTOFF
parameter can be used to suppress small solution values. Any solution value less-than-or-equal-to
CUTOFF will be reported as being zero. The default value for CUTOFF is 1e-9.
52. STRONG
The strong branch option uses a more intensive branching strategy during the first n levels of the
branch-and-bound tree, where n is the value of the STRONG parameter. During these initial levels,
LINGO picks a subset of the fractional variables as branching candidates. LINGO then performs a
tentative branch on each variable in the subset, selecting as the final candidate the variable that offers
the greatest improvement in the bound on the objective. Although strong branching is useful in
tightening the bound quickly, it does take additional computation time. So, you may want to try
different settings to determine what works best for your model.
The default setting is 10 levels.
268 CHAPTER 6
53. REOPTB
The warm start option controls the linear solver that is used by the branch-and-bound solver at each
node of the solution tree when a previous solution is present to use as a “warm start”. The cold start
option, discussed below, determines the solver to use when a previous solution does not exist.
There are four possible settings for REOPTB:
REOPTB Warm Start
Value Solver
0 LINGO Decides – LINGO chooses the most
appropriate solver.
1 Barrier – LINGO uses the barrier method,
assuming you have purchased a license for the
barrier solver. Otherwise, the dual solver will be
used.
2 Primal – The primal solver will be used
exclusively.
3 Dual – The dual solver will be used exclusively.
In general, LINGO Decides will yield the best results. The barrier solver can’t make use of a
pre-existing solution, so Barrier usually won’t give good results. In general, Dual will be faster than
Primal for reoptimization in branch-and-bound.
54. REOPTX
The cold start option controls the linear solver that is used by the branch-and-bound solver at each
node of the solution tree when a previous solution is not present to use as a “warm start”. The warm
start option, discussed above, determines the solver to use when a previous solution does exist.
There are four possible settings for REOPTX :
REOPTX Warm Start
Value Solver
0 LINGO Decides – LINGO chooses the most
appropriate solver.
1 Barrier – LINGO uses the barrier method,
assuming you have purchased a license for the
barrier solver. Otherwise, the dual solver will be
used.
2 Primal – The primal solver will be used
exclusively.
3 Dual – The dual solver will be used exclusively.
In general, LINGO Decides will yield the best results. However, experimentation with the other
options may be fruitful.
COMMAND-LINE COMMANDS 269
55. MAXCTP
The integer pre-solver makes iterative passes through a model determining appropriate constraint cuts
to append to the formulation. In general, the marginal benefits of each additional pass declines. At
some point, additional passes will only add to total solution times. Thus, LINGO imposes a limit on
the maximum number of passes.
LINGO applies constraint cuts at both the top, or root, node of the branch-and-bound tree, and at all
subsequent nodes within the tree. The MAXCTP parameter limits the maximum number of cuts at the
top node, while the MAXCTR parameter (see below) sets the cut limit on all subsequent nodes in the
tree. The default limit is 200 passes.
56. RCTLIM
Most integer programming models benefit from the addition of some constraint cuts. However, at
some point, additional cuts take more time to generate than they save in solution time. For this reason,
LINGO imposes a relative limit on the number of constraint cuts that are generated. The default limit
is set to .75 times the number of true constraints in the original formulation. You may override this
relative limit by changing the setting of RCTLIM.
70. MULTIS
LINGO exploits the convex nature of linear models to find globally optimal solutions. However, we
aren’t as fortunate with nonlinear models. With nonlinear programming (NLP) models, LINGO’s
default NLP solver uses a local search procedure. This can lead to LINGO stopping at locally optimal
points, perhaps missing a global point lying elsewhere. Refer to Chapter 14, On Mathematical
Modeling, for more information on how and why this can happen.
A strategy that has proven successful in overcoming this problem is to restart the NLP solver several
times from different initial points. It is not uncommon for a different starting point to lead to a
different local solution point. The idea is that, if we restart from enough unique points, saving the best
local solution as we go, then we have a much better chance of finding the true global solution.
The MULTIS parameter allows you to set the number of times you would like the NLP solver to re-
solve your model, starting each time from an intelligently generated, new starting point. We refer to
this feature as multistart. The default value for MULTIS, 0, entails restarting 5 times on small NLPs
and disabling multistart on larger models. Setting MULTIS to 1 disables multistart on all NLPs. Setting
MULTIS to any value greater than 1 will cause the NLP solver to restart that number of times on all
NLPs. We have found that setting MULTIS around 5 tends to be adequate for most models. Highly
nonlinear models may require a larger setting.
COMMAND-LINE COMMANDS 271
Keep in mind, however, that multistart will dramatically increase runtimes. Thus, one should avoid
using it unnecessarily on convex models that will converge to a global point in a single pass without
any additional prodding.
The following example illustrates the usefulness of multistart. Consider the simple, yet highly
nonlinear, model:
MODEL:
MIN = X * @COS( 3.1416 * X);
@BND( 0, X, 6);
END
The objective function has three local, minimal points over the feasible range. These points are
summarized in the following table:
Point X Objective
1 1.09 -1.05
2 3.03 -3.02
3 5.02 -5.01
272 CHAPTER 6
Clearly, the third local point is also the globally best point, and we would like the NLP solver to
converge to this point. Below, we attempt this by loading the model, turning off the multistart option,
and then solving:
: take wavy.lng
: look all
MODEL:
1] MIN = X * @COS( 3.1416 * X);
2] @BND( 0, X, 6);
END
: set multis 1 !set solver attempts to 1 only (i.e., disable ms)
Parameter Old Value New Value
MULTIS 0 1
: go
Local optimal solution found at step: 11
Objective value: -1.046719
Variable Value Reduced Cost
X 1.090405 0.1181082E-07
Row Slack or Surplus Dual Price
1 -1.046719 -1.000000
Unfortunately, as you can see, we converged to the least preferable of the local minimums. Below, we
will do the same as in the previous run. However, this time, we will set the number of multistarts to
five:
: take wavy.lng
: look all
MODEL:
1] MIN = X * @COS( 3.1416 * X);
2] @BND( 0, X, 6);
END
: set multis 5
Parameter Old Value New Value
MULTIS 0 5
: go
Local optimal solution found at step: 39
Objective value: -5.010083
Variable Value Reduced Cost
X 5.020143 -0.7076917E-08
Row Slack or Surplus Dual Price
1 -5.010083 -1.000000
The extra four restarts allowed LINGO to find the global optimal point.
COMMAND-LINE COMMANDS 273
71. USEQPR
The USEQPR parameter controls the Quadratic Recognition option. This option consists of an
algebraic preprocessor that automatically determines if an arbitrary nonlinear model is actually a
quadratic programming (QP) model. If a model is found to be a convex QP, then it can be passed to the
faster quadratic solver. Note that the QP solver is not included with the base version of LINGO, but
comes as part of the barrier option.
LINGO defaults to not using quadratic recognition. You may enable this option with the command:
SET USEQPR 1.
72. GLOBAL
Many nonlinear models are non-convex and/or non-smooth (for more information see Chapter 14, On
Mathematical Modeling). Nonlinear solvers that rely on local search procedures, as does LINGO’s
default nonlinear solver, will tend to do poorly on these types of models. Typically, they will converge
to a local, sub-optimal point that may be quite distant from the true, global optimal point. Global
solvers overcome this weakness through methods of range bounding (e.g., interval analysis and convex
analysis) and range reduction techniques (e.g., linear programming and constraint propagation) within
a branch-and-bound framework to find the global solutions to non-convex models. LINGO has a
global solver capability that is enabled through the GLOBAL parameter. Setting GLOBAL to 1 will
enable the global solver on nonlinear models, while setting it to 0 (the default) will not.
The following example illustrates the power of the global solver on a non-smooth model. Consider the
following model:
model:
sets:
projects: baths, sqft, beds, cost, est;
endsets
data:
projects, beds, baths, sqft, cost =
p1 5 4 6200 559608
p2 2 1 820 151826
p3 1 1 710 125943
p4 4 3 4300 420801
p5 4 2 3800 374751
p6 3 1 2200 251674
p7 3 2 3400 332426
;
enddata
min = @max( projects: @abs( cost - est));
@for( projects:
est = a0 + a1 * beds + a2 * baths + a3 * sqft
);
end
Model: COSTING
274 CHAPTER 6
This model estimates the cost of home construction jobs based on historical data on the number of
bedrooms, bathrooms, and square footage. The objective minimizes the maximum error over the
sample project set. Both the @MAX() and @ABS() functions in the objective are non-smooth, and, as a
result, can present problems for LINGO’s default, local search NLP solver. Running the model under
the default settings with the global solver disabled, we get the following result:
Local optimal solution found at step: 91
Objective value: 3997.347
Variable Value Reduced Cost
A0 37441.55 0.000000
A1 27234.51 0.000000
A2 23416.53 0.000000
A3 47.77956 0.000000
Enabling the global solver with the SET GLOBAL 1 command and re-optimizing yields the substantially
better solution:
Global optimal solution found at step: 186
Objective value: 1426.660
Variable Value Reduced Cost
A0 46814.64 0.000000
A1 22824.18 0.000000
A2 16717.33 0.000000
A3 53.74674 0.000000
Note that the maximum error has been reduced from 3,997 to 1,426!
This example illustrates the power of the global solver. Unfortunately, there is a drawback. You will
find the global solver runs considerably slower than the default local solver, and may be swamped
when trying to solve larger models. Therefore, the preferred option is to always try to create smooth,
convex models, so that the faster, default local solver can successfully solve them.
Keep in mind that the global solver supports most, but not all, of the functions available in the LINGO
language. The following is a list of the nonlinear functions not currently supported by the global
solver:
♦ @PBN()—Cumulative binomial probability
♦ @PCX()—Cumulative Chi-squared distrribution
♦ @PFD()—Cumulative F distribution
♦ @PHG()—Cumulative hypergeometric probability
♦ @PFS()—Poisson finite source
♦ @PPL()—Poisson linear loss
♦ @PTD()—Cumulative t distribution
♦ @USER()-User supplied function
Note: The global solver will not operate on models containing one or more unsupported nonlinear
operations that reference optimizable quantities; the default NLP solver will be called in this
case.
The global solver is disabled by default.
COMMAND-LINE COMMANDS 275
73-75. LNRISE, LNBIGM, LNDLTA
The LNRISE, LNBIGM, and LNDLTA parameters control the linearization option in LINGO. Many
nonlinear operations can be replaced by linear operations that are mathematically equivalent. The
ultimate goal is to replace all the nonlinear operations in a model with equivalent linear ones, thereby
allowing use of the faster and more robust linear solvers. We refer to this process as linearization.
The LNRISE parameter determines the extent to which LINGO will attempt to linearize models. The
available options are:
LNRISE Setting Linearization Level
0 Solver Decides
1 None
2 Low
3 High
Under the None option, no linearization occurs. With the Low option, LINGO linearizes @ABS(),
@MAX(), @MIN(), @SMAX(), and @SMIN() function references along with any products of binary
and continuous variables. The High option is equivalent to the Low option, plus LINGO will linearize
all logical operators (#LT#, #LE#, #EQ#, #GT#, #GE#, and #NE#). Under the Solver Decides option,
LINGO will do maximum linearization if the number of variables doesn’t exceed 12. Otherwise,
LINGO will not perform any linearization. LINGO defaults to the Solver Decides setting.
The LNDLTA parameter controls the Delta Coefficient, which is a tolerance indicating how closely you
want the additional constraints added as part of linearization to be satisfied. Most models won’t require
any changes to this parameter. However, some numerically challenging formulations may benefit from
increasing Delta slightly. LINGO defaults to a Delta of 1.e-6.
When LINGO linearizes a model, it adds forcing constraints to the mathematical program generated to
optimize your model. These forcing constraints are of the form:
f( Adjustable Cells) ≤ M • y
where M is the BigM Coefficient and y is a 0/1 variable. The idea is that, if some activity in the
variables is occurring, then the forcing constraint will drive y to take on the value of 1. Given this, if
we set the BigM value to be too small, we may end up with an infeasible model. Therefore, the astute
reader might conclude that it would be smart to make BigM quite large, thereby minimizing the chance
of an infeasible model. Unfortunately, setting BigM to a large number can lead to numerical stability
problems in the solver resulting in infeasible or sub-optimal solutions. So, getting a good value for the
BigM Coefficient may take some experimentation. The default value for BigM is 100,000.
276 CHAPTER 6
As an example of linearization, consider the following model:
model:
sets:
projects: baths, sqft, beds, cost, est;
endsets
data:
projects, beds, baths, sqft, cost =
p1 5 4 6200 559608
p2 2 1 820 151826
p3 1 1 710 125943
p4 4 3 4300 420801
p5 4 2 3800 374751
p6 3 1 2200 251674
p7 3 2 3400 332426
;
enddata
min = @max( projects: @abs( cost - est));
@for( projects:
est = a0 + a1 * beds + a2 * baths + a3 * sqft
);
end
Model: COSTING
This model estimates the cost of home construction jobs based on historical data on the number of
bedrooms, bathrooms, and square footage. The objective minimizes the maximum error over the
sample project set. Both the @MAX() and @ABS() functions in the objective are non-smooth nonlinear
functions, and, as a result, can present problems for LINGO’s default, local search NLP solver.
Running the model under the default settings with linearization disabled, we get the following result:
Local optimal solution found at step: 91
Objective value: 3997.347
Variable Value Reduced Cost
A0 37441.55 0.000000
A1 27234.51 0.000000
A2 23416.53 0.000000
A3 47.77956 0.000000
Note that the maximum error has been reduced from 3,997 to 1,426!
Linearization will substantially increase the size of your model. The sample model above, in un-
linearized form, has a mere 8 rows and 11 continuous variables. On the other hand, the linearized
version has 51 rows, 33 continuous variables, and 14 binary variables! Although linearization will
COMMAND-LINE COMMANDS 277
cause your model to grow in size, you will tend to get much better solution results if the model can be
converted entirely to an equivalent linear form.
Note: Linearization will be of most use when a nonlinear model can be 100% linearized. If LINGO
can only linearize a portion of your model, then you may actually end up with a more difficult
nonlinear model.
You may view the linearization components that are added to your model with the GEN command. We
will illustrate with the following model:
model:
min = @abs( x - 5) + 4 * @abs( y -3);
x + 2 * y <= 10;
end
Setting the Linearization option to High (SET LNRISE 2) and running the GEN command on this
model yields the following linearized model:
MIN _C03 + 4 _C07
SUBJECT TO
2] X + 2 Y <= 10
_R01]- _C01 - _C02 + _C03 = 0
_R02] _C01 - 100000 _C04 <= 0
_R03] _C02 + 100000 _C04 <= 100000
_R04] X - _C01 + _C02 = 5
_R05]- _C05 - _C06 + _C07 = 0
_R06] _C05 - 100000 _C08 <= 0
_R07] _C06 + 100000 _C08 <= 100000
_R08] Y - _C05 + _C06 = 3
END
SUB _C01 100000.000
SUB _C02 100000.000
SUB _C03 100000.000
INTE _C04
SUB _C05 100000.000
SUB _C06 100000.000
SUB _C07 100000.000
INTE _C08
Columns added due to linearization have names beginning with _C, while linearization rows have
names starting with _R. In this particular example, rows _R01 through _R08 and columns _C01
through _C08 were added due to linearization.
The linearization option is disabled by default.
76. BASCTS
Please refer to the Constraint Cut Types section above for information on this parameter.
77. MAXCTR
This parameter controls the number of passes the branch-and-bound solver makes at each node of the
tree for cut generation. There is one exception in that MAXCTR does not control the number of passes
at the root node of the tree. You must use MAXCTP, see above, to control the number of passes at the
root node. The default value for MAXCTR is 2 passes.
278 CHAPTER 6
78. HUMNTM
This parameter sets the minimum amount of time spent in heuristics at each node of the branch-and-
bound tree. The default value for HUMNTM is 0 seconds.
79. DECOMP
Many large scale linear and mixed integer problems have constraint matrices that are totally
decomposable into a series of block structures. If total decomposition is possible, LINGO can solve the
independent problems sequentially and report a solution for the original model, resulting in dramatic
speed improvements. Setting DECOMP to 1 enables the decomposition feature.
LINGO defaults to not using matrix decomposition.
80. GLBOPT
The GLBOPT tolerance specifies by how much a new solution must beat the objective value of the
incumbent solution in order to become the new incumbent in the global solver. The default value for
GLBOPT is 1. e-6.
81. GLBDLT
The GLBDLT tolerance specifies how closely the additional constraints, added as part of the global
solver’s convexification process, must be satisfied. The default value for GLBDLT is 1. e-7.
82. GLBVBD
The GLBVBD tolerance sets the default variable bounds while the global solver is running. If this
parameter is set to d, then variables will not be permitted to assume values outside the range of [-d, d].
Setting this parameter as tightly as possible in the Value Field restricts the global solver from straying
into uninteresting regions and will reduce run times. You may also need to set the GLBUBD tolerance
(see below) to control how the global solver uses the bound. The default value for GLBVBD is 1. e
+10.
83. GLBUBD
The GLBUBD tolerance controls how the global solver’s variable bound tolerance, GLBVBD (see
above), is applied. There are three choices available: 0:None, 1:All, and 2:Selected. Selecting None
removes the variable bound entirely and is not recommended. The All setting applies the bound to all
variables. Finally, the Selected setting causes the global solver to apply the bound after an initial solver
pass to find the first local solution. The bound will only be applied to a variable if it does not cut off
the initial local solution. LINGO defaults to the Selected setting.
COMMAND-LINE COMMANDS 279
84. GLBBRN
The GLBBRN tolerance specifies the branching direction for variables when the global solver initially
branches on them. Six options are available:
0 Absolute Width
1 Local Width
2 Global Width
3 Global Distance
4 Absolute Violation
5 Relative Violation
85. GLBBXS
The GLBBXS parameter specifies the strategy to use for choosing between all active nodes in the
global solver’s branch-and-bound tree. The choices are: 0:Depth First and 1:Worst Bound. The default
is 1, or Worst Bound.
86. GLBREF
The GLBREF option sets the degree of algebraic reformulation performed by the global solver.
Algebraic reformulation is critical for construction of tight, convex sub-regions to enclose the
nonlinear and nonconvex functions. The available settings are: 0:None, 1:Low, 2:Medium, and 3:High.
The default is 3, or High.
87. SUBOUT
The SUBOUT option is used to control the degree to which fixed variables are substituted out of the
ultimate math program passed to the solver engines.
For example, consider the model:
MAX= 20*X + 30*Y + 12*Z;
X = 2*Y;
X + Y + Z <= 110;
Y = 30;
If we run the GEN command, we see that LINGO is able to reduce this model down to the equivalent,
but smaller model:
MAX= 12 * Z + 2100;
Z <= 20;
From the third constraint of the original model it is obvious that Y is fixed at the value 30. Plugging
this value for Y into the first constraint, we can conclude that X has a value of 60. Substituting these
two fixed variables out of the original formulation yields the reduced formulation above.
In most cases, substituting out fixed variables yields a smaller, more manageable model. In some
cases, however, you may wish to avoid this substitution. An instance in which you might want to
280 CHAPTER 6
avoid substitution would be when equations have more than one root. When m multiple roots are
present, reduction may select a suboptimal root for a particular equation. On the other hand, the global
and multistart solvers are adept at handling equations containing multiple roots. Thus, when using
these solvers one may wish to forgo fixed variable reduction.
The available options are:
0 None
1 Always
2 Not with global and
multistart
Selecting None disables all fixed variable reduction. Selecting Always enables reduction. When Not
with global and multistart is selected, LINGO disables reduction whenever either the global or
multistart solvers are selected, otherwise reduction is performed.
Note: You should be careful when turning off fixed variable reduction. If the model generator is
unable to substitute out fixed variables, you may end up turning a linear model into a more
difficult nonlinear model.
LINGO defaults to selecting Not with global and multistart for fixed variable reduction.
88. NLPVER
The NLPVER option determines the version of the nonlinear solver that is invoked:
This option is available on the off chance that the older version of the nonlinear solver performs better
on a particular model.
LINGO defaults to Solver Decides for the nonlinear solver version.
COMMAND-LINE COMMANDS 281
89. DBGCLD and 90. DBGWRM
These two parameters give you control over the linear solver that is used by the DEBUG command for
model debugging. The available choices are:
DBGCLD selects the solver for cold starts (starting without an existing basis in memory) and
DBGWRM selects the solver for warm starts (restarting from an existing basis).
LINGO defaults to Solver Decides for both the cold and warm debug solver.
91. LCRASH
The LCRASH parameterError! Bookmark not defined. controls the use of aggressive crashing
techniques on nonlinear programs. Crashing is a heuristic process designed to find a good starting
point for a model. The available choices are: 0 for none, 1 for low and 2 for high. The default setting
is 1, or low.
92. BCROSS
The BCROSS parameter controls whether or not the barrier solver performs a basis crossover on linear
programs. Barrier solvers do not normally return basic solutions. For example, if alternate optima
exist, the barrier method will return a solution that is, loosely speaking, the “average” of all alternate
optima. The basis crossover process converts a non-basic barrier solver solution to a basic (i.e., corner
point) solution. The available choices are: 0 for no crossover and 1 (the default) to perform a
crossover.
93. LOWMEM
The LOWMEM option may be used to guide LINGO’s memory usage. Enabling this option (SET
LOWMEM 1) causes LINGO to opt for less memory usage when solving a model. The downside is
that opting for less memory may result in longer runtimes.
LINGO defaults to disabling the LOWMEM option.
94. FILOUT
LINGO can export a model’s solution to Excel and databases. When exporting to Excel, LINGO
sends solutions to user defined ranges in a workbook. Solutions exported to a database are sent to
tables within the database. In either case, the target range or table may contain more space for values
than you are actually exporting. In other words, there may be cells at the end of ranges or records at
282 CHAPTER 6
the end of tables that will not be receiving exported values from LINGO. The Fill Out Ranges and
Tables option determines how these extra cells and records are treated.
When the Fill Out Ranges and Tables option is enabled (SET FILOUT 1), LINGO overwrites the
extra values. Conversely, when the option is not enabled (SET FILOUT 0), LINGO leaves the extra
values untouched.
Fill Out Ranges and Tables is disabled by default.
95. DBGLVL
The DBGLVL option gives you control over the output level of the model debugging command,
DEBUG. The debugger is very useful in tracking down problems in models that are either infeasible
or unbounded. Possible output levels range from 1 (minimum output) to 15 (maximum output). In
general, you will want to generate as much output as possible. The only reason to restrict the amount
of output would be to speed debugging times on large models.
The default setting for the debugger output level is 15.
96. UNARYM
The UNARYM option is used to set the priority of the unary minus operator. The two available
options are High (SET UNARYM 1) are Low (SET UNARYM 0).
There are two theories as to the priority that should be assigned to the unary minus (i.e., negation)
operator in mathematical expressions. On the one hand, there is the Excel practice that the unary
minus operator should have the highest priority, in which case, the expression 3^2 would evaluate to
+9. On the other hand, there is the mathematicians’ preference for assigning a lower priority to unary
minus than is assigned to exponentiation, in which case, 3^2 evaluates to 9. Note that regardless
which relative priority is used, one can force the desired result through the use of parenthesis.
LINGO defaults to the Excel approach of setting a higher priority (High) on negation than on
exponentiation.
97. LINEAR
The LINEAR option can be enabled (SET LINEAR 1) to minimize memory usage on models that are
entirely linear. When this option is in effect, the model generator can take steps to dramatically reduce
overall memory consumption without sacrificing performance. In fact, if all your models are linear,
we recommend that you enable this option permanently as the default for your installation. The one
restriction is that models must prove to be entirely linear. If a single nonlinearity is detected, you will
receive an error message stating that the model is nonlinear and model generation will cease. At which
point, you should clear this option and attempt to solve the model again.
By default, the LINEAR option is disabled.
98. LOPTOL
The LOPTOL parameter allows you to control the setting for the linear optimality tolerance. This
tolerance is used to determine whether a reduced cost on a variable is significantly different from zero.
You may wish to loosen this tolerance (make it larger) on poorly scaled and/or large models to
improve performance.
The default setting for the LOPTOL parameter is 1.e-7.
COMMAND-LINE COMMANDS 283
99. SECORD
The SECORD option determines if the nonlinear solver will use second order derivates. If used (SET
SECORD 1), second order derivatives will always be computed analytically, as opposed to using
numerical differences. Computing second order derivatives will take more time, but the additional
information they provide may lead to faster runtimes and/or more accurate solutions.
LINGO defaults to not using second order derivatives.
100. NONNEG
When enabled (SET NONNEG 1), the NONNEG option tells LINGO to place a default lower bound of
0 on all variables. In other words, unless otherwise specified, variables will not be allowed to go
negative. Should you want a variable to take on a negative value, you may always override the default
lower bound of 0 using the @BND() function. If this option is disabled, then LINGO’s default
assumption is that variables are unconstrained and may take on any value, positive or negative.
Unconstrained variables are also referred to as be being free.
By default, LINGO enables the non-negative option, thereby setting a default lower bound of 0 on all
variables.
101. BIGMVL
Many integer programming models have constraints of the form:
f(x) ≤ M * z
where f(x) is some function of the decision variables, M is a large constant term, and z is a binary
variable. These types of constraints are called forcing constraints and are used to force the binary
variable, z, to 1 when f(x) is nonzero. In many instances, the binary variable is multiplied by a fixed
cost term in the objective; a fixed cost that is incurred when a particular activity, represented by f(x),
occurs. The large constant tem, M, Is frequently referred to as being a BigM coefficient.
Setting BigM too small can lead to infeasible or suboptimal models. Therefore, the BigM value will
typically have to be rather large in order to exceed the largest activity level of f(x). When BigM is
large, the solver may discover that by setting z slightly positive (within normal integrality tolerances),
it can increase f(x) to a significant level and thereby improve the objective. Although such solutions
are technically feasible to tolerances, they are invalid in that the activity is occurring without incurring
its associated fixed cost.
The BIGMVL parameter, or BigM threshold, is designed to avoid this problem by allowing LINGO to
identify the binary variables that are being set by forcing constraints. Any binary variable with a
coefficient larger than the BigM threshold will be subject to a much tighter integrality tolerance.
The default value for the BigM Threshold is 1.e8.
284 CHAPTER 6
9. Miscellaneous
The Miscellaneous category contains various LINGO commands that don’t fall into one of the other
eight command categories.
!
Place an exclamation mark in a command and LINGO ignores the remainder of the line following the
exclamation mark.
QUIT
Issue the QUIT command to close the LINGO application. Be sure to save any changes made to your
model before quitting.
TIME
Displays the current elapsed time since the start of the current LINGO session as illustrated in the
following example:
: TIME
Cumulative HR:MIN:SEC = 2:22:39.54
:
7 LINGO’s Operators and
Functions
LINGO provides the mathematical modeler with a number of functions and operators. For our
purposes, we have broken them down into the following categories:
♦ Standard Operators - Arithmetic, logical, and relational operators such as +,
-, =, >=, and <=.
♦ Mathematical - Trigonometric and general mathematical functions.
♦ Financial - Common financial functions used to determine present values.
♦ Probability - Functions used to determine a wide range of probability and
statistical answers. Poisson and Erlang queuing functions are among those
provided.
♦ Variable Domain - Functions used to define the range of values (domain) a
variable can take on (e.g., lower and upper bounds or integer restrictions).
♦ Set Handling - Functions useful for manipulating sets.
♦ Set Looping - Looping functions used to perform an operation over a set
(e.g., to compute the sum, maximum, or minimum of a set of numbers).
♦ Import/Export - Functions used to create links to external data sources.
♦ Miscellaneous - Miscellaneous functions are listed under this heading.
In the remainder of this chapter, we will give an in-depth description of the operators and functions
available in LINGO.
Standard Operators
LINGO has three types of standard operators:
1. Arithmetic,
2. Logical, and
3. Relational.
285
286 CHAPTER 7
Arithmetic Operators
Arithmetic operators work with numeric operands. LINGO has five binary (two-operand) arithmetic
operators, shown here:
Operator Interpretation
^ Exponentiation
* Multiplication
/ Division
+ Addition
- Subtraction
Since these are binary operators, they require two arguments—one immediately to the left of the
operator and one immediately to the right.
The only unary (one-operand) arithmetic operator in LINGO is negation (-). In this case, the operator
applies to the operand immediately to the right of the negation sign.
These operators should be familiar to all readers. The priority of the operators is given in the
following:
Priority Level Operator(s)
Highest - (negation)
^
*/
Lowest +-
Operators with the highest priority are evaluated first, in order from left to right. As an example,
consider the expression:
4+6/2
The division operator (/) has higher priority than the addition operator (+). Thus, it is evaluated first,
leaving: 4 + 3. Evaluating the remaining addition gives a final result of 7.
The order of evaluation of the operators can be controlled with parentheses. LINGO evaluates the
equation in the innermost parentheses first and works out from there. If we recast the expression from
above as:
(4 + 6) / 2
we will now get a final result of 5, instead of 7. The 4 and 6 are added first because they appear in
parentheses. The resulting sum of 10 is divided by 2, giving the final result of 5.
Note: LINGO follows the Excel convention of assigning the highest priority to the negation
operator. Given this, LINGO evaluates -3^2 as positive 9. Some users may prefer to give the
unary minus operator a lower priority so that -3^2 evaluates to minus 9. You can do this by
setting the Unary Minus Priority option to Low via the Model Generator tab of the
LINGO|Options command. Once you set the unary minus operator’s priority is set to low its
priority will be lower than multiplication and division, but higher than addition and
subtraction.
OPERATORS AND FUNCTIONS 287
Logical Operators
Logical operators were used in Chapter 2, Using Sets, when we introduced set looping functions. In
LINGO, logical operators are primarily used in conditional expressions on set looping functions to
control which members of a set are to be included or excluded in the function. They also play a role in
building set membership conditions.
Logical operators return either TRUE or FALSE as a result. LINGO uses the value 1 to represent
TRUE, and the value 0 to represent FALSE. LINGO considers an argument to be FALSE if, and only
if, it is equal to 0. Thus, for example, arguments of 1, 7, -1, and .1234 would all be considered TRUE.
LINGO has nine logical operators, which are all binary with the single exception of the #NOT#
operator, which is unary. LINGO’s logical operators and their return values are listed below:
Logical Operator Return Value
#NOT# TRUE if the operand immediately to the right is FALSE, else
FALSE.
#EQ# TRUE if both operands are equal, else FALSE.
#NE# TRUE if both operands are not equal, else FALSE.
#GT# TRUE if the left operand is strictly greater than the right
operand, else FALSE.
#GE# TRUE if the left operand is greater-than-or-equal-to the right
operand, else FALSE.
#LT# TRUE if the left operand is strictly less than the right operand,
else FALSE.
#LE# TRUE if the left operand is less-than-or-equal-to the right
operand, else FALSE.
#AND# TRUE only if both arguments are TRUE, else FALSE.
#OR# FALSE only if both its arguments are FALSE, else TRUE.
The priority ranking of the logical operators is:
Priority Level Operator(s)
Highest #NOT#
#EQ# #NE# #GT# #GE# #LT# #LE#
Lowest #AND# #OR#
Relational Operators
In LINGO, relational operators are used in a model to specify whether the left-hand side of an
expression should be equal to, less-than-or-equal-to, or greater-than-or-equal-to the right-hand side.
Relational operators are used to form the constraints of a model. Relational operators are distinct from
the logical operators #EQ#, #LE#, and #GE#, in that they tell LINGO the optimal solution of the
model must satisfy the direction of the relational operator. Logical operators, on the other hand, merely
report whether or not a condition is satisfied.
Relational operators have the lowest priority of all the operators.
288 CHAPTER 7
The three relational operators are described below:
Relational Operator Interpretation
= The expression to the left must equal the one on the right.
<= The expression to the left must be less-than-or-equal-to the
expression on the right
>= The expression to the left must be greater-than-or-equal-to
the expression on the right
LINGO will also accept “<” for less-than-or-equal-to, and “>” for greater-than-or-equal-to.
Note: LINGO does not directly support strictly less than and strictly greater than relational
operators. In general, it would be unusual to find a good formulation that requires such a
feature. However, if you want A to be strictly less than B:
A < B,
then convert this expression to an equivalent less-than-or-equal-to expression as follows:
A + e ≤ B,
where e is a small constant term whose value is dependent upon how much A must be “less
than” B in order for you to consider them to be “not equal”.
Mathematical Functions
LINGO offers a number of standard, mathematical functions. These functions return a single result
based on one or more scalar arguments. These functions are listed below:
@ABS(X)
This returns the absolute value of X.
@COS(X)
This returns the cosine of X, where X is an angle in radians.
@EXP(X)
This returns e (i.e., 2.718281 ...) raised to the power X.
@FLOOR(X)
This returns the integer part of X. To be specific, if X ≥ 0, @FLOOR returns the
largest integer, I, such that I ≤ X. If X is negative, @FLOOR returns the most
negative integer, I, such that I ≥ X.
@LGM(X)
This returns the natural (base e) logarithm of the gamma function of X (i.e., log
of (X - 1)!). It is extended to noninteger values of X by linear interpolation.
@LOG(X)
This returns the natural logarithm of X.
@MOD( X,Y)
This returns the value of X modulo Y, or, in other words, the remainder of an integer divide
of X by Y.
@POW( X,Y)
This returns the value of X rasied to the Y power.
@SIGN(X)
This returns -1 if X < 0. Otherwise, it returns +1.
@SIN(X)
This returns the sine of X, where X is the angle in radians.
@SMAX (X1, X2, ..., XN)
This returns the maximum value of X1, X2, ..., and XN.
@SMIN(X1, X2, ..., XN)
This returns the minimum value of X1, X2, ..., and XN.
@SQR( X)
This returns the value of X squared.
@SQRT( X)
This returns the square root of X.
@TAN(X)
This returns the tangent of X, where X is the angle in radians.
290 CHAPTER 7
Financial Functions
LINGO currently offers two financial functions. One computes the present value of an annuity. The
other returns the present value of a lump sum.
@FPA(I, N)
This returns the present value of an annuity. That is, a stream of $1 payments per period at an interest
rate of I for N periods starting one period from now. I is not a percentage, but a fraction representing
the interest rate (e.g., you would use .1 to represent 10%). To get the present value of an annuity
stream of $X payments, multiply the result by X.
@FPL(I, N)
This returns the present value of a lump sum of $1 N periods from now if the interest rate is I per
period. I is not a percentage, but a fraction representing the interest rate (e.g., you would use .1 to
represent 10%). To get the present value of a lump sum of $X, multiply the result by X.
Probability Functions
LINGO has a number of probability related functions. There are examples that make use of most of
these functions in Chapter 12, Developing More Advanced Models, and in Appendix A, Additional
Examples of LINGO Modeling.
@NORMSINV( P)
This is the inverse of the standard normal cumulative distribution. Given a probability, P, this function
returns the value Z such that the probability of a normally distributed random variable with standard
deviation of 1 being less-than-or-equal to Z is P.
@PBN(P, N, X)
This is the cumulative binomial probability. It returns the probability that a sample of N items, from a
universe with a fraction of P of those items defective, has X or less defective items. It is extended to
noninteger values of X and N by linear interpolation.
@PCX(N, X)
This is the cumulative distribution function for the Chi-squared distribution with N degrees of
freedom. It returns the probability that an observation from this distribution is less-than-or-equal-to X.
@PEB(A, X)
This is Erlang’s busy probability for a service system with X servers and an arriving load of A, with
infinite queue allowed. The result of @PEB can be interpreted as either the fraction of time all servers
are busy or the fraction of customers that must wait in the queue. It is extended to noninteger values of
X by linear interpolation. The arriving load, A, is the expected number of customers arriving per unit of
time multiplied by the expected time to process one customer.
OPERATORS AND FUNCTIONS 291
@PEL(A, X)
This is Erlang’s loss probability for a service system with X servers and an arriving load of A, no queue
allowed. The result of @PEL can be interpreted as either the fraction of time all servers are busy or the
fraction of customers lost due to all servers being busy when they arrive. It is extended to noninteger
values of X by linear interpolation. The arriving load, A, is the expected number of customers arriving
per unit of time multiplied by the expected time to process one customer.
@PFD(N, D, X)
This is the cumulative distribution function for the F distribution with N degrees of freedom in the
numerator and D degrees of freedom in the denominator. It returns the probability that an observation
from this distribution is less-than-or-equal-to X.
@PFS(A, X, C)
This returns the expected number of customers waiting for or under repair in a finite source Poisson
service system with X servers in parallel, C customers, and a limiting load A. It is extended to
noninteger values of X and C by linear interpolation. A, the limiting load, is the number of customers
multiplied by the mean service time divided by the mean repair time.
@PHG(POP, G, N, X)
This is the cumulative hypergeometric probability. It returns the probability that X or fewer items in
the sample are good, given a sample without replacement of N items from a population of size POP
where G items in the population are good. It is extended to noninteger values of POP, G, N, and X by
linear interpolation.
@PPL(A, X)
This is the linear loss function for the Poisson distribution. It returns the expected value of MAX(0,
Z-X), where Z is a Poisson random variable with mean value A.
@PPS(A, X)
This is the cumulative Poisson probability distribution. It returns the probability that a Poisson random
variable, with mean value A, is less-than-or-equal-to X. It is extended to noninteger values of X by
linear interpolation.
@PSL(X)
This is the unit normal linear loss function. It returns the expected value of MAX(0, Z-X), where Z is a
standard normal random variable. In inventory modeling, @PSL(X) is the expected amount that
demand exceeds a level X, if demand has a standard normal distribution.
@PSN(X)
This is the cumulative standard normal probability distribution. A standard normal random variable
has mean 0.0 and standard deviation 1.0 (the bell curve, centered on the origin). The value returned by
@PSN is the area under the curve to the left of the point on the ordinate indicated by X.
@PTD(N, X)
This is the cumulative distribution function for the t distribution with N degrees of freedom. It returns
the probability that an observation from this distribution is less-than-or-equal-to X.
292 CHAPTER 7
@QRAND(SEED)
The @QRAND function produces a sequence of “quasi-random” uniform numbers in the interval (0,
1). @QRAND is only permitted in a data section. It will fill an entire attribute with quasi-random
numbers. Generally, you will be filling two-dimensional tables with, say, m rows and n variables. m
represents the number of scenarios, or experiments, you want to run. n represents the number of
random variables you need for each scenario or experiment. Within a row, the numbers are
independently distributed. Among rows, the numbers are “super uniformly” distributed. That is, the
numbers are more uniformly distributed than you would expect by chance. These numbers are
generated by a form of “stratified sampling”.
For example, suppose m = 4 and n = 2. Even though the numbers are random, you will find that
there will be exactly one row in which both numbers are in the interval (0, .5), exactly one row in
which both numbers are in (.5, 1), and two rows in which one number is less than .5 and the other
is greater than .5. Using @QRAND allows you to get much more accurate results for a given
number of random numbers in a Monte Carlo model. If you want 8 ordinary random numbers,
then use @QRAND(1,8) rather than @QRAND(4,2). An example of @QRAND follows:
MODEL:
DATA:
M = 4;
N = 2;
SEED = 1234567;
ENDDATA
SETS:
ROWS /1..M/;
COLS /1..N/;
TABLE( ROWS, COLS): X;
ENDSETS
DATA:
X = @QRAND( SEED);
ENDDATA
END
Example of @QRAND function
If you don’t specify a seed value for @QRAND, then LINGO will use the system clock to construct a
seed value.
@RAND(SEED)
This returns a pseudo-random number between 0 and 1, depending deterministically on SEED.
OPERATORS AND FUNCTIONS 293
SETS:
PLANTS / SEATTLE, DENVER,
CHICAGO, ATLANTA/:;
CLOSED( PLANTS) /DENVER/:;
OPEN( PLANTS) |
#NOT# @IN( CLOSED, &1):;
ENDSETS
294 CHAPTER 7
The OPEN set is derived from the PLANTS set. We use a membership condition containing
the @IN function to allow only those plants not contained in the CLOSED set to be in the
OPEN set.
Example 2:
In this example, we illustrate how to determine if the set element (B, Y) belongs to the derived
S3 set. In this case, (B, Y) is indeed a member of S3, so X will be set to 1. Note that, in order
to get the index of the primitive set elements B and Y, we made use of the @INDEX function,
which is discussed next.
SETS:
S1 / A B C/:;
S2 / X Y Z/:;
S3( S1, S2) / A,X A,Z B,Y C,X/:;
ENDSETS
X = @IN( S3, @INDEX( S1, B), @INDEX( S2, Y));
Now, suppose you want to get the index of the boy named Sue within the set BOYS. The value
of this index should be 3. Simply using @INDEX(SUE) would return 2 instead of 3, because
LINGO finds SUE in the GIRLS set first. In this case, to get the desired result, you must
specify the set BOYS as an argument and enter @INDEX(BOYS, SUE).
OPERATORS AND FUNCTIONS 295
Example 2:
@INDEX may also be used to return the index of a set member of a derived set. In this
case, assuming an n-dimensional derived set, set_member would consist of n primitive
set members separated by commas as illustrated for the 2-dimensional set rxc below:
SETS:
ROWS /R1..R27/;
COLS /C1..C3/;
RXC( ROWS, COLS): XRNG;
ENDSETS
! return the index of (r1,c3) in the rxc set;
NDX = @INDEX( RXC, R1, C3);
@WRAP(INDEX, LIMIT)
This allows you to “wrap” an index around the end of a set and continue indexing at the other end of
the set. That is, when the last (first) member of a set is reached in a set looping function, use of
@WRAP will allow you to wrap the set index to the first (last) member of the set. This is a particularly
useful function in cyclical, multiperiod planning models.
Formally speaking, @WRAP returns J such that J = INDEX - K * LIMIT, where K is an integer such
that J is in the interval [1-LIMIT]. Informally speaking, @WRAP will subtract or add LIMIT to INDEX
until it falls in the range 1 to LIMIT.
For an example on the use of the @WRAP function in a staff-scheduling model, refer to the Primitive
Set Example section in Chapter 2, Using Sets.
@SIZE(set_name)
This returns the number of elements in the set set_name. Using the @SIZE function is preferred to
explicitly listing the size of a set in a model. This serves to make your models more data independent
and, therefore, easier to maintain should the size of your sets change.
To view an example of the @SIZE function, refer to the PERT/CPM example in the Sparse Derived
Set Example - Explicit List section of Chapter 2, Using Sets.
Set looping functions are discussed in more detail in Chapter 2, Using Sets.
Interface Functions
Interface functions allow you to link your model to external data sources such as text files, databases,
spreadsheets and external applications. With the exception of @FILE, interface functions are valid
only in sets and data sections, and may not be used in calc and model sections.The interface
functions currently available in LINGO are listed below.
@FILE( 'filename')
The @FILE function allows you to include data from external text files anywhere in your model,
where filename is the name of the file to include text form. This is particularly useful for
incorporating data stored in text files in your sets and data sections.
When this function is encountered in a model, LINGO will continue to take text from this file
until it encounters either the end-of-file or a LINGO end-of-record mark (~). For subsequent
@FILE references in the same model that use the same file name, LINGO resumes taking input
from the file at the point where it left off. Nesting of @FILE function calls (embedding an @FILE
in a file which is itself called by @FILE) is not allowed.
For more information on use of the @FILE function, refer to Interfacing with External Files.
OPERATORS AND FUNCTIONS 297
@POINTER( N)
This function is strictly for use with the LINGO Dynamic Link Library (DLL) under Windows.
@POINTER allows you to transfer data directly through shared memory locations. For more
information on the use of the @POINTER function, refer to Interfacing with Other Applications.
For additional documentation on the @TEXT function, see Interfacing with External Files.
298 CHAPTER 7
Report Functions
Report functions are used to construct reports based on a model’s results, and are valid on both calc
and data sections. Combining report functions with interface functions in a data section allows you to
export the reports to text files, spreadsheets, databases, or your own calling application.
Note: The interested reader will find an exhaustive example of the use of report functions in the
sample model TRANSOL.LG4 in the Samples subfolder. This model makes extensive use of
all the report functions to mimic the standard LINGO solution report.
@DUAL ( variable_or_row_name)
The @DUAL function outputs the dual value of a variable or a row. For example, consider a
model with the following data section:
DATA:
@TEXT( 'C:\RESULTS\OUTPUT.TXT') =
@WRITEFOR( SET1( I): X( I), @DUAL( X( I)), @NEWLINE(1));
ENDDATA
When this model is solved, the values of attribute X and their reduced costs will be written to the file
C:\RESULTS\OUTPUT.TXT. Output may be routed to a file, spreadsheet, database or memory
location. The exact destination will depend on the export function used on the left-hand side of the
output statement.
If the argument to the @DUAL function is a row name, then the dual price on the generated row will
be output.
@FORMAT( value, format_descriptor)
@FORMAT may be used in @WRITE and @WRITEFOR statements to format a numeric value for
output as text, where value is the numeric quantity to be formatted, and format_descriptor is a
string detailing how the number is to be formatted. The format descriptor is interpreted using C
programming conventions. For instance, a format descriptor of ‘12.2f’ would cause the number to
be printed in a field of 12 characters with 2 digits after the decimal point. You can refer to a C
reference manual for more details on the available formatting options.
Note: The @FORMAT function converts numeric values to text. Therefore, you will probably not
want to use it when exporting values to programs that require numeric values (e.g.,
workbooks or databases).
OPERATORS AND FUNCTIONS 299
The following example uses the @FORMAT function to place a shipping quantity into a field of
eight characters with no trailing decimal value:
DATA:
@TEXT() = @WRITE( ' From To Quantity',
@NEWLINE(1));
@TEXT() = @WRITE( '--------------------------',
@NEWLINE(1));
@TEXT() = @WRITEFOR( ROUTES( I, J) | X( I, J) #GT# 0:
3*' ', WAREHOUSE( I), 4*' ',
CUSTOMER( J), 4*' ', @FORMAT( X( I, J), '8.0f'),
@NEWLINE( 1));
ENDDATA
From To Quantity
--------------------------
WH1 C1 2
WH1 C2 17
WH1 C3 1
WH2 C1 13
WH2 C4 12
WH3 C3 21
This next example, GRAPHPSN.LG4, graphs the standard normal function, @PSN, over the
interval [ –2.4, 2.4]. The @FORMAT function is used to print the X coordinates with one trailing
decimal point.
300 CHAPTER 7
DATA:
@TEXT() = @WRITE('Iterations= ', @ITERS());
ENDDATA
@NAME( var_or_row_reference)
Use @NAME to return the name of a variable or row as text. @ITERS is available only in the data
and calc sections, and is not allowed in the constraints of a mode. The following example prints a
variable name and its value:
DATA:
@TEXT() = @WRITEFOR( ROUTES( I, J) |
X( I, J) #GT# 0: @NAME( X), ' ', X, @NEWLINE(1));
ENDDATA
X( WH1, C1) 2
X( WH1, C2) 17
X( WH1, C3) 1
X( WH2, C1) 13
X( WH2, C4) 12
X( WH3, C3) 21
@NEWLINE( n)
Use @NEWLINE to write n new lines to the output device. @NEWLINE is available only in the
data and calc sections, and is not allowed in the constraints of a model. See the example
immediately below in the @RANGED section.
@RANGED( variable_or_row_name)
@RANGED outputs the allowable decrease on a specified variable’s objective coefficient or on a
specified row’s right-hand side. @RANGED is available only in the data and calc sections, and is
not allowed in the constraints of a model. For example, consider a model with the following data
section:
DATA:
@TEXT( 'C:\RESULTS\OUTPUT.TXT') =
@WRITEFOR( SET( I): X( I), @RANGED( X( I), @NEWLINE(1));
ENDDATA
When this model is solved, the values of attribute X and the allowable decreases on its objective
coefficients will be written to the file C:\RESULTS\OUTPUT.TXT. If @RANGED is passed a row
name it will output the allowable decrease on the right-hand side value for the row. Output may
OPERATORS AND FUNCTIONS 303
be routed to a file, spreadsheet, database, or memory location. The exact destination will depend
on the export function used on the left-hand side of the output statement. Range computations
must be enabled in order for @RANGED to function properly. For more information on the
interpretation of allowable decreases, refer to the LINGO|Range command.
@RANGEU( variable_or_row_name)
@RANGEU outputs the allowable increase on a specified variable’s objective coefficient or on a
specified row’s right-hand side. For example, consider a model with the following data section:
DATA:
@TEXT( 'C:\RESULTS\OUTPUT.TXT') = @WRITEFOR( SET( I):
X, @RANGEU(X), @NEWLINE(1));
ENDDATA
When this model is solved, the values of X and the allowable increases on its objective
coefficients will be written to the file C:\RESULTS\OUTPUT.TXT. If @RANGEU is passed a row
name it will output the allowable increase on the right-hand side value for the row. Output may be
routed to a file, spreadsheet, database, or memory location. The exact destination will depend on
the export function used on the left-hand side of the output statement. Range computations must
be enabled in order for @RANGED to function properly. For more information on the
interpretation of allowable increases, refer to the LINGO|Range command.
304 CHAPTER 7
@STATUS()
This returns the final status of the solution process using the following codes:
@STATUS()
Code Interpretation
In general, if @STATUS does not return a code of 0, the solution is of little use and should not be
trusted. The @STATUS function is available only in the data and calc sections. The @STATUS
function is not allowed in the constraints of a model.
OPERATORS AND FUNCTIONS 305
For example, the following output statement uses @STATUS to print a message to let the user
know if the solution is globally optimal, or not:
DATA:
@TEXT() = @WRITE( @IF( @STATUS() #EQ# 0,
'Global solution found',
'WARNING: Solution *not* globally optimal!');
ENDDATA
For additional examples of the use of the @STATUS function, refer to Interfacing with Other
Applications.
@STRLEN( string)
Use @STRLEN to get the length of a specified string. This can be a useful feature when
formatting reports. As an example, @STRLEN( ‘123’) would return the value 3. @STRLEN is
available only in the data and calc sections, and is not allowed in the constraints of a model.
@TABLE( ‘attr|set’)
The @TABLE function is used to display either an attribute’s values or a set’s members in tabular
format. The @TABLE function is available only in the data section of a model. You can refer to
either QUEENS8.LG4 or PERT.LG4 for examples of @TABLE. These models can be found in
the SAMPLES folder off the main LINGO folder.
For instance, QUEENS8.LG4 is a model for positioning eight queens on a chessboard so that no
one queen can attack another. At the end of this model you will find the following data section:
DATA:
@TEXT() = ' The final chessboard:';
@TEXT() = @TABLE( X);
ENDDATA
Here we are using the @TABLE function to display the X attribute in the standard output window
via the @TEXT interface function (see below for more on @TEXT). The X attribute is an 8-by-8
table of 0’s and 1’s indicating if a queen is positioned in a given square of the chessboard, or not.
The output generated by @TABLE in this instance follows:
Note that all eight queens (indicated by the 1’s on the board) are safe from attack.
306 CHAPTER 7
In addition to displaying attributes, @TABLE can also display sets. When displaying a set,
@TABLE will print the letter X in the table cell if the corresponding set member exists, otherwise
it will leave the table cell blank.
The PERT.LG4 sample model is a project scheduling model. A project consists of many tasks,
and some tasks must be completed before others can begin. The list of all the tasks that must
precede certain other tasks is called the precedence relations. At the end of PERT4.LG4 there is
the following data section which uses @TABLE to display the precedence relations set, PRED:
DATA:
!Use @TABLE() to display the precedence relations set,
PRED;
@TEXT() = @TABLE( PRED);
ENDDATA
Whenever one task must precede another, an ‘X’ appears in the particular cell of the table. So, for
instance, the DESIGN task must precede the FORECAST and SURVEY tasks.
If a line of a table exceeds the page width setting in Lingo, it simply gets wrapped around. So, if
you want to display wide tables without line wraps, you may need to increase the page width.
Note: Currently, @TABLE can only send tables to the standard solution window or to text files. In
other words, it is not possible to send tables to Excel, databases or to applications calling the
LINGO DLL.
OPERATORS AND FUNCTIONS 307
@TABLE can also display sets and attributes of more than two dimensions. In fact, @TABLE allows
you great control over how multidimensional objects get displayed. Specifically, four forms of
@TABLE are supported:
♦ @TABLE( attr/set) – This is the simplest form of @TABLE. If the object is of one
dimension, it will be displayed as a column. Otherwise, the first n-1 dimensions will
be displayed on the vertical axis, while the n-th dimension will be displayed on the
horizontal axis of the table. An example follows:
@TABLE( X)
DATA:
@TEXT() = @WRITE( 'The ratio of X to Y is: ', X / Y);
ENDDATA
308 CHAPTER 7
@WRITEFOR( setname[ ( set_index_list) [ | cond_qualifier]]: obj1[, …, objn])
Use @WRITEFOR to output one or more objects across a set. @WRITEFOR is available only in the
data and calc sections, and is not allowed in the constraints of a model
@WRITEFOR operates like the other set looping functions in that you may, or may not, specify an
index list and a conditional qualifier. The ability to use a conditional qualifier allows great flexibility
in specifying exactly what you wish to display in your reports. @WRITEFOR may also be used to
display computations that are a function of variable values.
Using @WRITEFOR in conjunction with export functions in data sections allows you to route output
to a file, spreadsheet, or database. The exact destination will depend on the export function used on the
left-hand side of the output statement.
As an example, the output statement below prints the number of units to ship, the warehouse name,
and the customer name for each route with nonzero volume:
DATA:
@TEXT() = @WRITEFOR( ROUTES( I, J) | X( I, J) #GT# 0:
'Ship ', X( I, J), ' units from warehouse ', WAREHOUSE( I),
' to customer ', CUSTOMER( J), @NEWLINE( 1));
ENDDATA
DATA:
LEAD = 3;
@TEXT() = 'Staff on duty graph:';
@TEXT() = @WRITEFOR( DAY( D): LEAD*' ',
DAY( D), ' ', ON_DUTY( D), ' ', ON_DUTY( D)*'+',
@NEWLINE(1)
);
ENDDATA
OPERATORS AND FUNCTIONS 309
The graph would appear as follows, with one plus sign displayed for each staff member on duty:
Miscellaneous Functions
@IF (logical_condition, true_result, false_result)
The @IF function evaluates logical_condition and, if true, returns true_result. Otherwise, it returns
false_result. For example, consider the following simple model that uses @IF to compute fixed
production costs:
MIN = COST;
COST = XCOST +YCOST;
XCOST = @IF( X #GT# 0, 100, 0) + 2 * X;
YCOST = @IF( Y #GT# 0, 60, 0) + 3 * Y;
X + Y >= 30;
Model: IFCOST
We produce two products—X and Y. We want to minimize total cost, subject to producing at least 30
total units of X and Y. If we produce X, there is a fixed charge of 100 along with a variable cost of 2.
Similarly, for Y, these respective values are 60 and 3. We use the @IF function to determine if either
of the products are being produced in order to apply the relevant fixed cost. This is accomplished by
testing to see if their production levels are greater than 0. If so, we return the fixed cost value.
Otherwise, we return zero.
Experienced modelers know that, without the benefit of an @IF function, modeling fixed costs
requires invoking some “tricks” using binary integer variables. The resulting models are not as
intuitive as models constructed using @IF. However, the caveat is that the @IF function is not a linear
function. At best, the graph of an @IF function will be piecewise linear. In our current example, the
@IF functions are piecewise linear with a discontinuous break at the origin.
It is always best to try and keep a model linear (see Chapter 14, On Mathematical Modeling). Barring
this, it is best for all functions in a nonlinear model to be continuous. The @IF function violates both
these conditions. Thus, models containing @IF functions may be tough to solve to global optimality.
Fortunately, LINGO has two options that can help overcome the difficult nature of models containing
@IF functions—linearization and global optimization.
310 CHAPTER 7
To illustrate the difficulty in solving models with discontinuous functions such as @IF, we will solve
our example model with both linearization and global optimization disabled. When we do this, we get
the following solution:
Local optimal solution found at iteration: 42
Objective value: 160.0000
Variable Value
COST 160.0000
XCOST 160.0000
YCOST 0.000000
X 30.00000
Y 0.000000
This solution involves producing only X at a total cost of 160. Given that producing only Y and not X
will result in a lower total cost of 150, this is clearly a locally optimal point. In order to find the
globally optimal point, we must resort to either the linearization or global optimization features in
LINGO.
Briefly, linearization seeks to reformulate a nonlinear model into a mathematically equivalent linear
model. This is desirable for two reasons. First, linear models can always be solved to global optimality.
Secondly, linear models will tend to solve much faster than equivalent nonlinear models.
Unfortunately, linearization can not always transform a model into an equivalent linear state. In which
case, it may be of no benefit. Fortunately, our sample model can be entirely linearized. To enable the
linearization option, run the LINGO|Options command and set the Linearization Degree to High on the
General Solver tab.
Global optimization breaks a model down into a series of smaller, local models. Once this series of
local models has been solved, a globally optimal solution can be determined. To enable global
optimization, run the LINGO|Options command, select the Global Solver tab, then click on the Global
Solver checkbox. Note that the global solver is an add-on option to LINGO. The global solver feature
will not be enabled for some installations. Run the Help|About LINGO command to determine if your
installation has the global solver capability enabled.
Whether using the linearization option or the global solver, LINGO obtains the true, global solution:
Global optimal solution found at iteration: 6
Objective value: 150.0000
Variable Value
COST 150.0000
XCOST 0.000000
YCOST 150.0000
X 0.000000
Y 30.00000
Note: Starting with release 9.0, the false branch of the @IF function may contain arithmetic errors
without causing the solver to trigger an error. This makes the @IF function useful in
avoiding problems when the solver strays into areas where certain functions become
undefined. For instance, if your model involves division by a variable, you might use @IF as
follows: @IF( X #GT# 1.E-10, 1/X, 1.E10).
OPERATORS AND FUNCTIONS 311
@WARN(‘text’, logical_condition)
This displays the message ‘text’ if the logical_condition is met. This feature is useful for verifying the
validity of a model’s data. In the following example, if the user has entered a negative interest rate, the
message “INVALID INTEREST RATE” is displayed:
! A model of a home mortgage;
DATA:
! Prompt the user for the interest
rate, years, and value of mortgage.
We will compute the monthly payment;
YRATE = ?;
YEARS = ?;
LUMP = ?;
ENDDATA
! Number of monthly payment;
MONTHS = YEARS * 12;
! Monthly interest rate;
(1 + MRATE) ^ 12 = 1 + YRATE;
! Solve next line for monthly payment;
LUMP = PAYMENT * @FPA(MRATE, MONTHS);
! Warn them if interest rate is negative
@WARN('INVALID INTEREST RATE',
YRATE #LT# 0);
@USER (user_determined_arguments)
The user can supply this in an external DLL or object code file. For a detailed example on the use of
@USER, please see the User Defined Functions section in Chapter 11, Interfacing with Other
Applications.
8 Interfacing with External
Files
It can be cumbersome and impractical to try to maintain your data in a LINGO model file. In most
cases, your model’s data will reside externally in text files, spreadsheets, and databases. Also, a
solution generated by LINGO is of little use if you can’t export it to other applications. For these
reasons, LINGO has many methods to assist you in moving information in and out of the application.
The primary focus of this chapter is to illustrate how to move data in and out of LINGO through the
use of text based ASCII files. In Chapter 9, Interfacing with Spreadsheets, we will look at using
spreadsheets. In Chapter 10, Interfacing with Databases, we will illustrate the use of databases for
maintaining your model’s data.
313
314 CHAPTER 8
Suppose your staffing requirements data is maintained in an Excel worksheet resembling the
following:
To paste the staffing requirements data from Excel into the LINGO model above, follow these steps:
1. Select the range containing the data (C3:I3) by placing the cursor on the C3
cell, press and hold down the left mouse button, drag the mouse to cell I3,
then release the mouse button.
2. Select the Copy command from Excel’s Edit Menu.
3. Click once on the LINGO model window.
4. Place the LINGO cursor directly to the right of the data statement:
REQUIRED =.
5. Select the Paste command from LINGO’s Edit menu.
The data should now appear in the LINGO model as follows:
DATA:
REQUIRED = 20 16 13 16 19 14 12;
ENDDATA
You may need to adjust the font of the data to your liking. You can use the Edit|Select Font command
in LINGO to accomplish this. Your model now has the required data and is ready to be solved. Note
that LINGO also has features that allow you to import data directly from Excel. See Chapter 9,
Interfacing with Spreadsheets, for more information.
Note: We use the convention of placing the extension of .LDT on all LINGO data files.
Sections of the data file between end-of-record marks (~) are called records. If an included file has no
end-of-record marks, LINGO reads the whole file as a single record. Notice that, with the exception of
the end-of-record marks, the model text and data appear just as they would if they were in the model
itself.
Also, notice how the end-of-record marks in the include file work along with the @FILE function calls
in the model. The first call to @FILE opens WIDGETS2.LDT and includes the first record. The second
call includes the second record, and so on.
The last record in the file does not need an end-of-record mark. When LINGO encounters an
end-of-file, it includes the last record and closes the file. If you end the last record in an include file
with an end-of-record mark, LINGO will not close the file until it is done solving the current model.
This could cause problems if multiple data files are opened in the model—files that remain open can
cause the limit on open files to be exceeded.
When using the @FILE function, think of the contents of the record (except for any end-of-record
mark) as replacing the text @FILE(‘filename’) in the model. This way, you can include a whole
statement, part of a statement, or a whole series of statements in a record. For example, the first two
records of the WIDGETS2.LDT file in the above example:
!List of warehouses;
WH1 WH2 WH3 WH4 WH5 WH6 ~
!List of vendors;
V1 V2 V3 V4 V5 V6 V7 V8 ~
are included in the model in the sets section as follows:
WAREHOUSES / @FILE('WIDGETS2.LDT')/: CAPACITY;
VENDORS / @FILE('WIDGETS2.LDT')/ : DEMAND;
INTERFACING WITH EXTERNAL FILES 319
The net effect of these @FILE calls is to turn the model statements into:
WAREHOUSES / WH1 WH2 WH3 WH4 WH5 WH6/: CAPACITY;
VENDORS / V1 V2 V3 V4 V5 V6 V7 V8/ : DEMAND;
Comments in the include file are ignored. The maximum number of include files a model can
simultaneously reference is 16.
DATA:
REQUIRED = 20 16 13 16 19 14 12;
@TEXT('OUT.TXT') = DAYS, START;
ENDDATA
@FOR(DAYS(J):
@SUM(DAYS(I) | I #LE# 5:
START(@WRAP(J - I + 1, 7)))
>= REQUIRED(J)
);
To import the data into an Excel sheet, you must first use the File|Open command on the OUT.TXT file
to get the data into a spreadsheet by itself, as we have done here:
Once the results are imported into a spreadsheet, you may cut and paste them to any other sheet you
desire.
Before we move on, suppose you are not interested in all the output generated by the standard LINGO
solution report. Suppose, in the case of this example, all you want to see is the objective value and the
values for the DAYS set and the START attribute. Here's how you can do it. First, add the additional
output operation, shown here in bold, to your data section, so it looks like:
DATA:
@TEXT('OUT.TXT') = DAYS, START;
@TEXT() = DAYS, START;
ENDDATA
The new output operation causes the values of DAYS and START to be sent to the screen (since we
omitted a file name). Next, you will need to suppress the normal LINGO solution report. In Windows
versions of LINGO, select the LINGO|Options command, click on the Interface tab in the Options
dialog box, check the Terse output checkbox, then press the OK button (on platforms other than
322 CHAPTER 8
Windows, enter the TERSE command). Now, solve your model and you will be presented with the
following, abbreviated report:
Global optimal solution found at step: 8
Objective value: 22.00000
MON 8.0000000
TUE 2.0000000
WED 0.0000000
THU 6.0000000
FRI 3.0000000
SAT 3.0000000
SUN 0.0000000
In the example above, we simply listed the names of set DAYS and attribute START on the right-hand
side of our output operation. This causes LINGO to display all values for DAYS and START. Suppose
we'd like more control over what values do and do not get displayed. In particular, suppose we are
only interested in viewing those days in which the value for START is nonzero. We can do this by
using the @WRITEFOR report function, which allows us to provide a condition to test before printing
output:
DATA:
@TEXT( ) = @WRITEFOR( DAYS( D) | START( D) #GT# 0:
DAYS( D), @FORMAT( START( D), '6.1f'));
ENDDATA
Note how we now only display the days where START > 0 in our new report:
Global optimal solution found at iteration: 15
Objective value: 22.00000
MON 8.0
TUE 2.0
THU 6.0
FRI 3.0
SAT 3.0
Another feature of this last example to note is the use of the @FORMAT function, which we used to
display the nonzero start values in a field of six columns with one trailing decimal point.
@WRITEFOR also allows us to form arithmetic expressions of the variable values. Here's an example
that we could add to our staff model to compute the number of staff on duty each day:
DATA:
@TEXT( ) = @WRITE( 'Day On-Duty');
@TEXT( ) = @WRITE( 14*'-');
@TEXT( ) = @WRITEFOR( DAYS( D): DAYS(D),
@FORMAT( @SUM( DAYS( D2) | D2 #LE# 5:
START( @WRAP( D - D2 + 1, 7))), '11.1f'));
ENDDATA
INTERFACING WITH EXTERNAL FILES 323
Here's the report generated by these output operations:
Day On-Duty
--------------
MON 20.0
TUE 16.0
WED 13.0
THU 16.0
FRI 19.0
SAT 14.0
SUN 12.0
This previous example also illustrates the use of the @WRITE function. The @WRITE function is
similar to the @WRITEFOR function with the exception that it does not accept a set to loop on, and,
therefore, is used to write single occurrences of text output. As with @WRITEFOR, @WRITE accepts
expressions of variables. Here's an example that calculates the maximum number of employees
starting on a particular day.
DATA:
@TEXT() = @WRITE( 'Max start = ', @MAX( DAYS: START));
ENDDATA
We use the @FILE function to include the staffing requirements from an external file and we use the
@TEXT function to send the values of the START attribute to a file.
After the END statement, we have a GO command to solve the model for the Pluto stand. We then
include an ALTER command to change all occurrences of PLUTO with MARS. This command will
change the data section to (changes in bold):
DATA:
REQUIRED = @FILE('MARS.LDT');
@TEXT('MARS.TXT') = START;
ENDDATA
Assuming we have the staffing requirements for the Mars stand in the file MARS.LDT, our model is
then ready to run again. However, this time it will solve for the START values for the Mars hot dog
stand. We include commands to do the same for the Saturn location as well. Finally, we have two SET
commands to restore the modified parameters.
You can run this command script by issuing the File|Take Commands command in Windows versions
of LINGO, or you can use the TAKE command in other versions. Once the command script has been
executed, you will find the three solution files: PLUTO.TXT, MARS.TXT, and SATURN.TXT. These
files will contain the optimal values for the START attribute for the three locations.
To edit the command-line, you must right click on this icon and then select the Properties command.
You will then see the dialog box:
In the Target edit box, add the command -tTEST.LTF. If you want LINGO to run without opening up a
window, you can also select the Minimized option from the Run list box. Now, click the Apply
button followed by the OK button.
328 CHAPTER 8
You can now run LINGO and have it execute the script file by double clicking the shortcut icon on the
desktop. Once you have done this, the solution file, SOLU.TXT, should contain:
Variable Value Reduced Cost
X 50.00000 0.000000
Y 35.00000 0.000000
Row Slack or Surplus Dual Price
1 2050.000 1.000000
2 0.000000 5.000000
3 25.00000 0.000000
4 0.000000 15.00000
File: SOLU.TXT
331
332 CHAPTER 9
Importing in the Data and Init Sections with @OLE
The syntax for using @OLE to import data in both the data and init sections is:
object_list = @OLE([‘spreadsheet_file’] [, range_name_list]);
The object_list is a list of model objects, optionally separated by commas, which are to be initialized
from the spreadsheet. Object_list may contain any combination of set names, set attributes, and scalar
variables.
The spreadsheet_file is the name of the Excel spreadsheet file to retrieve the data from. If the name is
omitted, LINGO defaults to using whatever workbook is currently open in Excel.
The range_name_list is the list of named ranges in the sheet to retrieve the data from. The ranges must
contain exactly one element for each member in the object_list. There are three options available in
how you specify the ranges. First, you can omit the range arguments entirely. In which case, LINGO
defaults to using a list of range names identical to the list of object names in object_list. Second, you
can specify a single range name to retrieve all the data from. In which case, all the objects in
object_list must be defined on the same set, and LINGO reads the data as if it was a table. When
specifying a single range name for multiple objects, all the objects must be of the same data type.
Furthermore, you can’t mix set members (text) with set attributes (numeric). Finally, you can specify
one range name for each object in object_list. In which case, the objects do not have to be defined on
the same set and can be of differing data types. Examples of these three methods for using @OLE
follow:
Example 1: COST, CAPACITY = @OLE();
In this example we specify no arguments to the @OLE() function. In which case, LINGO
will supply the default arguments. Given that no workbook name was specified, LINGO
will use whatever workbook is currently open and active in Excel. In addition, no range
names were specified, so LINGO defaults to using the names of the model objects. Thus,
COST and CAPACITY are initialized to the values found, respectively, in the ranges COST
and CAPACITY in the currently open workbook in Excel.
Note: If you do not specify a workbook name in @OLE() function references, LINGO defaults to
using whatever workbook is currently open in Excel. Therfore, you will need to open Excel
and load the relevant workbook prior to solving your model.
In addition to inputting the data into this sheet, we also had to define range names for the cost,
capacity, demand, vendor name, and warehouse name regions. Specifically, we defined the following
range names:
Name Range
Capacity K5:K10
Cost C5:J10
Demand C11:J11
Vendors C4:J4
Warehouses B5:B10
Syntax Form 1
The first form of syntax for using @OLE to export data is:
@OLE( ['spreadsheet_file’] [, range_name_list]) = object_list;
The object_list is a comma delimited list of sets, set attributes, and/or scalar variables to be exported.
The 'spreadsheet_file’is the name of the workbook file to export the values to. If the name is omitted,
LINGO defaults to using whatever workbook is currently open in Excel.
The range_name_list is the list of named ranges in the sheet to export solution values to. The ranges
must contain exactly one cell for each exported value for each object in the object_list. Primitive sets
and set attributes export one value per element. Derived sets, on the other hand, export one value for
each dimension of the set. Thus, a two-dimensional set exports two values per set member.
As an example, consider the following model and its solution:
SETS:
S1: X;
S2(S1, S1): Y;
ENDSETS
DATA:
S1,X = M1,1 M2,2 M3,3;
S2,Y = M1,M2,4 M3,M1,5;
ENDDATA
Variable Value
X(M1) 1.000000
X(M2) 2.000000
X(M3) 3.000000
Y(M1, M2) 4.000000
Y(M3, M1) 5.000000
X and Y, both set attributes, export one numeric value per element. More specifically, X exports 1,2,
and 3; while Y exports 4 and 5. S1, a primitive set, exports one text value per element, or the values:
M1, M2, and M3. S2, on the other hand, is a two-dimensional derived set. Thus, it exports two text
values per element. In this case, S2 has two members, so it exports the four values M1, M2, M3, and
M1.
338 CHAPTER 9
There are three options available for how you specify the range names. First, you can explicitly specify
one receiving range for each model object in object_list. Secondly, you can omit the range arguments
entirely. In that case, LINGO supplies the range names with default values that are the same as the
names of the model objects. Finally, you can specify a single range name to export all the solution
values to. In this final case, all the variables in object_list must be defined on the same set, and LINGO
exports the data in a tabular format. Examples of these three methods for using @OLE to export
solutions follow:
Example 1: @OLE('\XLS\DEVELOP.XLS', 'BUILD_IT', 'HOW_BIG') =
BUILD, SQ_FEET;
Here, an individual range for receiving each model object is specified. Thus, the values
of BUILD will be placed in the range BUILD_IT and SQ_FEET in the range HOW_BIG.
When specifying individual ranges, model objects are not required to be defined on the
same set.
Example 2: @OLE('\XLS\DEVELOP.XLS') = BUILD, SQ_FEET;
In this case, we omitted the range name argument. Thus, LINGO defaults to using the
model object names for the range names. So, LINGO exports BUILD and SQ_FEET to
ranges of the same name in the DEVELOP.XLS Excel sheet.
Example 3: @OLE('\XLS\DEVELOP.XLS', 'SOLUTION') = BUILD, SQ_FEET;
Here we have specified a single range, SOLUTION, to receive both model objects.
Assuming that the receiving range has two columns and our model objects are both
one-dimensional, the values of BUILD will be placed in column 1 and SQ_FEET in
column 2. In order for this method to work, BUILD and SQ_FEET must be defined on
the same set.
Note: The major difference to notice between using @OLE for exports, as opposed to imports, is the
side of the statement the @OLE function appears on. When the @OLE function appears on
the left of the equals sign, you are exporting. When it appears on the right, you are importing.
So, always remember:
@OLE( … ) = object_list; ↔ Export, and
object_list = @OLE( … ); ↔ Import.
Another way to remember this convention is that the left-hand side of the expression is
receiving the data, while the right-hand side is the source. For those familiar with computer
programming languages, this follows the convention used in assignment statements.
Syntax Form 2
As with @TEXT, you may also use the @WRITEFOR function in conjunction with @OLE to give you more
control over the values that are exported, which brings us to our second form of syntax for exporting to
Excel:
@OLE( ['spreadsheet_file’], range_name_list) = @WRITEFOR( setname
[ ( set_index_list) [ | conditional_qualifier]] : output_obj_1[,…,output_obj_n]);
One thing to note that differs from the previous syntax is that the range name list is now required when
exporting via the @WRITEFOR function. The range name list can be a single-cell range, a single
multiple-cell range, or a list of multiple-cell ranges.
INTERFACING WITH SPREADSHEETS 339
In the case of a single cell range, the i-th output object will be written to the (i-1)-th column to the right
of the named cell. Note that single-cell ranges act dynamically in that all values will be written to the
workbook even though, of course, they lie outside the single-cell range. When all output is written, the
original single-cell range will be at the upper left corner of the table of output values.
In the case of a single multiple-cell range, LINGO creates a table of all the output values, where output
object i forms the i-th column of the table. This table is then written to the output range. Items are
written from upper-left to lower-right. In general, your output range will have one column for each
output object. If not, the columns will get scrambled on output.
If a list of multiple cell ranges is specified, then you must specify one range name for each output
object. Each output object will we written to its output range in upper-left to lower-right direction.
@WRITEFOR functions like any other set looping function in that, as a minimum, you will need to
specify the set to loop over. Optionally, you may also specify an explicit set index list and a
conditional qualifier. If a conditional qualifier is used, it is tested for each member of the looping set
and output will not occur for any members that don't pass the test. It's this feature of being able to base
output on the results of a condition that distinguish this second style of syntax.
The list of output objects, of course, specifies what it is you want to output. As with the first form of
syntax, the output objects may be labels, set members and variable values. However, you have
additional latitude in that the output objects may now consist of complex expressions of the variable
values (e.g., you could compute the ratio of two variables). This is a useful feature when you need to
report statistics and quantities derived from the variable values. By placing these calculations in the
data section, as opposed to the model section, you avoid adding unnecessary complications to the
constraints of the model.
In general, you can do everything in the second form of syntax that you can do in the first, and more.
However, the first form has an advantage in that it can be very concise.
Some examples follow:
Example 1: @OLE( 'RESULTS.XLS', 'A1') =
@WRITEFOR( DAYS( D) | START( DAYS) #GT# 0:
DAYS( D), START( D));
Here, our target is the single cell A1. Starting at A1 we will write two columns. The first
column will contain the names of the DAYS set for which the attribute START is nonzero.
The second column will contain the START values. Assuming that there are five days that
have nonzero values, then range A1:A5 will contain the names of the days and B1:B5 will
contain the start values.
Example 2: @OLE( 'RESULTS.XLS', 'SKED') =
@WRITEFOR( DAYS( D) | START( DAYS) #GT# 0:
DAYS( D), START( D));
Here, our target is the multiple-cell range SKED. Assuming SKED is a two-column range,
column one will receive the DAYS set members and column 2 will receive the START
values.
340 CHAPTER 9
Example 3: @OLE( 'RESULTS.XLS', 'DAYS', 'START') =
@WRITEFOR( DAYS( D) | START( DAYS) #GT# 0:
DAYS( D), START( D));
In this example, we specify one named range for each output object. In which case, each
output object will be written to its corresponding range.
Note: When exporting to workbooks, receiving ranges that are larger than the number of exported
values can be filled out by either erasing the contents of the extra cells or leaving the extra
cells untouched. The default is to leave the extra cells untouched. If you would like to erase
the contents of the extra cells, you'll need to enable the Fill Out Ranges and Tables option.
INTERFACING WITH SPREADSHEETS 341
We have also used the Insert|Name|Define command in Excel to assign the range name VOLUME to
the receiving range of C16:J21. To define a range name in Excel:
1. select the range by dragging over it with the mouse with the left button
down,
2. release the mouse button,
3. select the Insert|Name|Define command,
4. enter the desired name (VOLUME in this case), and
5. click the OK button.
INTERFACING WITH SPREADSHEETS 343
When we solve this model, LINGO will load Excel (assuming it isn't already running), load the
WIDGETS worksheet, and then pull the data for WAREHOUSES, VENDORS, CAPACITY, COST, and
DEMAND from the worksheet. Once the solver has found the optimal solution, LINGO will send the
values for the VOLUME attribute back to the worksheet storing them in the range of the same name
and the updated range will appear as follows:
The Transfer Method field lists the type of transfer used. @OLE transfers will be listed as “OLE
BASED”.
344 CHAPTER 9
The Workbook field lists the name of the workbook the export was performed on.
The Ranges Specified field lists the total number of ranges specified in the export function followed by
a listing of the range names.
The Ranges Found figure lists the number of ranges actually found in the sheet from the total number
specified.
In general, the spreadsheet range should have one cell for each data value being exported. If a range
has too few or too many cells, it gets counted in the Range Size Mismatches field.
The Values Transferred field lists the total number of data values actually transferred into all the
specified ranges.
Spreadsheet: STAFOLE1.XLS
We have placed the staffing requirements in the range C16:I16 and assigned the name
REQUIREMENTS to this range. We have also assigned the name START to the range C18:I18. LINGO
will be sending the solution to the START range. We have also included two graphs in the sheet to help
visualize the solution. The graph on the left shows how many employees to start on each day of the
week, while the graph on the right compares the number on duty to the number required for each day.
346 CHAPTER 9
Note that our spreadsheet has a second tab at the bottom titled Model. Select this tab and you will find
the following:
Spreadsheet: STAFOLE1.XLS
This page contains the command script we will use to solve the staffing model. For more information
on command scripts, refer to page 323. In line 1, we turn on terminal echoing, so LINGO will echo the
command script to the command window as it is read. Lines 2 through 21 contain the text of the
model, which should be familiar by now. Note, in the data section, we are using two @OLE
functions—the first to import the data from the spreadsheet and the second to export the solution back
to the spreadsheet. The data is read from the range named REQUIRED, and the solution is written to
the START range on the first tab of the sheet. In line 22, we use the GO command to solve the model.
We have also assigned the range name MODEL to the range that contains this script (Model!A1:A23).
INTERFACING WITH SPREADSHEETS 347
Given that we have our LINGO command script contained in our spreadsheet, the next question is how
we pass it to LINGO to run it. This is where OLE Automation comes in. If you recall, the first tab of
our sheet (the tab labeled Data) had a Solve button. We added this button to the sheet and attached the
following Excel Visual Basic macro to it:
Sub LINGOSolve()
Dim iErr As Integer
iErr = LINGO.RunScriptRange("MODEL")
If (iErr > 0) Then
MsgBox ("Unable to solve model")
End If
End Sub
We use OLE Automation to call the LINGO exported method RunScriptRange, passing it the range
name MODEL. This, of course, is the name of the range that contains the command script. The
RunScriptRange routine calls Excel to obtain the contents of the range and begins processing the
commands contained therein. Processing continues until either a QUIT command is encountered or
there are no further commands remaining in the range.
RunScriptRange will return a value of 0 if it was successfully able to queue the script for processing. If
RunScriptRange was not successful, it will return one of the following error codes:
Error
Code Description
1 Invalid argument
2 <Reserved>
3 Unable to open log file
4 Null script
5 Invalid array format
6 Invalid array dimension
7 Invalid array bounds
8 Unable to lock data
9 Unable to allocate memory
10 Unable to configure script reader
11 LINGO is busy
12 OLE exception
13 Unable to initialize Excel
14 Unable to read Excel range
15 Unable to find Excel range
We have also added the following Auto_Open macro to the sheet:
Dim LINGO As Object
Sub Auto_Open()
Set LINGO = CreateObject("LINGO.Document.4")
End Sub
An Auto_Open macro is automatically executed each time a sheet is opened. We declare LINGO as an
object and attach the LINGO object to the LINGO application with the CreateObject function.
348 CHAPTER 9
Now, go back to the first tab on the workbook and press the Solve button. After a brief pause, you
should see the optimal solution installed, so the sheet resembles:
Spreadsheet: STAFOLE1.XLS
The optimal number of employees to start on each day of the week is now contained in the START
range (C18:I18), and the graphs have been updated to reflect this solution.
Click the OK button and a blank LINGO model window will be embedded within the spreadsheet. You
can enter text directly into this window just as you would in LINGO, or you can paste it in from
another application. When you save the Excel sheet, the embedded LINGO model will automatically
be saved with it. Similarly, whenever you read the sheet back into Excel, the embedded LINGO model
will be restored, as well.
350 CHAPTER 9
To illustrate this feature, we will continue with the staffing model introduced in Chapter 2, Using Sets.
The spreadsheet will contain the data for the model, and it will also contain an embedded LINGO
model to perform the optimization and install the solution back into the spreadsheet. This example may
be found in the spreadsheet file SAMPLES\STAFOLE2.XLS. If you load this sheet into Excel, you
should see the following:
Spreadsheet: STAFOLE2.XLS
As in the previous example, the staffing requirements are in the range C16:I16. This range has been
titled REQUIRED. The range C18:I18 has been assigned the name START and will receive the solution
after the model is solved. In the upper right-hand corner of the sheet, we have defined a graph to help
visualize the solution.
INTERFACING WITH SPREADSHEETS 351
In the upper left corner, there is a region labeled <Embedded LINGO Model>. This region contains a
LINGO model that will solve the staffing model and place the solution values into the spreadsheet. If
you double-click on this region, you will be able to see the model:
Spreadsheet: STAFOLE2.XLS
Note, when this LINGO model is active, the LINGO menus and toolbar replace the Excel menus and
toolbar. Thus, when working with an embedded LINGO model in Excel, you have all the functionality
of LINGO available to you. When you deselect the LINGO model, the Excel menus and toolbar will
automatically become available once again. This begins to illustrate the power of embedded OLE—it
allows the user to seamlessly combine the features of two or more applications together as if they were
a single, integrated application.
352 CHAPTER 9
You can drag the lower right-hand corner of the LINGO model region to expose the contents of the
entire model:
SETS:
DAYS / MON TUE WED THU FRI SAT SUN/:
REQUIRED, START;
ENDSETS
DATA:
REQUIRED =
@OLE('C:\LINGO\SAMPLES\STAFOLE2.XLS');
@OLE('C:\LINGO\SAMPLES\STAFOLE2.XLS') = START;
ENDDATA
MIN = @SUM(DAYS: START);
@FOR(DAYS(J):
@SUM(DAYS(I) | I #LE# 5:
START(@WRAP(J - I + 1, 7)))
>= REQUIRED(J));
@FOR(DAYS: @GIN(START));
Once again, we are making use of our familiar staff-scheduling model. The main feature to note is that
we are using two instances of the @OLE function. The first instance gets the staffing requirements
from the spreadsheet. The second sends the solution back to the START range.
INTERFACING WITH SPREADSHEETS 353
To solve the model, double-click on the region containing the LINGO model. The LINGO command
menus will become visible along the top of the screen. Select the LINGO|Solve command. After
LINGO finishes optimizing the model, it will return the solution to the sheet and we will have:
Spreadsheet: STAFOLE2.XLS
354 CHAPTER 9
Model: STAFFOLE
INTERFACING WITH SPREADSHEETS 355
This model reads data from and writes the solution to the Excel file STAFFOLE.XLS. To make our
lives easier, it would be nice to embed this spreadsheet in the LINGO model to avoid having to load it
into Excel each time we need to work with our model. To do this, select the Edit|Insert New Object
command in LINGO. You will be presented with the dialog box:
Click on the Create from File button, enter the spreadsheet file name in the File field, click on the Link
checkbox, and then press the OK button.
356 CHAPTER 9
Your LINGO model should now appear as:
The data spreadsheet is now embedded at the top of the LINGO document. You can easily edit the
spreadsheet by double-clicking on it. When you save the LINGO model, the link to the Excel file will
be saved as well.
INTERFACING WITH SPREADSHEETS 357
At this point, go ahead and optimize the model by selecting the LINGO|Solve command. When LINGO
has finished solving the model, you should see the following:
The optimal solution now appears in the embedded spreadsheet and the graphs of the solution have
been updated.
358 CHAPTER 9
Summary
We have demonstrated a number of intriguing methods for combining the modeling features of
LINGO with the data management features of spreadsheets. While spreadsheet solver products such as
What’sBest! offer the convenience of being able to formulate the entire model in a spreadsheet file, the
examples above begin to demonstrate the unique advantages of combining the stand alone modeling
features of LINGO with the data management features of spreadsheets. The concise, structured nature
of LINGO’s modeling language makes reading and understanding even large models relatively easy.
Whereas, it can often be a challenge to discern the underlying mathematical relationships of large,
spreadsheet solver models with formulas spread throughout numerous cells on multiple tabbed sheets.
LINGO models are also easier to scale than spreadsheet solver models. As the dimensions of your
problem change, spreadsheet solver models can require the insertion or deletion of rows or columns
and the editing of cell formulas. Whereas, if your data is separate from the model, your LINGO
formulation will generally require little or no modification. Combining the power of model expression
with LINGO and the ease of data handling in spreadsheets gives the mathematical modeler the best of
both worlds.
10 Interfacing with
Databases
Spreadsheets are good at managing small to moderate amounts of data. Once your models start dealing
with large amounts of data, database management systems (DBMSs) are, unquestionably, the tool of
choice. Also, in many business modeling situations, you will find most, if not all, of the data is
contained in one or more databases. For these reasons, LINGO supports links to any DBMS that has an
Open DataBase Connectivity (ODBC) driver (effectively all popular DBMSs). ODBC defines a
standardized interface to DBMSs. Given this standardized interface, LINGO can access any database
that supports ODBC.
LINGO has one interface function for accessing databases. This function’s name is @ODBC. The
@ODBC function is used to import data from and export data to any ODBC data source. @ODBC is
currently available only in Windows versions of LINGO.
359
360 CHAPTER 10
You should now see the ODBC Administrator dialog box shown below:
362 CHAPTER 10
To install the TRANDB.MDB database as a data source, do the following:
1. Click the Add button in the ODBC Administrator dialog box to reveal the dialog box
below:
2. We are installing an Access data source, so select the Microsoft Access Driver option
and press the Finish button.
3. In the next dialog box:
INTERFACING WITH DATABASES 363
assign the data source the name Transportation in the Data Source Name field. In the
Description field, enter “Datasource for a LINGO transportation model”. Press the Select
button and enter the name of the database “LINGO\SAMPLES\TRANDB.MDB” (this
assumes LINGO has been installed in the LINGO directory—your installation may differ).
The dialog box should now resemble the one below:
364 CHAPTER 10
4. Press the OK button and you should see the Transportation data source has been
added to the list of ODBC data sources:
DATA:
TASKS = @ODBC('PERTODBC', 'TASKS', 'TASKS');
PRED = @ODBC('PERTODBC', 'PRECEDENCE',
'BEFORE', 'AFTER');
TIME = @ODBC('PERTODBC');
ENDDATA
@FOR(TASKS(J)| J #GT# 1:
ES(J) = @MAX(PRED(I, J): ES(I) + TIME(I))
);
ES(1) = 0;
LTASK = @SIZE(TASKS);
LS(LTASK) = ES(LTASK);
Model: PERTODBC
370 CHAPTER 10
With the statement:
TASKS = @ODBC('PERTODBC', 'TASKS', 'TASKS');
we now fetch the members of the TASKS set from our ODBC data source, as opposed to explicitly
listing them in the model. Specifically, we get the members of the TASKS set from the data column, or
field, TASKS contained in the table named TASKS from the ODBC data source PERTODBC. Here is
the data table as it appears in Access:
Note: When importing, @ODBC appears on the right of the equality sign. When exporting, the
@ODBC function appears on the left of the equals sign.
The object_list is a list, optionally separated by commas, containing model objects (i.e., attributes, sets,
or variables) that are to be exported to the ODBC data source. Object_list may contain up to one set
and/or multiple set attributes. All set attributes in object_list must be defined on the same set. If
object_list contains a set, then all attributes in object_list must be defined on this set. The data_source
argument is the name of the ODBC data source containing the data table that will receive the exported
values. The table_name argument is the name of the data table within the data source that will receive
the data. Finally, the column_name arguments are the names of the receiving columns, or fields, in the
data table table_name. Set attributes and primitive sets require one receiving column name each.
Derived sets require one receiving column name for each dimension of the set. Thus, a two-
dimensional derived set would require two receiving columns in a data table.
If the data_source argument is omitted, the model's title is used in its place (see the discussion of the
TITLE statement in Chapter 1, Getting Started with LINGO). If table_name is omitted, the name of any
set in the object_list is used in its place. If there is no set in object_list, then the name of the set where
the attributes in object_list are defined is used.
If the column_name arguments are omitted, LINGO will choose default names based on whether the
corresponding object in object_list is either a set attribute, a primitive set, or a derived set. When the
object to be initialized is a set attribute or a primitive set, LINGO will use the name of the object as the
default column name. When the object is a derived set, LINGO will generate one default column name
for each dimension of the derived set, with each name being the same as the parent set that the given
372 CHAPTER 10
dimension is derived from. As an example, a two-dimensional set named LINKS derived from the
SOURCE and DESTINATION primitive sets would default to being exported to the two columns titled
SOURCE and DESTINATION.
Keep in mind that set members are exported as text, while set attributes are exported as double
precision floating point values.
Some examples of using @ODBC to export data values to an ODBC data source are:
The first form of syntax will generally be sufficient for most database export operations. However,
there may be times when you need to export only portions of the attributes, or you need to export
quantities computed from the attribute values. Our second form of syntax uses the @WRITEFOR
reporting function to handle these more general cases:
@ODBC( 'data_source', 'table_name', 'column_name_1'[,…,
'column_name_n']) = @WRITEFOR( setname
[ ( set_index_list) [ | conditional_qualifier]] : output_obj_1[,…, output_obj_n]);
@WRITEFOR functions like any other set looping function in that, as a minimum, you will need to
specify the set to loop over. Optionally, you may also specify an explicit set index list and a
conditional qualifier. If a conditional qualifier is used, it is tested for each member of the looping set
and output will not occur for any members that don't pass the test. It's this feature of being able to base
output on the results of a condition that distinguish this second style of syntax.
The list of output objects, of course, specifies what it is you want to output. As with the first form of
syntax, the output objects may be labels, set members and variable values. However, you have
additional latitude in that the output objects may now consist of complex expressions of the variable
values (e.g., you could compute the ratio of two variables). This is a useful feature when you need to
report statistics and quantities derived from the variable values. By placing these calculations in the
data section, as opposed to the model section, you avoid adding unnecessary complications to the
constraints of the model.
In general, you can do everything in the second form of syntax that you can do in the first, and more.
However, the first form has an advantage in that it can be very concise.
INTERFACING WITH DATABASES 373
Some examples of using @WRITEFOR for ODBC exports follow:
Example 1: @ODBC( 'TRANSPORTATION',
'SOLUTION', 'FROM', 'TO', 'VOLUME') =
@WRITEFOR( LINKS( I, J) | VOLUME( I, J) #GT# 0:
WAREHOUSE( I), CUSTOMER( J), VOLUME( I, J));
In this example, we exploit the ability to specify a conditional expression to weed zero
shipments out of the export. The nonzero values of the VOLUME attribute are sent to the
SOLUTION table in the TRANSPORTATION data source. The shipping warehouse set
name is placed in column FROM, the receiving customer set name goes to column TO, and
the shipping volume for the arc in placed in the VOLUME column.
Note: When exporting to data tables, receiving columns that are longer than the number of exported
values can be filled out by either erasing the contents of the extra cells or leaving the extra
cells untouched. The default is to leave the extra cells untouched. If you would like to erase
the contents of the extra cells, you'll need to enable the Fill Out Ranges and Tables option. If
this option is enabled, extra text fields will be blanked out, while extra numeric fields will be
zeroed out.
374 CHAPTER 10
DATA:
TASKS = @ODBC('PERTODBC', 'TASKS', 'TASKS');
PRED = @ODBC('PERTODBC', 'PRECEDENCE', 'BEFORE', 'AFTER');
TIME = @ODBC('PERTODBC');
@ODBC('PERTODBC', 'SOLUTION', 'TASKS',
'EARLIEST START', 'LATEST START') =
TASKS, ES, LS;
ENDDATA
@FOR(TASKS(J)| J #GT# 1:
ES(J) = @MAX(PRED(I, J): ES(I) + TIME(I))
);
ES(1) = 0;
LTASK = @SIZE(TASKS);
LS(LTASK) = ES(LTASK);
Model: PERTODBC
With the data statement:
@ODBC('PERTODBC', 'SOLUTION', 'TASKS',
'EARLIEST START', 'LATEST START') =
TASKS, ES, LS;
we are sending the set TASKS to the text column TASKS, and the ES and LS attributes to the numeric
columns EARLIEST START and LATEST START. The data table is called SOLUTION, while the
ODBC data source name is PERTODBC.
376 CHAPTER 10
Once the model has been solved, the updated data table will resemble:
The Transfer Method will always list “ODBC BASED” when doing ODBC exports. Next, the data
source and table names are listed along with the number of columns specified and the column names.
The LINGO Column Length field lists the number of elements in each attribute. The Database Column
Length lists the length of the receiving columns in the database. In general, the LINGO Column
Length will agree with the Database Column Length. If not, LINGO must either truncate its output or
it will have insufficient data to fill out the columns in the database.
Export summary reports are not displayed when LINGO is in terse output mode. To place LINGO in
terse output mode, click on the Terse Output checkbox on the Interface tab of the LINGO|Options
dialog box.
INTERFACING WITH DATABASES 377
This version of the PERT model and its supporting database are contained in the SAMPLES directory.
Feel free to run the model to experiment with it if you like—you will find it under the name
PERTODBC. The supporting Access database file, PERTODBC.MDB, is also in the SAMPLES
subdirectory and you will need to register it with the ODBC Administrator as described above in
ODBC Data Sources.
Note that we exported start and finish times for all the tasks in the project. If we were dealing with a
large project there could be thousands of tasks to consider. With such an abundance of tasks, we might
be interested in reporting only those tasks that lie on the critical path. We'll modify our PERTODBC
example one last time to accomplish this using the @WRITEFOR reporting function.
For those unfamiliar with the concept of a critical path, it is the subset of tasks such that if any are
delayed the entire project will be delayed. Generally, and somewhat counterintuitive to what one
would expect, the set of tasks on the critical path will tend to be quite small compared to the total
number of tasks in a large project. A task is considered to be on the critical path when its earliest start
time is equal to its latest start time (i.e., there is no slack with respect to when the task must be started).
Below, we have modified PERTODBC to export only those tasks on the critical path.
MODEL:
SETS:
TASKS: TIME, ES, LS, SLACK;
PRED( TASKS, TASKS);
ENDSETS
DATA:
TASKS = @ODBC( 'PERTODBC', 'TASKS', 'TASKS');
PRED = @ODBC( 'PERTODBC', 'PRECEDENCE', 'BEFORE', 'AFTER');
TIME = @ODBC( 'PERTODBC');
@ODBC( 'PERTODBC', 'SOLUTION', 'TASKS',
'EARLIEST START', 'LATEST START') =
@WRITEFOR( TASKS( I) | ES( I) #EQ# LS( I):
TASKS( I), ES( I), LS( I));
ENDDATA
ES( 1) = 0;
LTASK = @SIZE( TASKS);
LS( LTASK) = ES( LTASK);
END
378 CHAPTER 10
We specify a conditional expression to test for the earliest start time equaling the latest start time,
thereby restricting the list of exported tasks to those that lie on the critical path. Note that if you limit
the number of output records there won't be enough records to completely fill our output table. We
can have LINGO fill out any extra fields with blanks and zeroes by enabling the Fill Out Ranges and
Tables option. Doing this, the solution table will resemble the following after solving the model:
Note that that LINGO nulled out all the extra records.
11 Interfacing with Other
Applications
Although LINGO has a convenient, interactive interface and a large library of callable functions that
make it easy to set up and solve models, there may be times when you would like to bundle LINGO's
functionality into your own applications, or call functions from within your LINGO models that were
written in an external programming language. LINGO makes use of the Dynamic Link Library (DLL)
standard under Windows to provide you with a “hook” to access LINGO's functionality from within
your own custom applications. This gives you the ability to build application specific front-ends to
LINGO that allow naïve users to input data and view solutions in a simple, familiar way without
having to worry about the details of LINGO modeling. Your application handles the details of driving
LINGO in the background, invisible to the user. LINGO also allows you to provide custom functions
in the DLL format that you can call from within any model with the @USER function.
In the following section, The LINGO Dynamic Link Library, we document how to call the LINGO
DLL in order to add LINGO's functionality to your own applications. Following this, in the section
User Defined Functions, we will show you how to build a function in an external programming
language and call it from a LINGO model with the @USER function.
379
380 CHAPTER 11
When LINGO is installed a number of examples on calling the DLL are installed, too. These
examples may be found in the Programming Samples folder below the main LINGO folder. You will
find examples for each of the following development environments:
♦ Visual C/C++
♦ Visual Basic
♦ Excel
♦ FORTRAN
♦ C# .NET
♦ VB .NET
♦ Delphi
♦ Java
In this chapter, we will walk through detailed examples on calling the LINGO DLL using both Visual
C/C++ and Visual Basic. Users of other development environments will also be interested in these
programming examples. Many of the ideas presented carry over to other development environments.
we see the staffing requirements attribute, NEEDS, is read from the first memory location. The START,
ONDUTY, and OBJECTIVE values are written to the second, third, and fourth memory locations,
respectively. Finally, the value for the @STATUS function is written to the fifth memory location.
384 CHAPTER 11
The @STATUS Function
The @STATUS function returns the status of the solution process using the following codes:
@STATUS Interpretation
Code
0 Global Optimum - The optimal solution has been found.
1 Infeasible - No solution exists that satisfies all constraints.
2 Unbounded - The objective can be improved without
bound.
3 Undetermined - The solution process failed.
4 Feasible - A feasible solution was found that may, or may
not, be the optimal solution.
5 Infeasible or Unbounded - The preprocessor determined
the model is either infeasible or unbounded. Turn off
presolving and re-solve to determine which.
6 Local Optimum - Although a better solution may exist, a
locally optimal solution has been found.
7 Locally Infeasible - Although feasible solutions may
exist, LINGO was not able to find one.
8 Cutoff - The objective cutoff level was achieved.
9 Numeric Error - The solver stopped due to an undefined
arithmetic operation in one of the constraints.
In general, if @STATUS does not return a code of 0 or 7, the solution is of little use and should not be
trusted. In fact, if @STATUS does not return 0 or 7, in many cases LINGO will not even export data to
the @POINTER memory locations.
The @POINTER function reads and writes all data using double precision floating point format.
BASIC and C/C++ developers know this as the double data type, while FORTRAN developers refer to
it as either REAL*8 or DOUBLE PRECISION.
When setting aside memory locations for @POINTER, you must be sure to allocate adequate space.
LINGO assumes adequate space and/or supplied values are available at the passed memory locations.
If you do not allocate sufficient space or values, you may experience memory protection faults or, even
worse, erroneous results without warning.
Supporting DLLs
The main LINGO DLL requires many additional component DLLs. All the required DLLs are
installed as part of the normal LINGO installation in the main LINGO folder. If you need a specific list
of all the supporting DLLs required by the LINGO DLL, then you can open Lingd10.Dll in the
Dependency Walker utility available at no charge from the following website:
http://www.dependencywalker.com. The Dependency Walker program is a useful tool that lists all the
required DLLs for an application, their current location on your system, and flags any DLLs that aren’t
currently available.
Click the OK button, and AppWizard will generate the skeleton code base.
392 CHAPTER 11
Use the resource editor to modify the application’s dialog box, so it resembles the following:
Next, use the ClassWizard in Visual C++ to associate member variables with each of the fields in the
dialog box. From the View menu, select the ClassWizard command and then select the Member
Variables tab.
At this point, the LINGO DLL import library must be added to the project in order to make the LINGO
DLL available to the code. Do this by running the Project|Add to Project|Files command and select the
file \LINGO10\Programming Samples\LINGD10.LIB for addition to the project.
After doing that, add the definitions of the LINGO DLL routines to the project. Simply include the
Lingd90.h header file at the top of the dialog class header file as follows (code changes listed in bold):
// staffDlg.h : header file
//
#include "lingd10.h"
#if
!defined(AFX_STAFFDLG_H__74D746B7_CA4D_11D6_AC89_00010240D2AE__INCLUD
ED_)
#define
AFX_STAFFDLG_H__74D746B7_CA4D_11D6_AC89_00010240D2AE__INCLUDED_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
/////////////////////////////////////////////////////////////////////
////////
// CStaffDlg dialog
.
.
.
INTERFACING WITH OTHER APPLICATIONS 393
All the groundwork has now been laid down and we’re ready to begin writing the actual code to call
LINGO to solve the staffing model. Go to the Resource View of the project, open the dialog box
resource, and then double click on the Solve button. You will be prompted to create a handler function
for the button, which should be named OnSolve. Now, edit the stub version of OnSolve, so it contains
the following code:
void CStaffDlg::OnSolve()
{
int nError, nPointersNow;
CString csScript, cs;
double dNeeds[7], dStart[7], dOnDuty[7], dStatus, dTotal;
// Get user's staffing requirements from our dialog box
UpdateData();
// Load staffing requirements into the LINGO transfer array.
// LINGO uses double precision for all values.
dNeeds[ 0] = (double) m_nNeedsMon;
dNeeds[ 1] = (double) m_nNeedsTue;
dNeeds[ 2] = (double) m_nNeedsWed;
dNeeds[ 3] = (double) m_nNeedsThu;
dNeeds[ 4] = (double) m_nNeedsFri;
dNeeds[ 5] = (double) m_nNeedsSat;
dNeeds[ 6] = (double) m_nNeedsSun;
// create the LINGO environment object
pLSenvLINGO pLINGO;
pLINGO = LScreateEnvLng();
if ( !pLINGO)
{
AfxMessageBox("Unable to create LINGO Environment");
return;
}
// Open LINGO's log file
nError = LSopenLogFileLng( pLINGO, "\\LINGO10\\LINGO.log");
if ( nError) goto ErrorExit;
// Pass memory transfer pointers to LINGO
// @POINTER(1)
nError = LSsetPointerLng( pLINGO, dNeeds, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(2)
nError = LSsetPointerLng( pLINGO, dStart, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(3)
nError = LSsetPointerLng( pLINGO, dOnDuty, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(4)
nError = LSsetPointerLng( pLINGO, &dTotal, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(5)
nError = LSsetPointerLng( pLINGO, &dStatus, &nPointersNow);
if ( nError) goto ErrorExit;
394 CHAPTER 11
// Here is the script we want LINGO to run
csScript = "SET ECHOIN 1\n";
csScript = csScript +
"TAKE \\LINGO10\\SAMPLES\\STAFFPTR.LNG\n";
csScript = csScript +
"GO\n";
csScript = csScript +
"QUIT\n";
// Run the script
dStatus = -1.e0;
nError = LSexecuteScriptLng( pLINGO, (LPCTSTR) csScript);
// Close the log file
LScloseLogFileLng( pLINGO);
// Any problems?
if ( nError || dStatus)
{
// Had a problem
AfxMessageBox("Unable to solve!");
} else {
// Everything went ok ... load results into the dialog box
m_csStartMon.Format( "%d", (int) dStart[0]);
m_csStartTue.Format( "%d", (int) dStart[1]);
m_csStartWed.Format( "%d", (int) dStart[2]);
m_csStartThu.Format( "%d", (int) dStart[3]);
m_csStartFri.Format( "%d", (int) dStart[4]);
m_csStartSat.Format( "%d", (int) dStart[5]);
m_csStartSun.Format( "%d", (int) dStart[6]);
m_csOnDutyMon.Format( "%d", (int) dOnDuty[0]);
m_csOnDutyTue.Format( "%d", (int) dOnDuty[1]);
m_csOnDutyWed.Format( "%d", (int) dOnDuty[2]);
m_csOnDutyThu.Format( "%d", (int) dOnDuty[3]);
m_csOnDutyFri.Format( "%d", (int) dOnDuty[4]);
m_csOnDutySat.Format( "%d", (int) dOnDuty[5]);
m_csOnDutySun.Format( "%d", (int) dOnDuty[6]);
m_csCost.Format( "%g", dTotal);
UpdateData( FALSE);
}
goto Exit;
ErrorExit:
cs.Format( "LINGO Errorcode: %d", nError);
AfxMessageBox( cs);
return;
Exit:
LSdeleteEnvLng( pLINGO);
}
The first section of OnSolve is straightforward and deals with extracting the user’s staffing
requirements from the dialog box. Note that the data is stored in a double precision array rather than as
integers. This is because these values will be passed to LINGO, which only passes values in double
precision format.
INTERFACING WITH OTHER APPLICATIONS 395
Our first call to LINGO creates the LINGO environment object with the following code:
// create the LINGO environment object
pLSenvLINGO pLINGO;
pLINGO = LScreateEnvLng();
if ( !pLINGO)
{
AfxMessageBox("Unable to create LINGO Environment");
return;
}
The pLSenvLINGO data type is defined in the LINGO header file, lingd10.h.
Then, a log file for LINGO is established with the following call:
// Open LINGO's log file
nError = LSopenLogFileLng( pLINGO, "\\LINGO.log");
if ( nError) goto ErrorExit;
As mentioned above, opening a log file for LINGO is good practice, at least when you’re debugging
the application. If something should go wrong, the log file will generally contain a helpful clue.
Our next step is to pass LINGO the physical addresses it will use to resolve the @POINTER() function
references used in the data section of the model (refer to the Staff Scheduling Example Using the
LINGO DLL section above for more details). This is done as follows:
// Pass memory transfer pointers to LINGO
// @POINTER(1)
nError = LSsetPointerLng( pLINGO, dNeeds, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(2)
nError = LSsetPointerLng( pLINGO, dStart, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(3)
nError = LSsetPointerLng( pLINGO, dOnDuty, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(4)
nError = LSsetPointerLng( pLINGO, &dTotal, &nPointersNow);
if ( nError) goto ErrorExit;
// @POINTER(5)
nError = LSsetPointerLng( pLINGO, &dStatus, &nPointersNow);
if ( nError) goto ErrorExit;
In summary, when LINGO is called to solve the model, the staffing needs are passed to LINGO in the
dNeeds array via the @POINTER( 1) reference. Solution information is passed from LINGO back to
the application in the dStart, dOnDuty, dTotal, and dStatus structures via @POINTER() references 2
through 5, respectively. If any of this is unfamiliar, review The @POINTER() Function section under
the Staff Scheduling Example Using the LINGO DLL section above.
396 CHAPTER 11
Next, the following code is used to build the command script:
// Here is the script we want LINGO to run
csScript = "SET ECHOIN 1\n";
csScript = csScript +
"TAKE \\LINGO10\\SAMPLES\\STAFFPTR.LNG\n";
csScript = csScript +
"GO\n";
csScript = csScript +
"QUIT\n";
The script consists of four commands, which are each terminated with a new line character (ASCII
10). The end of the script is terminated with a NULL character (ASCII 0). These commands and their
functions are:
Command Function
SET ECHOIN 1 Causes LINGO to echo input to the log file. This is a useful feature
while building and debugging an application.
TAKE Loads the model from a disk file. The TAKE command may be used to
load model files, as in this example. It may also be used to run nested
command scripts contained in files.
GO Calls the solver to optimize the model.
QUIT Closes down LINGO’s script processor and returns control to the
calling application.
For more information on scripting commands, refer to Chapter 6, Command-line Commands.
At this point, the script is ready to be passed off to LINGO for processing with the following call:
// Run the script
dStatus = -1.e0;
nError = LSexecuteScriptLng( pLINGO, (LPCTSTR) csScript);
Note that dStatus is initialized to –1. LINGO returns the model status through memory transfer
location number 5 (i.e., @POINTER( 5)) to the dStatus variable. LINGO will only return a status value
if it was able to solve the model. If an unexpected error were to occur, LINGO might not ever reach
the solution phase. In that case, dStatus would never be set. Initializing dStatus to a negative value
tests for this situation. Given that LINGO returns only non-negative status codes, a negative status
code would indicate a problem. This method of error trapping is effective, but not particularly elegant.
Another method that involves specifying an error callback routine is demonstrated below.
Now, LINGO’s log file may be closed down by calling LScloseLogFileLng():
// Close the log file
LScloseLogFileLng( pLINGO);
Next, the following code tests to see if LINGO was able to find an optimal solution:
// Any problems?
if ( nError || dStatus)
// Had a problem
AfxMessageBox("Unable to solve!");
} else {
INTERFACING WITH OTHER APPLICATIONS 397
Note that the code returned in nError pertains only to the mechanical execution of the script processor.
It has nothing to do with the status of the model’s solution, which is returned in dStatus via the use of
the @POINTER() and @STATUS() functions (see the Staff Scheduling Example Using the LINGO
DLL section above). A model may actually be infeasible or unbounded, and the error code returned by
LSexecuteScript() will give no indication. Thus, it is important to add a mechanism to return a
solution’s status to the calling application, as done here with the @STATUS() -> @POINTER(5) ->
dStatus link. The end result in the sample code is that “Unable to solve” is printed if either error
condition occurs. A more user-friendly application would offer more specific information regarding
the error condition.
As a final step, in order to avoid memory leaks in your application, remember to free up LINGO’s
environment when through:
Exit:
LSdeleteEnvLng( pLINGO);
If everything has been entered correctly, you should now be able to build and run the project.
Now, add handler code for the two buttons in the form. The Exit button is easy. All the Exit button
does when the user clicks it is exit the application. So, double click on the Exit button and enter the
following code:
Private Sub Exit_Click()
End
End Sub
A little more work will be required to setup the Solve button. When the Solve button is pressed, the
application will retrieve the staffing requirements from the form, pass them along with the model to
the LINGO script processor (LGVBSCRIPT) to solve the model, retrieve the solution, and post the
solution in the form.
INTERFACING WITH OTHER APPLICATIONS 399
First, we must declare the external LINGO functions. Do this by adding the
\LINGO10\Programming Samples\LINGD10.BAS module to the project using the Project|Add
Module command in VB. This module contains the definitions of all the exported function in the
LINGO DLL, and makes them available to our project.
Now, add the handler code for the Solve button. Go to the form, double click on the Solve button, and
enter the following code:
Private Sub Solve_Click()
' Calls the LINGO DLL to solve the staffing
' model in STAFFPTR.LNG. Staffing
' requirements are taken from the dialog
' box.
' Get the staffing needs from the dialog box
Dim dNeeds(7) As Double
For i = 1 To 7
dNeeds(i) = Needs(i - 1).Text
Next i
' Create the LINGO environment object
Dim pLINGO As Long
pLINGO = LScreateEnvLng()
If pLINGO = 0 Then
MsgBox ("Unable to create LINGO Environment.")
GoTo FinalExit
End If
' Open LINGO's log file
Dim nError As Long
nError = LSopenLogFileLng(pLINGO, "\LINGO.log")
If nError <> 0 Then GoTo ErrorExit
' Pass memory transfer pointers to LINGO
Dim dStart(7) As Double, dOnDuty(7) As Double
Dim dTotal As Double, dStatus As Double
' @POINTER(1)
nError = LSsetPointerLng(pLINGO, dNeeds(1), nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(2)
nError = LSsetPointerLng(pLINGO, dStart(1), nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(3)
nError = LSsetPointerLng(pLINGO, dOnDuty(1), nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(4)
nError = LSsetPointerLng(pLINGO, dTotal, nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(5)
nError = LSsetPointerLng(pLINGO, dStatus, nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' Build LINGO's command script (commands
' are terminated with an ASCII 10
Dim cScript As String
400 CHAPTER 11
' Causes LINGO to echo input
cScript = "SET ECHOIN 1" & Chr(10)
' Read in the model file
cScript = cScript & _
"TAKE \LINGO10\SAMPLES\STAFFPTR.LNG" & Chr(10)
' Solve the model
cScript = cScript & "GO" & Chr(10)
' Quit LINGO DLL
cScript = cScript & "QUIT" & Chr(10)
' Mark end of script with a null byte
cScript = cScript & Chr(0)
' Run the script
dStatus = -1#
nError = LSexecuteScriptLng(pLINGO, cScript)
' Close the log file
LScloseLogFileLng (pLINGO)
' Problems?
If nError > 0 Or dStatus <> 0 Then
MsgBox ("Unable to solve!")
GoTo ErrorExit
End If
' Place Start values in dialog box
For i = 1 To 7
Start(i - 1).Caption = dStart(i)
Next i
' Place On Duty values in dialog box
For i = 1 To 7
OnDuty(i - 1).Caption = dOnDuty(i)
Next i
' Put Total staffing in dialog box
Total.Caption = dTotal
LSdeleteEnvLng (pLINGO)
GoTo FinalExit:
ErrorExit:
MsgBox ("LINGO Error Code: " & nError&)
LSdeleteEnvLng (pLINGO)
FinalExit:
End Sub
The first section of the Solve_Click procedure is straightforward and deals with extracting the user’s
staffing requirements from the dialog box. Note that the data is stored in a double precision array,
rather than as integers. This is because these values will be passed to LINGO, which only passes
values in double precision format.
INTERFACING WITH OTHER APPLICATIONS 401
The first call to LINGO creates the LINGO environment object with the following code:
' Create the LINGO environment object
Dim pLINGO As Long
pLINGO = LScreateEnvLng()
If pLINGO = 0 Then
MsgBox ("Unable to create LINGO Environment.")
GoTo FinalExit
End If
As mentioned above, opening a log file for LINGO is good practice, at least when you’re debugging
the application. If something should go wrong, the log file will generally contain a helpful clue.
The next step is to pass LINGO the physical addresses it will use to resolve the @POINTER() function
references used in the data section of the model (refer to the Staff Scheduling Example Using the
LINGO DLL section above for more details). This is done as follows:
' Pass memory transfer pointers to LINGO
Dim dStart(7) As Double, dOnDuty(7) As Double
Dim dTotal As Double, dStatus As Double
' @POINTER(1)
nError = LSsetPointerLng(pLINGO, dNeeds(1), nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(2)
nError = LSsetPointerLng(pLINGO, dStart(1), nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(3)
nError = LSsetPointerLng(pLINGO, dOnDuty(1), nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(4)
nError = LSsetPointerLng(pLINGO, dTotal, nPointersNow)
If nError <> 0 Then GoTo ErrorExit
' @POINTER(5)
nError = LSsetPointerLng(pLINGO, dStatus, nPointersNow)
If nError <> 0 Then GoTo ErrorExit
In summary, when LINGO is called to solve the model, the staffing needs are passed to LINGO in the
dNeeds array via the @POINTER( 1) reference. Solution information is passed from LINGO back to
the application in the dStart, dOnDuty, dTotal, and dStatus structures via @POINTER() references 2
through 5, respectively. If any of this is unfamiliar, review The @POINTER() Function section under
the Staff Scheduling Example Using the LINGO DLL section above.
402 CHAPTER 11
Next, the following code is used to build the command script:
' Build LINGO's command script (commands
' are terminated with an ASCII 10
Dim cScript As String
' Causes LINGO to echo input
cScript = "SET ECHOIN 1" & Chr(10)
' Read in the model file
cScript = cScript & _
"TAKE \LINGO10\SAMPLES\STAFFPTR.LNG" & Chr(10)
' Solve the model
cScript = cScript & "GO" & Chr(10)
' Quit LINGO DLL
cScript = cScript & "QUIT" & Chr(10)
' Mark end of script with a null byte
cScript = cScript & Chr(0)
The script consists of four commands, which are each terminated with a new line character (ASCII
10). The end of the script is terminated with a NULL character (ASCII 0). These commands and their
functions are:
Command Function
SET ECHOIN 1 Causes LINGO to echo input to the log file. This is a useful feature
while building and debugging an application.
TAKE Loads the model from a disk file. The TAKE command may be used to
load model files, as in this example. It may also be used to run nested
command scripts contained in files.
GO Calls the solver to optimize the model.
QUIT Closes down LINGO’s script processor and returns control to the
calling application.
At this point, the script is ready to be passed off to LINGO for processing with the following call:
' Run the script
dStatus = -1#
nError = LSexecuteScriptLng(pLINGO, cScript)
Note that dStatus is initialized to –1. LINGO returns the model status through memory transfer
location number 5 (i.e., @POINTER( 5)) to the dStatus variable. LINGO will only return a status
value, if it was able to solve the model. If an unexpected error were to occur, LINGO might not ever
reach the solution phase. In that case, dStatus would never be set. Initializing dStatus to a negative
value tests for this situation. Given that LINGO returns only non-negative status codes, a negative
status code would indicate a problem. This method of error trapping is effective, but not particularly
elegant. Another method that involves specifying an error callback routine is demonstrated below.
Now, LINGO’s log file may be closed down by calling LScloseLogFileLng():
' Close the log file
LScloseLogFileLng (pLINGO)
INTERFACING WITH OTHER APPLICATIONS 403
Next, the following code tests to see if LINGO was able to find an optimal solution:
' Problems?
If nError > 0 Or dStatus <> 0 Then
MsgBox ("Unable to solve!")
GoTo ErrorExit
End If
Note that the code returned in nError pertains only to the mechanical execution of the script processor.
It has nothing to do with the status of the model’s solution, which is returned in dStatus via the use of
the @POINTER() and @STATUS() functions (see the Staff Scheduling Example Using the LINGO
DLL section above). A model may actually be infeasible or unbounded, and the error code returned by
LSexecuteScriptLng()will give no indication. Thus, it is important to add a mechanism to return a
solution’s status to the calling application, as done here with the @STATUS() -> @POINTER(5) ->
dStatus link. The end result in the sample code is that “Unable to solve” is printed if either error
condition occurs. A more user-friendly application would offer more specific information regarding
the error condition.
As a final step, in order to avoid a memory leak in the application, remember to free up LINGO’s
environment when through:
LSdeleteEnvLng (pLINGO)
GoTo FinalExit:
If everything has been entered correctly, you should now be able to run the project.
Callback Functions
In many instances, solving a model can be a lengthy operation. Given this, it may be useful to provide
some device to keep the user posted as to the progress of the optimization. The standard interactive
version of LINGO displays a status window showing the iteration count, objective value, and various
other model statistics each time the solver is invoked. In addition, the user has the ability to interrupt
the solver by clicking on a button in the status window. You can provide similar functionality to users
through the LINGO DLL by supplying LINGO with a callback function—so named because the code
calls the LINGO solver, and the LINGO solver then calls back to the supplied callback routine.
In the next section, the calling sequences used in establishing a callback function are discussed. After
that, there are sample callback routines to the Visual C++ and Visual Basic staff scheduling examples
illustrated above.
404 CHAPTER 11
Specifying a Callback Function
To specify a callback function, the LINGO exported routine LSsetCallbackSolverLng() needs to be
called before calling LINGO’s script processor. The callback function will be called frequently by the
LINGO solver.
You will recall from above the calling sequence for LSsetCallbackSolverLng:
The box has three edit fields for the iterations, objective, and bound. There are two buttons—one to
interrupt the solver and the other to clear the dialog box.
406 CHAPTER 11
The next step is to use the ClassWizard to attach a handler class to the new dialog box. This class was
named CNewIPDlg. When a class has been attached to the dialog box, then the ClassWizard must be
used to assign member variables to handle the Iterations, Objective, and Bound edit fields. Once this is
done, the header file for the dialog box will resemble:
// NewIPDlg.h : header file
//
////////////////////////////////////////////////////////////////
// CNewIPDlg dialog
class CNewIPDlg : public CDialog
{
// Construction
public:
CNewIPDlg(CWnd* pParent = NULL); // standard constructor
// Dialog Data
//{{AFX_DATA(CNewIPDlg)
enum { IDD = IDD_NEW_IP_SOLUTION };
CString m_csBound;
CString m_csIteration;
CString m_csObjective;
//}}AFX_DATA
// Overrides
// ClassWizard generated virtual function overrides
//{{AFX_VIRTUAL(CNewIPDlg)
protected:
virtual void DoDataExchange(CDataExchange* pDX);
//}}AFX_VIRTUAL
// Implementation
protected:
// Generated message map functions
//{{AFX_MSG(CNewIPDlg)
// NOTE: the ClassWizard will add member functions here
//}}AFX_MSG
DECLARE_MESSAGE_MAP()
};
Callback Dialog Header File (NewIPDlg.h)
INTERFACING WITH OTHER APPLICATIONS 407
Here is the code to handle events from the new dialog box:
// NewIPDlg.h : header file
//
/////////////////////////////////////////////////////////////////
// CNewIPDlg dialog
class CNewIPDlg : public CDialog
{
// Construction
public:
CNewIPDlg(CWnd* pParent = NULL); // standard constructor
// Dialog Data
//{{AFX_DATA(CNewIPDlg)
enum { IDD = IDD_NEW_IP_SOLUTION };
CString m_csBound;
CString m_csIteration;
CString m_csObjective;
//}}AFX_DATA
// Overrides
// ClassWizard generated virtual function overrides
//{{AFX_VIRTUAL(CNewIPDlg) protected:
virtual void DoDataExchange(CDataExchange* pDX);
//}}AFX_VIRTUAL
// Implementation
protected:
// Generated message map functions
//{{AFX_MSG(CNewIPDlg)
// NOTE: the ClassWizard will add member functions here
//}}AFX_MSG
DECLARE_MESSAGE_MAP()
};
Callback Dialog Handler (NewIPDlg.cpp)
The code in these two files was entirely generated by the ClassWizard, requiring no actual user input.
408 CHAPTER 11
Next, the code for the callback function must be added. Here is a copy of the global routine that was
added for this purpose:
int __stdcall MyCallback( void* pModel, int nReserved,
void* pUserData)
{
// Callback function called by the LINGO solver
//
// return value: >= 0 if solver is to continue, else < 0 to interrupt
// Get current best IP
int nErr;
double dBestIP;
nErr = LSgetCallbackInfoLng( pModel,
LS_DINFO_MIP_BEST_OBJECTIVE_LNG, &dBestIP);
if ( nErr) return( 0);
// Get best IP value published in dialog box
double* pdBestIPShown = (double*) pUserData;
// Is current better than incumbent?
if ( dBestIP < *pdBestIPShown)
{
// Yes ... save its value
*pdBestIPShown = dBestIP;
// Get iteration count from LINGO
int nIterations;
LSgetCallbackInfoLng( pModel, LS_IINFO_ITERATIONS_LNG,
&nIterations);
// Get bound on solution
double dBound;
LSgetCallbackInfoLng( pModel, LS_DINFO_MIP_BOUND_LNG, &dBound);
// Create a dialog to show the current solution value
CNewIPDlg dlgNewIP;
// Initialize the fields of the dialog
dlgNewIP.m_csIteration.Format( "%d", nIterations);
dlgNewIP.m_csObjective.Format( "%d", (int) dBestIP);
dlgNewIP.m_csBound.Format( "%d", (int) dBound);
// Post the dialog and check for a user interrupt
if ( dlgNewIP.DoModal() == IDCANCEL) return( -1);
}
return( 0);
}
Callback Routine
Of course, this routine has the same interface that was described in the previous section:
int __stdcall MySolverCallback( pLSenvLINGO pL, int nReserved, void* pUserData)
INTERFACING WITH OTHER APPLICATIONS 409
The following code uses the LSgetCallbackInfo routine to get the value of the current best integer
solution:
// Get current best IP
int nErr;
double dBestIP;
nErr = LSgetCallbackInfoLng( pModel,
LS_DINFO_MIP_BEST_OBJECTIVE_LNG, &dBestIP);
if ( nErr) return( 0);
Note that this statement references the user data pointer, pUserData. This pointer is passed to LINGO
when the callback function is established, and it is useful as a means of accessing the data from within
the callback function. In this particular case, a single variable is being pointing to. If needed, the
pointer could reference a large data structure containing whatever data desired.
Next, test to see if the latest integer solution is better than the incumbent solution with the statement:
// Is current better than incumbent?
if ( dBestIP < *pdBestIPShown)
If the new solution is better, then we get additional information from LINGO on the iteration count and
solution bound. Also, an instance of the callback dialog box to display the latest information in is
created with the line:
// Create a dialog to show the current solution value
CNewIPDlg dlgNewIP;
The new data is then loaded into the fields of the dialog box:
// Initialize the fields of the dialog
dlgNewIP.m_csIteration.Format( "%d", nIterations);
dlgNewIP.m_csObjective.Format( "%d", (int) dBestIP);
dlgNewIP.m_csBound.Format( "%d", (int) dBound);
As the final step in this callback routine, we display the dialog box, and if the user hits the interrupt
button we return a –1, indicating that the solver should stop and return with the best answer found so
far:
// Post the dialog and check for a user interrupt
if ( dlgNewIP.DoModal() == IDCANCEL) return( -1);
410 CHAPTER 11
At this point, there is one more piece of code to add. We must make a call to the
LSsetCallbackSolverLng() routine to pass LINGO a pointer to our callback routine. A good place to
do this is right after creating the LINGO environment in the OnSolve() handler code for our Solve
button. The changes are listed below in bold type:
// create the LINGO environment object
pLSenvLINGO pLINGO;
pLINGO = LScreateEnvLng();
if ( !pLINGO)
{
AfxMessageBox("Unable to create LINGO Environment");
return;
}
// Pass LINGO a pointer to our callback function
nError = LSsetCallbackSolverLng( pLINGO, &MyCallback,
&dBestIPShown);
if ( nError) goto ErrorExit;
// Open LINGO's log file
nError = LSopenLogFileLng( pLINGO, "\\LINGO.log");
if ( nError) goto ErrorExit;
Note: A VB callback function must be placed in a Module file (.bas file) . The callback function
will not work if it is placed in a Forms file (.frm file). Also, the callback function and the
module file must have different names. If the module file has the same name as the callback
function, then the VB AddressOf operator will not be able to return the address of the callback
function.
You will recall from the section Specifying a Callback Function above that the callback routine must
use the following calling sequence:
int __stdcall MySolverCallback( pLSenvLINGO pL, int nReserved, void* pUserData)
An equivalent function definition using VB code is:
Public Function MySolverCallback(ByVal pModel As Long, _
ByVal nReserved As Long, ByRef dBestIP As Double) As Long
VB uses the standard call (__stdcall) convention by default, so we need not specify this explicitly.
We will make use of the user data pointer to pass the value of the best objective displayed so far in the
dBestIP argument. This variable will hold the objective value of the best integer solution found so far.
We compare each new objective value to the best one found so far. If the latest is an improvement
over the incumbent, then we display a dialog box summarizing the new solution.
412 CHAPTER 11
The following code uses the LSgetCallbackInfo routine to get the value of the current best integer
solution:
' Get current best IP
Dim dObj As Double
Dim nError As Long
nError = LSgetCallbackInfoDoubleLng(pModel, _
LS_DINFO_MIP_BEST_OBJECTIVE_LNG, dObj)
In the VB header file for LINGO (LINGD10.BAS), we created two aliases for the
LSgetCallbackInfoLng() function: LSgetCallbackInfoDoubleLng() and LSgetCallbackInfoLongLng().
These were for retrieving, respectively, double and long data from LINGO. This is required due to VB
not supporting the void data type found in C. We use LSgetCallbackInfoDoubleLng() to retrieve the
objective value given that it is a double precision quantity.
Next, we check for any errors in retrieving the objective value. If none occurred, we check to see if the
latest objective is better than the incumbent:
' Check for any error
If (nError = LSERR_NO_ERROR_LNG) Then
' Is it better than the best one displayed so far?
If (dObj < dBestIP) Then
If the new objective is better than the incumbent, then we save the new objective value, and retrieve
the iteration count and objective bound:
' Save the new best objective value
dBestIP = dObj
' Get the iteration count from LINGO
Dim nIterations As Long
nResult = LSgetCallbackInfoLongLng(pModel, _
LS_IINFO_ITERATIONS_LNG, nIterations)
' Get the objective bound from LINGO
Dim dBound As Double
nResult = LSgetCallbackInfoDoubleLng(pModel, _
LS_DINFO_MIP_BOUND_LNG, dBound)
At this point, there is one more piece of code to add. We must make a call to the
LssetCallbackSolverLng() routine to pass LINGO a pointer to our callback routine. This is
accomplished at the start of the Solve button handler routine with the call:
' Pass LINGO a pointer to the callback routine
Dim nError As Long
Dim dBestObj As Double
dBestObj = 1E+30
nError = LSsetCallbackSolverLng(pLINGO, _
AddressOf MySolverCallback, dBestObj)
Note the use of the VB AddressOf operator in the call to LSsetCallbackSolverLng(). This operator
may be used only in function calls to pass the address of routines contained in module files.
Note: The AddressOf operator was added to VB starting with release 5.0. Thus, earlier releases of
VB won’t be able to exploit the callback feature in LINGO. Also, Visual Basic for
Applications (VBA), the VB macro capability supplied with Microsoft Office, does not
support the AddressOf operator. Thus, VBA applications calling LINGO will also not be able
to establish callback routines.
414 CHAPTER 11
Summary
The LINGO DLL has a very simple structure. You need only acquaint yourself with a handful of
functions in order to access the DLL and add the power of the LINGO solver to your applications.
We’ve given brief examples on calling the DLL from Visual Basic and Visual C++. Additional
examples are provided in the Programming Samples. Application developers working in other
environments should be able to access the LINGO DLL in a fashion similar to the examples given
here.
Finally, keep in mind that any application you build that makes use of the LINGO DLL is protected
under the LINGO License Agreement. Thus, you may not distribute such applications without explicit
permission from LINDO Systems. If you would like to make arrangements for distributing your
application, please feel free to contact LINDO Systems regarding available runtime licensing
arrangements.
Click the OK button to finish creating the base code for our DLL.
INTERFACING WITH OTHER APPLICATIONS 417
Now, edit the SQROOT.CPP file and add the modifications listed below in bold:
// sqroot.cpp : Defines the initialization
// routines for the DLL.
//
#include "stdafx.h"
#include "sqroot.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/////////////////////////////////////////////////////
// CSqrootApp
BEGIN_MESSAGE_MAP(CSqrootApp, CWinApp)
//{{AFX_MSG_MAP(CSqrootApp)
//NOTE-the ClassWizard will add and
// remove mapping macros here.
// DO NOT EDIT what you see in these
// blocks of generated code!
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
CSqrootApp::CSqrootApp()
{
// The constructor
// Remove next line for a "quiet" version
// of MyUser.DLL
AfxMessageBox("@USER DLL installed");
}
CSqrootApp theApp;
#include <math.h>
extern "C" __declspec(dllexport)
void MyUser(int* pnNumberOfArgs,
double* pdArgs, double* dResult)
{
// This is an @USER routine callable by LINGO. In
// this particular case we simply take the
// square root of the first argument.
*dResult = sqrt(*pdArgs);
}
File: SQROOT.CPP
418 CHAPTER 11
You should now be able to build the DLL. When Visual C++ completes the build, copy the
SQROOT.DLL file to LINGO’s startup directory (the one where LINGO10.EXE is located) and rename
SQROOT.DLL to be MYUSER.DLL. Now, start LINGO and you should see the following dialog box
confirming the DLL was successfully loaded:
Input a small model to compute the square root of 9 and solve it to get the following results:
If you don’t have a copy of Visual C++, you may experiment with this @USER routine by copying the
DLL supplied with LINGO into your LINGO startup directory. You can find the SQROOT.DLL file in
the USER\VC++ subdirectory off the main LINGO directory.
12 Developing More
Advanced Models
In this chapter, we will walk through the development of models in a handful of different application
areas. The goal here is twofold. On the one hand, walking through a number of different models should
help hone your skills at building LINGO models. Secondly, by applying LINGO to an array of
different application areas, we hope to illustrate the potential benefits of mathematical modeling in
almost any business situation. Below is a list of the application areas we will be touching upon in this
chapter:
♦ Production Management
♦ Logistics
♦ Finance
♦ Queuing
♦ Marketing
The reader interested in additional examples of LINGO modeling should refer to Optimization
Modeling with LINGO, by Linus Schrage, published by LINDO Systems. This book contains an
exhaustive set of modeling examples from a wide array of application areas. In many cases, you should
be able to find a model that will suit your needs with only modest modifications. Interested readers
may also refer to Appendix A, Additional Examples of LINGO Modeling, to find additional samples of
LINGO models.
419
420 CHAPTER 12
The Model
MODEL:
TITLE BLEND;
SETS:
!Each raw material has an availability
and cost/unit;
RAWMAT/ BUTANE, CATREF, NAPHTHA/: AVAIL, COST;
!Each finished good has a min required,
max sellable, selling price,
and batch size to be determined;
FINGOOD/ REGULAR, PREMIUM/:
MINREQ, MAXSELL, PRICE, BATCH;
!Here is the set of quality measures;
QUALMES/ OCTANE, VAPOR, VOLATILITY/;
!For each combo of raw material and
quality measure there is a quality
level;
RXQ(RAWMAT, QUALMES): QLEVEL;
!For each combination of quality
measure and finished good there are
upper and lower limits on quality,
and a slack on upper quality to be
determined;
QXF(QUALMES, FINGOOD):
QUP, QLOW, QSLACK;
!For each combination of raw material
and finished good there is an amount
Developing More Advanced Models 421
of raw material used to be solved for;
RXF(RAWMAT, FINGOOD): USED;
ENDSETS
DATA:
!Raw material availability;
AVAIL = 1000, 4000, 5000;
!Raw material costs;
COST = 7.3, 18.2, 12.5;
!Quality parameters of raw
materials;
QLEVEL = 120, 60, 105,
100, 2.6, 3,
74, 4.1, 12;
!Limits on finished goods;
MINREQ = 4000, 2000;
MAXSELL = 8000, 6000;
!Finished goods prices;
PRICE = 18.4, 22;
!Upper and lower limits on
quality for each finished good;
QUP = 110, 110,
11, 11,
25, 25;
QLOW = 90, 95,
8, 8,
17, 17;
ENDDATA
!Subject to raw material availability;
@FOR(RAWMAT(R):
[RMLIM] @SUM(FINGOOD(F): USED(R, F))
<= AVAIL(R);
);
@FOR(FINGOOD(F):
!Batch size computation;
[BATCOMP] BATCH(F) =
@SUM(RAWMAT(R): USED(R, F));
!Batch size limits;
@BND(MINREQ, BATCH, MAXSELL);
!Quality restrictions for each
quality measure;
@FOR(QUALMES(Q):
[QRESUP] @SUM(RAWMAT(R):
QLEVEL(R, Q) * USED(R, F))
+ QSLACK(Q, F) = QUP(Q, F) *
BATCH(F);
[QRESDN] QSLACK(Q, F) <=
(QUP(Q, F) - QLOW(Q, F)) *
BATCH(F);
);
);
422 CHAPTER 12
The Sets
We have three primitive sets in this model raw materials (RAWMAT), finished goods (FINGOOD),
and quality measures (QUALMES).
From these three primitive sets, we create three derived sets.
The first derived set, RXQ, is a dense derived set and is a cross on the raw materials set and the quality
measures set. We need this derived set in order to define an attribute to hold the quality values for the
raw materials.
The second derived set, QXF, is a cross on the quality measures and the finished goods. We need this
set because we will be concerned with the levels of the quality measures in the finished goods.
The final derived set, RXF, is a cross on the raw materials and finished goods. We need this set
because we need to compute the amount of raw material used in each finished good.
The Variables
The primary variable that drives this model is the amount of each raw material used in each finished
good (USED). We have also added two other variables for computation purposes. First, there is the
BATCH variable, which is used to compute the batch size of each finished good. There is also the
QSLACK variable, which computes the slack on the upper quality limit for each finished good and
each quality measure.
The Objective
The objective in this model is to maximize the total profit contribution. This is computed using the
following expression:
[OBJECTIVE] MAX =
@SUM(FINGOOD: PRICE * BATCH) -
@SUM(RAWMAT(R): COST(R) *
@SUM(FINGOOD(F): USED(R, F)));
Total profit, of course, is total revenue minus total expenses. Total revenue is the sum over the finished
goods of the price of each finished good multiplied by the batch size of each finished good, or, in
LINGO syntax:
@SUM(FINGOOD: PRICE * BATCH)
Total expenses are computed by taking the sum over the raw materials of the price of the raw material
multiplied by the amount of each raw material used. This is computed with the expression:
@SUM(RAWMAT(R): COST(R) *
@SUM(FINGOOD(F): USED(R, F))
Developing More Advanced Models 423
The Constraints
There are four sets of constraints in this model. Two of them are merely computational.
This first computational set of constraints computes the batch size of each finished good with the
following expression:
!Batch size computation;
[BATCOMP] BATCH(F) =
@SUM(RAWMAT(R): USED(R, F));
In words, the batch size of finished good F is the sum of all the raw materials used in the production of
the finished good.
The second set of computational constraints computes the slack on the upper quality limit for each
finished good and each quality measure as follows:
[QRESUP] @SUM(RAWMAT(R):
QLEVEL(R, Q) * USED(R, F))
+ QSLACK(Q, F) = QUP(Q, F) *
BATCH(F);
In words, the actual quality level plus the slack on the upper quality level is equal to the upper quality
level.
The first true constraint is on the finished products’ batch sizes, which must fall within minimum and
maximum levels. We use the @BND function to set this up as follows:
!Batch size limits;
@BND(MINREQ, BATCH, MAXSELL);
Note that we could have entered explicit constraints for the batch size limits, but the @BND function is
more efficient at handling simple bounds on variables.
The final constraint forces the quality level of each finished good for each quality measure to fall
within specifications. We do this with the following:
[QRESDN] QSLACK(Q, F) <=
(QUP(Q, F) - QLOW(Q, F)) *
BATCH(F);
In words, the slack must be less than the difference between the upper limit and the lower limit. If this
were not true, quality would be beneath the lower limit. The fact that the slack variable cannot be
negative guarantees we won’t exceed the quality limit.
The Solution
Rounding all solution values to the nearest whole integer, total profit is $44,905 with batch sizes of
4,000 Regular and 4,095 Premium. The following matrix shows the recommended quantities of each
raw material to use in the Regular and Premium blends:
Raw Material Regular Premium
Butane 534 466
Catalytic Reformate 1,516 2,484
Naphtha 1,950 1,146
424 CHAPTER 12
DATA:
! Parts list;
PART = @FILE( 'MRP.LDT');
! Time periods;
TIME = @FILE( 'MRP.LDT');
ENDDATA
DATA:
426 CHAPTER 12
The Data
An interesting feature of this model is all of the data is imported from an external text file using the
@FILE function (for more information on @FILE, see Chapter 8, Interfacing with External Files).
More specifically, @FILE reads all the data from a single file, MRP.LDT, that is pictured below:
! Parts list;
U, ! Unicycles;
B, ! Bicycles;
T, ! Tandems;
S, ! Seats;
W, ! Wheels;
C, ! Chains;
H, ! Hubs;
P, ! sPokes;
L~ ! Links;
The Sets
We have two primitive sets in this modelthe component parts (PARTS) and the time periods (TIME).
From these two primitive sets, we create two derived sets.
The first derived set, USES, is a sparse set derived from the cross of the PARTS set on itself. We use
this set to construct a data table, or input-output matrix (NEEDS), containing the parts usage data,
which tells us how many of each of the other parts are required to produce a given part. The set is
sparse because not all parts are required in the production of certain other parts (e.g., chains are not
required to produce a spoke).
The other derived set, PXT, is a dense set formed from the cross on the parts and time periods sets. We
need this set because we will be concerned with the demand for each part in each period and the
amount of each part to begin producing in each period.
The Variables
The only unknown in this model is the total demand (TD) attribute, where TD(p, t) is the total demand
for product p in period t. Total demand stems from two sources—external demand (ED) from
customers and internal demand for production requirements. We compute TD by incorporating the
part’s lead time. Thus, when TD is nonzero in a period, production must begin in that period.
The Formulas
The key computation in this model is:
! For each part P and period T, the total demand =
external demand + demand generated by parents
one lead time in the future;
@FOR(PXT(P, T) | T + LT(P) #LE# NP:
TD(P, T) = ED(P, T + LT(P)) +
@SUM(USES(P2, P): TD(P2, T + LT(P)) *
NEEDS(P2, P));
);
For each part in each time period, the amount of that part we must begin producing in the period is the
amount we will need one lead time away to satisfy 1) external demand, and 2) internal production
requirements. The subexpression that gives external demand is simply:
ED(P, T + LT(P))
The remainder of the expression:
@SUM(USES(P2, P): TD(P2, T + LT(P)) *
NEEDS(P2, P));
sums up the amount of the part needed for internal production of all other parts one lead time away.
Note, we place a logical condition on the outer @FOR loop in this calculation (shown here in bold):
@FOR(PXT(P, T) | T + LT(P) #LE# NP:
Without this condition, the calculation would extend beyond the final period in the TIME set.
428 CHAPTER 12
The Solution
Solving the model, we get the following nonzero values for TD:
Variable Value
TD(U, 7) 10.00000
TD(B, 7) 20.00000
TD(T, 8) 20.00000
TD(S, 6) 30.00000
TD(S, 7) 40.00000
TD(W, 4) 50.00000
TD(W, 5) 40.00000
TD(C, 6) 20.00000
TD(C, 7) 40.00000
TD(H, 3) 50.00000
TD(H, 4) 40.00000
TD(P, 2) 1800.000
TD(P, 3) 1440.000
TD(L, 4) 1680.000
TD(L, 5) 3360.000
Solution: MRP
Putting this solution in tabular form, we get the following production schedule:
2 3 4 5 6 7 8
Unicycles 10
Bicycles 20
Tandems 20
Seats 30 40
Wheels 50 40
Chains 20 40
Hubs 50 40
Spokes 1,800 1,440
Links 1,680 3,360
MRP Production Schedule
Note the use of the @TABLE output function in the data section at the end of the model that displays
the TD attribute in table form similar to the table above:
The production schedule:
1 2 3 4 5 6 7 8 9
U 0 0 0 0 0 0 10 0 0
B 0 0 0 0 0 0 20 0 0
T 0 0 0 0 0 0 0 20 0
S 0 0 0 0 0 30 40 0 0
W 0 0 0 50 40 0 0 0 0
C 0 0 0 0 0 20 40 0 0
H 0 0 50 40 0 0 0 0 0
P 0 1800 1440 0 0 0 0 0 0
L 0 0 0 1680 3360 0 0 0 0
Developing More Advanced Models 429
A B C
G
J K
H
D E
I
The times to complete the various tasks are given in the table below:
Task: A B C D E F G H I J K
Minutes: 45 11 9 50 15 12 12 12 12 8 9
We need to find an assignment of tasks to workstations that minimize the assembly line's cycle time.
430 CHAPTER 12
The Model
MODEL:
! Assembly line balancing model;
!This model involves assigning tasks to stations in an
assembly line so bottlenecks are avoided. Ideally, each
station would be assigned an equal amount of work.;
SETS:
! The set of tasks to be assigned are A through
K, and each task has a time to complete, T;
TASK/ A B C D E F G H I J K/: T;
! Some predecessor,successor pairings must be
observed(e.g. A must be done before B, B
before C, etc.);
PRED(TASK, TASK)/ A,B B,C C,F C,G F,J G,J
J,K D,E E,H E,I H,J I,J /;
! There are 4 workstations;
STATION/1..4/;
TXS(TASK, STATION): X;
! X is the attribute from the derived set TXS
that represents the assignment. X(I,K) = 1
if task I is assigned to station K;
ENDSETS
DATA:
! Data taken from Chase and Aquilano, POM;
! There is an estimated time required for each
task:
A B C D E F G H I J K;
T = 45 11 9 50 15 12 12 12 12 8 9;
ENDDATA
! The model;
! *Warning* may be slow for more than 15 tasks;
! For each task, there must be one assigned station;
@FOR(TASK(I): @SUM(STATION(K): X(I, K)) = 1);
! Precedence constraints;
! For each precedence pair, the predecessor task
I cannot be assigned to a later station than
its successor task J;
@FOR(PRED(I, J):
@SUM(STATION(K):
K * X(J, K) - K * X(I, K)) >= 0);
Developing More Advanced Models 431
! For each station, the total time for the assigned tasks must
less than the maximum cycle time, CYCTIME;
@FOR(STATION(K):
@SUM(TXS(I, K): T(I) * X(I, K)) <= CYCTIME);
! Minimize the maximum cycle time;
MIN = CYCTIME;
! The X(I,J) assignment variables are
binary integers;
@FOR(TXS: @BIN(X));
END
Model: ASLBAL
The Sets
We have two primitive sets in this modelthe tasks (TASK) and the workstations (STATION). From
these two primitive sets, we create two derived sets.
The first derived set, PRED, is a sparse derived set and is based on a cross of the TASK set on itself.
The members of this set are the precedence relations amongst the tasks. For instance, the first member
of this set is the pair (A,B), indicating task A must precede task B.
The other derived set, TXS, is a dense derived set formed by taking the cross of the task set on the
workstation set. We need this set because we will be determining what tasks get assigned to what
workstations.
The Variables
The decision variables in this model are the members of the X attribute that is defined in the TXS set.
X(t, s) is a binary variable that is 1 if task t is assigned to station s, otherwise 0. The X attribute is
forced to being binary in the expression:
! The X(I,J) assignment variables are
binary integers;
@FOR(TXS: @BIN(X));
We also introduce the scalar variable, CYCTIME, to represent the entire assembly line’s cycle time,
which is computed by taking the maximum cycle time over the workstations.
The Objective
The objective in this model is simply to minimize total cycle time for the line and is given as:
! Minimize the maximum cycle time;
MIN = CYCTIME;
The Constraints
We have the following three types of constraints in this model:
1. each task must be assigned to one station,
2. precedence relations must be observed amongst the tasks, and
3. the line cycle time variable, CYCTIME, must be greater-than-or-equal-to the
actual cycle time.
432 CHAPTER 12
The following expression sums up the assignment variable for each task, and sets the sum to equal 1:
! For each task, there must be one assigned station;
@FOR(TASK(I): @SUM(STATION(K): X(I, K)) = 1);
This forces each task to be assigned to a single station.
We use the following expression to enforce the precedence relationships amongst the tasks:
! Precedence constraints;
! For each precedence pair, the predecessor task
I cannot be assigned to a later station than its
successor task J;
@FOR(PRED(I, J):
@SUM(STATION(K):
K * X(J, K) - K * X(I, K)) >= 0);
Suppose task I is a predecessor to task J. If I were incorrectly assigned to a workstation later than J, the
sum of the terms K * X(I, K) would exceed the sum of the terms K * X(J, K) and the constraint would
be violated. Thus, this constraint effectively enforces the predecessor relations.
We restrain the cycle time using the following constraints:
! For each station, the total time for the
assigned tasks must be less than the maximum
cycle time, CYCTIME;
@FOR(STATION(K):
@SUM(TXS(I, K): T(I) * X(I, K)) <= CYCTIME);
The quantity:
@SUM(TXS(I, K): T(I) * X(I, K))
in this constraint computes the cycle time for station K. We use the @FOR statement to make the
CYCTIME variable greater-than-or-equal-to the cycle times for all the workstations. If we couple this
with the fact that we are minimizing CYCTIME in the objective, CYCTIME will be “squeezed” into
exactly equaling the maximum of the cycle times for the workstations.
By “squeezing” CYCTIME to the correct value, we avoid using the @MAX function. Had the @MAX
function been used, LINGO would have had to resort to its nonlinear solver to handle the piecewise
linear @MAX. Avoiding nonlinear models whenever possible is a critical modeling practice.
Developing More Advanced Models 433
The Solution
Solving the model, we get the following nonzero values for the assignment X variable:
Variable Value
X(A, 2) 1.000000
X(B, 3) 1.000000
X(C, 4) 1.000000
X(D, 1) 1.000000
X(E, 3) 1.000000
X(F, 4) 1.000000
X(G, 4) 1.000000
X(H, 3) 1.000000
X(I, 3) 1.000000
X(J, 4) 1.000000
X(K, 4) 1.000000
Solution: ASLBAL
Summarizing this solution, we have:
Workstation Assigned Tasks Cycle Time
1 D 50
2 A 45
3 B, E, H, I 50
4 C, F, G, J, K 50
The cycle time for the entire line is 50 minutesthe maximum of the cycle times across all the
workstations. We have a well-balanced line in that only workstation 2 has slack time totaling 5
minutes.
434 CHAPTER 12
Logistics Models
Capacitated Plant Location Model: CAPLOC
Background
The capacitated plant location model is a generalization of the transportation model we introduced in
Chapter 1, Getting Started with LINGO. The capacitated plant location problem allows greater latitude
of decision making in that the points of origin (plant locations) are variable. Manufacturers and
wholesale businesses are likely to encounter problems of this sort in matching existing customer
demand to product availability and minimizing transportation costs.
The Model
MODEL:
! Capacitated Plant Location Problem;
SETS:
PLANTS: FCOST, CAP, OPEN;
CUSTOMERS: DEM;
ARCS( PLANTS, CUSTOMERS) : COST, VOL;
ENDSETS
DATA:
! The plant, their fixed costs
and capacity;
PLANTS, FCOST, CAP =
P1 91 39
P2 70 35
P3 24 31;
! Customers and their demands;
CUSTOMERS, DEM =
C1 15
C2 17
C3 22
C4 12;
! The plant to cust cost/unit
shipment matrix;
COST = 6 2 6 7
4 9 5 3
8 8 1 5;
ENDDATA
! The objective;
[TTL_COST] MIN = @SUM( ARCS: COST * VOL) +
Developing More Advanced Models 435
@SUM( PLANTS: FCOST * OPEN);
The Sets
We have two primitive sets in this modelthe plants (PLANTS) and the customers (CUSTOMERS).
From these two primitive sets, we create a dense derived set, ARCS, which is the cross of the plants
and customers sets. We use this set to represent the shipping arcs between the plants and customers.
The Variables
There are two sets of decision variables in this model. The VOL attribute, defined on the ARCS set,
represents the shipment volume from the plants to the customers along each arc. The OPEN attribute,
defined on the PLANTS set, is used to represent the plants that are open. Specifically, OPEN(p) is 1 if
plant p is opened, else it is 0. The members of the OPEN attribute are set to being binary using the
expression:
! Make OPEN binary(0/1);
@FOR(PLANTS: @BIN(OPEN));
The Objective
The objective in this model is to minimize total costs, which is the sum of the shipping costs and fixed
plant costs. This is computed using the following expression:
! The objective;
[TTL_COST] MIN = @SUM(ARCS: COST * VOL) +
@SUM(PLANTS: FCOST * OPEN);
The shipping cost component of the objective is computed with:
@SUM(ARCS: COST * VOL)
while the fixed plant costs component is given by:
@SUM(PLANTS: FCOST * OPEN)
436 CHAPTER 12
The Constraints
There are two sets of constraints in the model:
1. each customer must be sent enough product to satisfy demand, and
2. each plant can’t supply more than its capacity.
The following expression guarantees each customer receives the quantity of product demanded:
! The demand constraints;
@FOR(CUSTOMERS(J): [DEMAND]
@SUM(PLANTS(I): VOL(I, J)) >= DEM(J)
);
For each customer, we sum the amount being shipped to that customer and set it to be
greater-than-or-equal-to the customer’s demand.
To limit shipments from a plant to the plant’s capacity, we use:
! The supply constraints;
@FOR(PLANTS(I): [SUPPLY]
@SUM(CUSTOMERS(J): VOL(I, J)) <=
CAP(I) * OPEN(I)
);
For each plant, we sum up the amount being shipped from the plant and set this quantity to be
less-than-or-equal-to the plant’s capacity multiplied by the plant’s 0/1 OPEN indicator. Note that, in
order for the plant to be able to ship any quantity of product, the OPEN binary variable will be forced
to 1 by these constraints.
Developing More Advanced Models 437
The Solution
Solving the model, we get the following solution:
Global optimal solution found.
Objective value: 327.0000
Extended solver steps: 4
Total solver iterations: 25
Variable Value
OPEN(P1) 1.000000
OPEN(P3) 1.000000
VOL(P1, C1) 15.00000
VOL(P1, C2) 17.00000
VOL(P1, C4) 3.000000
VOL(P3, C3) 22.00000
VOL(P3, C4) 9.000000
Total costs are minimized at 327 by opening plants 1 and 3. From plant 1, we ship 15, 17, and 3 units
respectively to customers 1, 2, and 4. From plant 3, we ship 22 units to customer 3, and 9 units to
customer 4.
438 CHAPTER 12
where F(i) is the minimal travel distance from point i to the final destination point, and D(i, j) is the
distance from point i to point j. In words, the minimal distance from node i to the terminal node is the
minimum over all points reachable along a single arc from i of the sum of the distance from i to the
adjoining node plus the minimal distance from the adjoining node to the terminal node.
2 5
8
1 3 6 10
9
4 7
Links are assumed to be one-way. The distances between the cities are given to us. We want to
determine the shortest distance between cities 1 and 10.
Developing More Advanced Models 439
The Model
SETS:
! Dynamic programming illustration (see
Anderson, Sweeney & Williams, An Intro to Mgt
Science, 6th Ed.). We have a network of 10
cities. We want to find the length of the
shortest route from city 1 to city 10.;
! Here is our primitive set of ten cities,
where F(i) represents the shortest path
distance from city i to the last city;
CITIES /1..10/: F;
! The derived set ROADS lists the roads that
exist between the cities (note: not all city
pairs are directly linked by a road, and
roads are assumed to be one way.);
ROADS(CITIES, CITIES)/
1,2 1,3 1,4
2,5 2,6 2,7
3,5 3,6 3,7
4,5 4,6
5,8 5,9
6,8 6,9
7,8 7,9
8,10
9,10/: D;
! D(i, j) is the distance from city i to j;
ENDSETS
DATA:
! Here are the distances that correspond to the
above links;
D =
1 5 2
13 12 11
6 10 4
12 14
3 9
6 5
8 10
5
2;
ENDDATA
! If you are already in City 10, then the cost
to travel to City 10 is 0;
F(@SIZE(CITIES)) = 0;
! The following is the classic dynamic
programming recursion. In words, the shortest
distance from City i to City 10 is the minimum
over all cities j reachable from i of the sum
of the distance from i to j plus the minimal
distance from j to City 10;
@FOR(CITIES(i)| i #LT# @SIZE(CITIES):
F(i) = @MIN(ROADS(i, j): D(i, j) + F(j))
);
Model: DYNAMB
440 CHAPTER 12
The Sets
We have the single primitive set, CITIES, which corresponds to the cities in the network. From this
primitive set, we form a single derived set, ROADS, to represent the links between the cities. We
specify the members of this set. Thus, ROADS is a sparse derived set.
The Variables
The F attribute defined on the CITIES set is used to store the distance from each city to the destination
city.
The Formulas
The recursion, discussed above, is entered in LINGO with the following statement:
@FOR(CITIES(i)| i #LT# @SIZE(CITIES):
F(i) = @MIN(ROADS(i, j): D(i, j) + F(j))
);
The Solution
Solving the model, we get the following values for F:
Variable Value
F(1) 19.00000
F(2) 19.00000
F(3) 14.00000
F(4) 20.00000
F(5) 8.000000
F(6) 7.000000
F(7) 12.00000
F(8) 5.000000
F(9) 2.000000
F(10) 0.000000
F(1), the shortest distance from city 1 to city 10, gives us the distance of the shortest path of 19. For
the curious reader, this distance corresponds to the path 1→3→5→8→10. Refer to model
DYNAMB2.LG4 to see how to extend this model to compute the actual path as well as its distance.
Developing More Advanced Models 441
Financial Models
Markowitz Portfolio Selection Model: GENPRT
Background
In the March, 1952 issue of Journal of Finance, Harry M. Markowitz published an article titled
Portfolio Selection. In the article, he demonstrates how to reduce the risk of asset portfolios by
selecting assets whose values aren’t highly correlated. At the same time, he laid down some basic
principles for establishing an advantageous relationship between risk and return. This has come to be
known as diversification of assets. In other words, don’t put all your eggs in one basket.
A key to understanding the Markowitz model is to be comfortable with the statistic known as the
variance of a portfolio. Mathematically, the variance of a portfolio is:
∑i∑j Xi Xj σi,j
where,
Xi is the fraction of the portfolio invested in asset i,
σi,j for i≠j: the covariance of asset i with asset j, and
for i=j: the variance of asset i.
Variance is a measure of the expected fluctuation in return—the higher the variance, the riskier the
investment. The covariance is a measure of the correlation of return fluctuations of one stock with the
fluctuations of another. High covariance indicates an increase in one stock’s return is likely to
correspond to an increase in the other. A covariance close to zero means the return rates are relatively
independent. A negative covariance means that an increase in one stock’s return is likely to correspond
to a decrease in the other.
The Markowitz model seeks to minimize a portfolio’s variance, while meeting a desired level of
overall expected return.
The Model
! GENPRT: Generic Markowitz portfolio;
SETS:
ASSET/1..3/: RATE, UB, X;
COVMAT(ASSET, ASSET): V;
ENDSETS
DATA:
! The data;
! Expected growth rate of each asset;
RATE = 1.3 1.2 1.08;
! Upper bound on investment in each;
UB = .75 .75 .75;
! Covariance matrix;
V = 3 1 -.5
1 2 -.4
-.5 -.4 1;
! Desired growth rate of portfolio;
GROWTH = 1.12;
ENDDATA
! The model;
! Min the variance;
[VAR] MIN = @SUM(COVMAT(I, J):
V(I, J) * X(I) * X(J));
! Must be fully invested;
[FULL] @SUM(ASSET: X) = 1;
! Upper bounds on each;
@FOR(ASSET: @BND(0, X, UB));
! Desired value or return after 1 period;
[RET] @SUM(ASSET: RATE * X) >= GROWTH;
Model: GENPRT
The Sets
We define a single primitive set, ASSETS, corresponding to the three stocks in the model. From the
ASSETS set, we derive the dense set named COVMAT, which is the cross of the ASSETS set on itself.
We use the COVMAT set for defining the covariance matrix.
The Attributes
We define four attributes in this model.
The RATE, UB and V attributes are for storing data. RATE stores the expected return for each asset, UB
stores the upper bound on the fraction of the asset allowed in the portfolio, and V stores the covariance
matrix. (Note the covariance matrix is symmetric and larger portfolio models would benefit from
storing just half of the matrix, rather than the entire matrix as we have done here for simplicity
reasons.)
The final attribute, X, constitutes the decision variables for the model. Specifically, X(i) is the fraction
of the portfolio devoted to asset i.
Developing More Advanced Models 443
The Objective
The objective in this model is to minimize the portfolio’s risk. As we mentioned above, we use the
portfolio’s variance as a measure of risk in the following expression:
! Min the variance;
[VAR] MIN = @SUM(COVMAT(I, J):
V(I, J) * X(I) * X(J));
The Constraints
There are three forms of constraints in the model:
1. we must be fully invested,
2. we can’t invest too much in any one asset, and
3. expected return should meet, or exceed, our threshold level of 12%.
The following expression enforces the 100% investment requirement:
! Must be fully invested;
[FULL] @SUM(ASSET: X) = 1;
In words, the sum of all the weights of all the assets in the portfolio must equal 1. Without this
constraint, LINGO will tend to under invest in order to get a lower variance. You can confirm this by
dropping the constraint and running the model.
To keep the solution from investing too much in any one asset, we use the @BND function:
! Upper bounds on each;
@FOR(ASSET: @BND(0, X, UB));
As you recall, the @BND function is the most efficient method for placing simple bounds on variables.
We constrain the portfolio’s expected return to be greater-than-or-equal-to our target with the
expression:
! Desired value or return after 1 period;
[RET] @SUM(ASSET: RATE * X) >= GROWTH;
The left-hand side of this expression is the expected rate of return, and is the sum of the fractions
invested in each asset weighted by each asset’s return.
444 CHAPTER 12
The Solution
Solving the model, we get the following solution:
Local optimal solution found.
Objective value: 0.4173749
Total solver iterations: 13
Variable Value
X(1) 0.1548631
X(2) 0.2502361
X(3) 0.5949008
Row Slack or Surplus
VAR 0.4173749
FULL 0.0000000
RET 0.2409821E-01
Solution: GENPRT
Total variance is minimized at .4174 when we put 15.5% in asset 1, 25% in asset 2, and 59.5% in asset
3.
Developing More Advanced Models 445
The Model
! Scenario portfolio model;
SETS:
SCENE/1..12/: PRB, R, DVU, DVL;
STOCKS/ ATT, GMT, USX/: X;
SXI(SCENE, STOCKS): VE;
ENDSETS
DATA:
TARGET = 1.15;
! Data based on original Markowitz example;
VE =
1.300 1.225 1.149
1.103 1.290 1.260
1.216 1.216 1.419
0.954 0.728 0.922
0.929 1.144 1.169
1.056 1.107 0.965
1.038 1.321 1.133
1.089 1.305 1.732
1.090 1.195 1.021
1.083 1.390 1.131
446 CHAPTER 12
The Sets
We define two primitive setsSCENE and STOCKS. The SCENE set corresponds to the set of 12
scenarios, while STOCKS is the set of three candidate stocks. We form a single dense derived set,
STXSC, which is the cross of these two sets. We need the STXSC set in order to establish the table of
returns for each stock under each scenario.
The Attributes
We define four attributes on the scenarios setPRB, R, DVU, and DVL. PRB stores the probabilities
of the scenarios. R is the expected rate of return under each scenario for a given allocation amongst the
stocks. DVU is the deviation above average return for each scenario for a given stock allocation.
Finally, DVL is the deviation below average return for each scenario for a given stock allocation.
The X attribute is defined on the stocks set. X denotes the fraction of the portfolio allocated to each
stock. The members of X must be determined and constitute the decision variables of the model.
Finally, on the SXI set, we define the VE attribute, which stores the table containing the returns of each
stock under each scenario.
Developing More Advanced Models 447
The Objective
Once again, the objective in this model is to minimize the portfolio’s risk. The default version of the
objective minimizes variance as follows:
! Set objective to VAR, SEMIVAR, or DNRISK;
[OBJ] MIN = VAR;
To solve using the other two measures of risk, simply change the name of the variable from VAR to
either SEMIVAR or DNRISK.
The Formulas
There are six categories of formulas in this model:
1. computing the expected rate of return (AVG),
2. constraining the expected rate of return to exceed our target,
3. computing the expected rate of return under each scenario (R),
4. computing deviations from average return for each scenario (DVU and
DVL),
5. constraining the portfolio to be fully invested, and
6. computing the three measures of risk (VAR, SEMIVAR, and DNRISK).
The following expression computes the expected rate of return for the entire portfolio:
! Compute expected value of ending position;
AVG = @SUM(SCENE: PRB * R);
We do this by taking the sum of the expected returns for each scenario (R) weighted by the
probabilities of the scenarios (PRB).
We constrain expected return to be greater-than-or-equal-to our target using the constraint:
! Target ending value;
AVG >= TARGET;
The expected return under scenario S is computed using:
R(S) = @SUM(STOCKS(J): VE(S, J) * X(J));
This is just the sum of the return of each stock under the scenario weighted by the fraction of the
portfolio in each stock.
The deviations from average return for the scenarios are computed with:
DVU(S) - DVL(S) = R(S) - AVG
If the expected return from the scenario is greater than the average, DVU will be positive. If the
expected return is less than average, DVL will be positive. Given the fact that DVL and DVU impact
the objective, there will never be an optimal solution where both DVL and DVU are greater than 0. If
both are greater than 0, then a better solution can always be obtained by driving one of the two to zero.
By setting things up this way, we have partitioned the deviations into two parts. DVU represents
deviations above the average, which only the variance measure is concerned with. On the other hand,
DVL represents deviations below average, which both the downside risk and semi-variance measures
are computed from.
448 CHAPTER 12
The following constraint forces us to be fully invested and should be familiar from the discussion of
the Markowitz model:
! Budget;
@SUM(STOCKS: X) = 1;
We compute our three measures of risk using:
! Our three measures of risk;
[VARI] VAR = @SUM(SCENE: PRB * (DVU + DVL)^2);
[SEMI] SEMIVAR = @SUM(SCENE: PRB * (DVL) ^2);
[DOWN] DNRISK = @SUM(SCENE: PRB * DVL);
These are simply the sum of a measure of deviation across the scenarios weighted by the probabilities
of the scenarios. The variance measure takes the square of both the above and below average
deviations. The semi-variance measure squares just the below average deviation. While, the downside
risk measure uses the below average deviations without squaring.
The Solution
If we solve the model using the three risk measures in the objective, we get the following three
solutions:
Variance Semi-variance Downside Risk
ATT .530 .575 .511
GMT .357 .039 .489
USX .113 .386 .000
The fraction of the portfolio devoted to ATT is fairly consistent. However, the fractions of GMT and
USX can vary significantly when the risk measure changes.
Developing More Advanced Models 449
Option
Value
Stock Price
Exercise
Price
Unfortunately, these bounds are not very tight. In reality, the option value function will resemble the
curve P in the graph. The exact shape of this curve is influenced by three additional factors. These are,
1) the time to expiration, 2) the volatility, or variance, in the price movements of the stock, and 3) the
interest rate. The underlying formula for curve P eluded researchers for many years until Fischer Black
and Myron Scholes derived it in 1973. A Nobel Prize was subsequently awarded for their work in
1997.
450 CHAPTER 12
The Model
! Computing the value of an option using the Black
& Scholes formula (see "The Pricing of Options
and Corporate Liabilities", Journal of Political
Economy, May-June, 1973);
SETS:
! We have 27 weeks of prices P(t), LOGP(t) is log
of prices;
WEEK/1..27/: P, LOGP;
ENDSETS
DATA:
! Weekly prices of National Semiconductor;
P = 26.375, 27.125, 28.875, 29.625, 32.250,
35.000, 36.000, 38.625, 38.250, 40.250,
36.250, 41.500, 38.250, 41.125, 42.250,
41.500, 39.250, 37.500, 37.750, 42.000,
44.000, 49.750, 42.750, 42.000, 38.625,
41.000, 40.750;
! The current share price;
S = 40.75;
! Time until expiration of the option, expressed
in years;
T = .3644;
! The exercise price at expiration;
K = 40;
! The yearly interest rate;
I = .163;
ENDDATA
SETS:
! We will have one less week of differences;
WEEK1(WEEK)| &1 #LT# @SIZE(WEEK): LDIF;
ENDSETS
! Take log of each week's price;
@FOR(WEEK: LOGP = @LOG(P));
! and the differences in the logs;
@FOR(WEEK1(J): LDIF(J) =
LOGP(J + 1) - LOGP(J));
! Compute the mean of the differences;
MEAN = @SUM(WEEK1: LDIF)/ @SIZE(WEEK1);
Developing More Advanced Models 451
! and the variance;
WVAR = @SUM(WEEK1: (LDIF - MEAN)^2)/
(@SIZE(WEEK1) - 1);
! Get the yearly variance and standard deviation;
YVAR = 52 * WVAR;
YSD = YVAR^.5;
! The Black & Scholes option pricing formula;
Z = ((I + YVAR/2) *
T + @LOG(S/ K))/(YSD * T^.5);
! where VALUE is the expected value of the option;
VALUE = S *@PSN(Z) - K *@EXP(- I * T) *
@PSN(Z - YSD *T^.5);
! LDIF may take on negative values;
@FOR(WEEK1: @FREE(LDIF));
! The price quoted in the Wall Street Journal for
this option when there were 133 days left was $6.625;
Model: OPTION
The Formulas
It is beyond the scope of this document to get into the theoretical details behind the Black & Scholes
pricing model. The interested reader should refer to Black and Scholes (1973) for the details. However,
we will briefly discuss some of the mechanics of the key formulas involved in the model.
In our model, we compute the option’s value in two key steps. First, we compute Z as follows:
Z = ((I + YVAR/2) * T + @LOG(S/ K))/(YSD * T^.5);
where,
I = the yearly interest rate,
YVAR = variance of the stock’s price,
T = time until expiration in years,
S = current share price,
K = exercise price, and
YSD = standard deviation on stock’s price.
We then use the value of Z in the following formula to determine the expected value of the option:
VALUE = S *@PSN(Z)-K *@EXP(- I * T) * @PSN(Z-YSD *T^.5);
where,
@PSN(Z) = returns the cumulative standard normal probability, and
@EXP(X) = returns ex.
452 CHAPTER 12
The Solution
Solving the model, LINGO comes up with a value of $6.58 for the call option―not too far from the
quoted price of $6.63.
Developing More Advanced Models 453
There are two bonds currently being offered that you feel are of sufficient quality to guarantee the
future stream of prize payments. These bonds are listed below:
Bond Years to Maturity Price ($M) Coupon ($M)
A 6 .98 .06
B 13 .965 .065
If funds are not invested in bonds, they can be placed into short-term money market funds. You
conservatively estimate that short-term rates will be about 4% over the 15 year time horizon.
How many of each bond should you buy, and how much additional cash should you allocate to money
market funds to minimize your total outlay while still being able to meet all the future prize payments?
454 CHAPTER 12
The Model
! Bond portfolio/cash matching problem. Given cash
needs in a series of future periods, what
collection of bonds should we buy to meet these
needs?;
SETS:
BOND/A B/ :
MATAT, ! Maturity period;
PRICE, ! Price;
CAMNT, ! Coupon;
BUY; ! Amount to buy;
PERIOD/1..15/:
NEED, !Cash needed each period;
SINVEST; !Short term investment each period;
ENDSETS
DATA:
STRTE = .04; !Short term interest rate;
MATAT = 6, 13; !Years to maturity;
PRICE = .980, .965; !Bond purchase prices;
CAMNT = .060, .065; !Bond coupon amounts;
NEED = 10, 11, 12, 14, 15, 17, 19, 20, 22, 24,
26, 29, 31, 33, 36; ! Cash needs;
ENDDATA
!Minimize total investment required to generate
the stream of future cash needs;
MIN = LUMP;
! First period is slightly special;
LUMP = NEED(1) + SINVEST(1) +
@SUM(BOND: PRICE * BUY);
! For subsequent periods;
@FOR(PERIOD(I)| I #GT# 1:
@SUM(BOND(J)| MATAT(J) #GE# I:
CAMNT(J) * BUY(J)) +
@SUM(BOND(J)| MATAT(J) #EQ# I:
BUY(J)) +
(1 + STRTE) * SINVEST(I - 1) =
NEED(I) + SINVEST(I);
);
! Can only buy integer bonds;
@FOR(BOND(J): @GIN(BUY(J)));
Model: PBOND
The Sets
We define two primitive setsBOND and PERIOD. The BOND set corresponds to the two bonds,
while PERIOD is the set of 15 years in the time horizon.
The Attributes
We define four attributes on the BONDS setMATAT, PRICE, CAMNT, and BUY. MATAT stores the
bonds’ maturities, PRICE the price, CAMNT the coupon amount, and BUY the number of each bond to
buy.
Developing More Advanced Models 455
The NEED and SINVEST attributes are defined on the PERIOD set. NEED stores the cash needs in
each period and SINVEST stores the amount in short-term investments in each period.
The Objective
The objective of this model is to minimize the initial cash outlay. The initial cash outlay, or lump sum,
is stored in the LUMP variable. Thus, our objective is written simply as:
! Minimize the total investment required to generate
the stream of future cash needs;
MIN = LUMP;
The Formulas
There are three categories of formulas in this model:
1. computing the initial lump sum payment (LUMP),
2. sources=uses constraints, which enforce the condition that all sources of
cash in a period must equal uses for cash in a period, and
3. integer restrictions on the BUY variable limiting us to buying only whole
numbers of bonds.
The following expression computes the lump outflow of cash in the initial period:
LUMP = NEED(1) + SINVEST(1) +
@SUM(BOND: PRICE * BUY);
Cash is needed for three purposes in the initial period:
1. payment of lottery prizes (NEED(1)),
2. allocations to short-term money funds (SINVEST(1)), and
3. bond purchases (@SUM(BOND: PRICE * BUY)).
In the remaining 14 periods, sources of cash must equal uses of cash. We enforce this condition with:
! For subsequent periods;
@FOR(PERIOD(I)| I #GT# 1:
@SUM(BOND(J)| MATAT(J) #GE# I:
CAMNT(J) * BUY(J)) +
@SUM(BOND(J)| MATAT(J) #EQ# I:
BUY(J)) +
(1 + STRTE) * SINVEST(I - 1) =
NEED(I) + SINVEST(I);
);
The sources of cash in a period are threefold:
1. coupon receipts:
@SUM(BOND(J)| MATAT(J) #GE# I:
CAMNT(J) * BUY(J))
2. maturing bonds:
@SUM(BOND(J)| MATAT(J) #EQ# I:
BUY(J))
456 CHAPTER 12
The Solution
If we solve the model, we get the following values:
Global optimal solution found.
Objective value: 195.7265
Extended solver steps: 4
Total solver iterations: 27
Variable Value
LUMP 195.7265
BUY(A) 96.00000
BUY(B) 90.00000
SINVEST(1) 4.796526
SINVEST(2) 5.598387
SINVEST(3) 5.432322
SINVEST(4) 3.259615
SINVEST(5) 0.000000
SINVEST(6) 90.61000
SINVEST(7) 81.08440
SINVEST(8) 70.17778
SINVEST(9) 56.83489
SINVEST(10) 40.95828
SINVEST(11) 22.44661
SINVEST(12) 0.1944784
SINVEST(13) 65.05226
SINVEST(14) 34.65435
SINVEST(15) 0.4052172E-01
Solution: PBOND
You have been able to cover a total future debt of $319 million with a lump payment of $195.73
million. To do this, you buy 96 A bonds, 90 B bonds, put $4.8 million into short-term investment
funds, and hold $10 million in cash to pay prizes in the initial period.
Developing More Advanced Models 457
Queuing Models
Erlang Queuing Models Model: EZQUEUE
Background
The telephone, communications, and computer industries have long used queuing models to estimate
the performance of a service system in the face of random demand. The two most frequently used
models are the Erlang loss and Erlang waiting models. In both cases, customers arrive randomly at a
number of identical servers. In the Erlang loss model, there is no queue, so any customer finding all
servers busy is lost. In the Erlang waiting model, there is an infinite queue space, so any customer
finding all servers busy waits until a server is free. In either case, the major measure of performance is
the fraction of customers that find all servers busy.
To compute a system’s performance, we must know the load placed on the system per unit of time and
the number of servers. The load is a unitless measure of the amount of work arriving per unit of time.
For example, if 20 customers arrive each hour and each requires ½ hour of work, then the arriving load
is 10 (20 customers per hour multiplied by ½ hour per customer).
The most crucial probabilistic assumption in both cases is the number of arrivals per unit of time is
Poisson distributed with a constant mean. The held case (with a queue) further requires that service
times be exponentially distributed. If the arriving load is denoted AL and the number of servers by NS,
then the expected fraction finding all servers busy is given in the loss case by @PEL(AL, NS) and in
the held case by @PEB(AL, NS).
The Model
! Arrival rate of customers/ hour;
AR = 25;
! Service time per customer in minutes;
STM = 6;
! Service time per customer in hours;
STH = STM/ 60;
! Fraction customers finding all servers busy;
FB = .05;
! The PEL function finds number of servers
needed, NS;
FB = @PEL(AR * STH, NS);
Model: EZQUEUE
458 CHAPTER 12
The Solution
Feasible solution found at step: 0
Variable Value
AR 25.00000
STM 6.000000
STH 0.1000000
FB 0.5000000E-01
NS 5.475485
Because we cannot have fractional servers, we need at least six servers to meet our requirement that no
more than 5% of customers find all servers busy.
Suppose you install a sufficient number of incoming lines, so customers finding all servers busy can
wait. Further, you will still use six servers. You want to find the following:
♦ the fraction of customers finding all servers busy,
♦ the average waiting time for customers who wait,
♦ the average overall waiting time, and
♦ the average number waiting.
The following variation on the previous model computes these four statistics:
! Arrival rate of customers/ hour;
AR = 25;
! Service time per customer in minutes;
STM = 6;
! Service time per customer in hours;
STH = STM/ 60;
! The number of servers;
NS = 6;
! The PEL function finds number of servers
needed, NS;
FB = @PEB(AR * STH, NS);
! The conditional wait time for those who wait;
WAITC = 1 / (NS / STH - AR);
! The unconditional wait time;
WAITU = FB * WAITC;
! The average number waiting;
NWAIT = AR * WAITU;
Developing More Advanced Models 459
Note how we now use the @PEB function, rather than @PEL, to account for the presence of a queue
to hold callers finding all lines busy. The solution to the modified model is:
Variable Value
AR 25.00000
STM 6.000000
STH 0.1000000
NS 6.000000
FB 0.4744481E-01
WAITC 0.2857143E-01
WAITU 0.1355566E-02
NWAIT 0.3388915E-01
Remember the unit of time is an hour, so the expected waiting time for those who wait is
.2857 * 60 = 1.7 minutes.
460 CHAPTER 12
The Solution
Variable Value
MTBF 40.00000
MTTR 2.000000
NUSER 32.00000
NREPR 1.000000
NDOWN 12.06761
FR 0.4983098
MTD 24.21707
Solution: EZMREPAR
This would probably be considered a heavily loaded systemon average, about 12 users are waiting
for a response from the server. Each request for processing requires an expected elapsed time of over
24 seconds, even though the average request is for only two seconds of processing. As a group, users
request service at the rate of almost one request every 2 seconds.
462 CHAPTER 12
The Model
! Model of a queue with arrivals in batches. In
this particular example, arrivals may show up in
batches of 1, 2, 3, or 4 units;
SETS:
! Look at enough states so P(i) for large i is
effectively zero, where P(i) is the steady state
probability of i-1 customers in the system;
STATE/ 1..41/: P;
! Potential batch sizes are 1, 2, 3 or 4 ,
customers and A(i) = the probability that an
arriving batch contains i customers;
BSIZE/ 1..4/: A;
ENDSETS
DATA:
! Batch size distribution;
A = .1, .2, .3, .4;
! Number of batches arriving per day;
LMDA = 1.5;
! Number of servers;
S = 5;
! Number of customers a server can
process per day;
MU = 2;
ENDDATA
Developing More Advanced Models 463
! LAST = number of STATES;
LAST = @SIZE(STATE);
! Balance equations for states where the number of
customers in the system is less than or equal to
the number of servers;
@FOR(STATE(N)| N #LE# S:
P(N) * ((N - 1)* MU + LMDA) =
P(N + 1) * MU * N +
LMDA * @SUM(BSIZE(I)| I #LT# N: A(I)
* P(N - I))
);
! Balance equations for states where number in
system is greater than the number of servers,
but less than the limit;
@FOR(STATE(N)| N #GT# S #AND# N #LT# LAST:
P(N) * (S * MU + LMDA) =
P(N + 1) * MU * S +
LMDA * @SUM(BSIZE(I)| I #LT# N: A(I) *
P(N - I))
);
! Probabilities must sum to 1;
@SUM(STATE: P) = 1;
Model: QUEUEM
The Formulas
The model computes the probabilities P(i), i = 1 to 41, where P(i) is the probability there are i - 1
motors in the system for repair. We have chosen to stop at 41 because the probability the system would
have more than 40 machines waiting for repair is effectively 0.
In order to solve for the 41 unknown P(i), we will need 41 equations. One of these equations comes
from the fact that the probabilities must all sum to 1:
! Probabilities must sum to 1;
@SUM(STATE: P) = 1;
The remaining 40 equations are derived from the steady state assumptions that the rate of movement
into any state must equal the rate out. We partition these balance equations into two types. In the first
case, there are the states where the number of motors in the system is less-than-or-equal-to the number
of servers. The balance equations in this case are:
@FOR(STATE(N)| N #LE# S:
P(N) * ((N - 1)* MU + LMDA) =
P(N + 1) * MU * N +
LMDA * @SUM(BSIZE(I)| I #LT# N: A(I)
* P(N - I))
);
For each state where the number of motors in the system is less-than-or-equal-to the number of
servers, we set the rate of flow out of the state:
P(N) * ((N - 1)* MU + LMDA)
464 CHAPTER 12
The Solution
An excerpt from the full solution report showing the first 10 values of P is listed below:
Variable Value
LMDA 1.500000
S 5.000000
MU 2.000000
LAST 41.00000
P(1) 0.2450015
P(2) 0.1837511
P(3) 0.1515947
P(4) 0.1221179
P(5) 0.9097116E-01
P(6) 0.5707410E-01
P(7) 0.4276028E-01
P(8) 0.3099809E-01
P(9) 0.2187340E-01
P(10) 0.1538003E-01
The probability at least one workstation is free is the probability that four or fewer motors are in the
system, in other words, the sum of the first five P(i). In this case, it works out that at least one
workstation will be free only about 70% of the time. Experimenting with the model by increasing the
number of servers reveals you will need at least seven workstations to provide the level of service
required by the new priority plan.
Developing More Advanced Models 465
Marketing Models
Markov Chain Model Model: MARKOV
Background
A standard approach used in modeling random variables over time is the Markov chain approach.
Refer to an operations research or probability text for complete details. The basic idea is to think of the
system as being in one of a discrete number of states at each point in time. The behavior of the system
is described by a transition probability matrix that gives the probability the system will move to a
specified other state from some given state. Some example situations are:
System States Cause of Transition
Consumer brand Brand of product most Consumer changes mind,
switching recently purchased by advertising
consumer
Inventory System Amount of inventory on Orders for new material,
hand demands
An interesting problem is to determine the long-term, steady state probabilities of the system. If we
assume the system can reach equilibrium, then it must be true the probability of leaving a particular
state must equal the probability of arriving in that state. You will recall this is the Rate In = Rate Out
Principle (RIRO) we used above in building the multi-server queuing model, QUEUEM. If we let:
The Model
! Markov chain model;
SETS:
! There are four states in our model and over
time the model will arrive at a steady state
equilibrium.
SPROB(J) = steady state probability;
STATE/ A B C D/: SPROB;
! For each state, there's a probability of moving
to each other state. TPROB(I, J) = transition
probability;
SXS(STATE, STATE): TPROB;
ENDSETS
DATA:
! The transition probabilities. These are proba-
bilities of moving from one state to the next
in each time period. Our model has four states,
for each time period there's a probability of
moving to each of the four states. The sum of
probabilities across each of the rows is 1,
since the system either moves to a new state or
remains in the current one.;
TPROB = .75 .1 .05 .1
.4 .2 .1 .3
.1 .2 .4 .3
.2 .2 .3 .3;
ENDDATA
Developing More Advanced Models 467
! Steady state equations;
! Only need N equations, so drop last;
@FOR(STATE(J)| J #LT# @SIZE(STATE):
SPROB(J) = @SUM(SXS(I, J): SPROB(I) *
TPROB(I, J))
);
! The steady state probabilities must sum to 1;
@SUM(STATE: SPROB) = 1;
! Check the input data, warn the user if the sum
of probabilities in a row does not equal 1.;
@FOR(STATE(I):
@WARN('Probabilities in a row must sum to 1.',
@ABS(1 - @SUM(SXS(I, K): TPROB(I, K)))
#GT# .000001);
);
Model: MARKOV
The Sets
The primitive STATE set represents the four states of purchasing detergents A, B, C, and D. We build
one derived set, SXS that is the cross of the STATE set on itself. The SXS set is used to establish the
state transition matrix.
The Attributes
We have two attributes. The first, SPROB, is defined on the STATES set and is used to store the steady
state probabilities of the system. We will be solving for the values of the SPROB attribute. The second
attribute, TPROB, is defined on the two-dimensional SXS set and is used to store the values of the state
transition matrix.
The Formulas
First off, to ensure data integrity, we use the @WARN function to verify the probabilities in each row
of the state transition matrix sum to 1 using:
! Check the input data, warn the user if the sum of
probabilities in a row does not equal 1.;
@FOR(STATE(I):
@WARN('Probabilities in a row must sum to 1.',
@ABS(1 - @SUM(SXS(I, K): TPROB(I, K)))
#GT# .000001);
Due to the potential for roundoff error, we allow for a modest transgression by using a tolerance factor
of .000001. If the probabilities in a row sum up to more than 1.000001 or less than .999999, the user
will receive a warning.
Next, the steady state probabilities must be exhaustive. We guarantee this by setting their sum to 1
with:
! The steady state probabilities must sum to 1;
@SUM(STATE: SPROB) = 1;
468 CHAPTER 12
Finally, in addition to this last equation, we need an additional n-1 equations to solve for the n steady
state probabilities. We get these equations from the RIRO derivation above using:
! Steady state equations;
! Only need N equations, so drop last;
@FOR(STATE(J)| J #LT# @SIZE(STATE):
SPROB(J) = @SUM(SXS(I, J): SPROB(I) *
TPROB(I, J))
);
The Solution
The solution is:
Variable Value
SPROB(A) 0.4750000
SPROB(B) 0.1525000
SPROB(C) 0.1675000
SPROB(D) 0.2050000
Solution: MARKOV
So, we see the long run share of our new brand A will only amount to about 47.5% of the market,
which is considerably less than the conjectured share of 75%.
Developing More Advanced Models 469
The Model
! Conjoint analysis model to decide how much
weight to give to the two product attributes of
warranty length and price;
SETS:
! The three possible warranty lengths;
WARRANTY /LONG, MEDIUM, SHORT/ : WWT;
! where WWT(i) = utility assigned to warranty i;
! The three possible price levels (high,
medium, low);
PRICE /HIGH, MEDIUM, LOW/ : PWT;
! where PWT(j) = utility assigned to price j;
! We have a customer preference ranking for each
combination;
WP(WARRANTY, PRICE) : RANK;
ENDSETS
DATA:
! Here is the customer preference rankings running
from a least preferred score of 1 to the most
preferred of 9. Note that long warranty and low
price are most preferred with a score of 9,
while short warranty and high price are least
preferred with a score of 1;
470 CHAPTER 12
RANK = 7 8 9
3 4 6
1 2 5;
ENDDATA
SETS:
! The next set generates all unique pairs of
product configurations such that the second
configuration is preferred to the first;
WPWP(WP, WP) | RANK(&1, &2) #LT#
RANK(&3, &4): ERROR;
! The attribute ERROR computes the error of our
estimated preference from the preferences given
us by the customer;
ENDSETS
! For every pair of rankings, compute the amount
by which our computed ranking violates the true
ranking. Our computed ranking for the (i,j)
combination is given by the sum WWT(i) + PWT(j).
(NOTE: This makes the bold assumption that
utilities are additive!);
@FOR(WPWP(i, j, k, l): ERROR(i, j, k, l) >=
1 + (WWT(i) + PWT(j)) - (WWT(k) + PWT(l))
);
! The 1 is required on the right-hand-side of the
above equation to force ERROR to be nonzero in
the case where our weighting scheme incorrectly
predicts that the combination (i,j) is equally
preferred to the (k,l) combination.
Since variables in LINGO have a default lower
bound of 0, ERROR will be driven to zero when we
correctly predict (k,l) is preferred to (i,j).
Next, we minimize the sum of all errors in order
to make our computed utilities as accurate as possible;
MIN = @SUM(WPWP: ERROR);
Model: CONJNT
The Sets
We have two primitive sets in the model: WARRANTY is the set of warranty lengths and PRICE is the
set of prices. We form the derived set WP by taking the cross of WARRANTY and PRICE in order to
create an attribute to store the preferences for each (WARRANTY, PRICE) pair.
The interesting set in this model is the sparse derived set WPWP:
! The next set generates all unique pairs of product
configurations such that the second configuration
is preferred to the first;
WPWP(WP, WP) | RANK(&1, &2) #LT#
RANK(&3, &4): ERROR;
Developing More Advanced Models 471
This set is derived from the cross of WP on itself. Each member of this set contains two product
configurations. Furthermore, we use a membership condition to limit the set to combinations of
product configurations where the first configuration is preferred to the second configuration. We need
this set because our model will be testing its proposed utility functions against all unique pairings of
configurations.
Note that this set grows on the order of n2, where n is the number of product configurations. Thus, it
will tend to get big for large numbers of product configurations.
The Attributes
The model defines four attributes: WWT, PWT, RANK and ERROR. WWT and PWT are used to store
the utilities of the warranty and price configurations, respectively. The model will solve for values of
WWT and PWT that minimize total prediction error. These values will then give us the implied utility
functions for the two product attributes. RANK is used to store the customer preference rankings.
Finally, ERROR stores the error in predicting the preference of one configuration over another given
the implied utility values contained in WWT and PWT.
The Objective
The objective is quite simplewe want to minimize the total prediction error over all the unique
product configurations:
MIN = @SUM(WPWP: ERROR);
The Constraints
The model has just one class of constraints that computes the error term in predicting the preferences
over each product configuration:
@FOR(WPWP(i, j, k, l): ERROR(i, j, k, l) >=
1 + (WWT(i) + PWT(j)) -
(WWT(k) + PWT(l))
);
We need to add a 1 to the right-hand side of the constraint to account for the case where we predict
that configuration (i,j) is equally preferred to (k,l). Because of the way that we defined the WPWP set,
(i,j) will always be preferred to (k,l). Thus, it would be an error to predict they are equally preferred.
Note, also, because the lower bound on ERROR is 0 and we are also minimizing ERROR,
ERROR(i,j,k,l) will be driven to 0 when the model correctly predicts that (i,j) is preferred to (k,l).
472 CHAPTER 12
The Solution
Portions of the solution are reproduced below:
Global optimal solution found at step: 35
Objective value: 0.000000
Variable Value
WWT(LONG) 7.000000
WWT(MEDIUM) 2.000000
WWT(SHORT) 0.000000
PWT(HIGH) 0.000000
PWT(MEDIUM) 1.000000
PWT(LOW) 4.000000
Solution: CONJNT
Short, medium and long warranties rate utilities of 0, 2, and 7, while high, medium and low prices rate
utilities of 0, 1, and 4. Note that, given the objective value of zero, this utility weighting scheme
exactly predicts preferences for each pair of product configurations.
13 Programming LINGO
Up to this point, we have primarily been concerned with self-contained models, which are solved once
and don’t receive input from, nor pass output to, other models. However, in many modeling situations,
there may be requirements for one, or more, models to be run in a sequence. Furthermore, the outputs
of one model may be required as inputs to a subsequent model. Such a series of models will typically
be run in a loop until some termination criterion is met. LINGO provides a number of programming
control structures to automate processing in situations where one has multiple, dependent models that
one must run in a sequence.
In the section immediately following, we will provide a brief introduction to the programming features
available in LINGO. We will then illustrate the use of these features in several models.
Programming Features
The programming capabilities in LINGO fall into five categories:
♦ Model control
♦ Flow of control
♦ Model generation
♦ Output statements
♦ Setting parameters
Model Control
In this category we have two types of statements. First are the SUBMODEL and ENDSUBMODEL
statements used to identify submodels, which are separate models contained within larger models. We
also have the @SOLVE statement for use in solving submodels.
473
474 CHAPTER 13
In this example, the submodel is titled Pattern_Gen and contains one objective, R_OBJ, one constraint,
R_WIDTH, and one @FOR loop to force the Y variables to be general integers via the @GIN function.
Note: Submodels must have been defined in the model prior to any references to them via the
@SOLVE statement.
Flow of Control
In a calc section, model statements are normally executed sequentially. Flow of control statements can
be used to alter the execution order of statements. In effect, this gives you a programming, or
scripting, capability within the LINGO environment.
Note: Be aware of the use of the letter ‘C’ in @IFC. This is to distinguish the flow of control if
statement (@IFC) from the arithmetic if (@IF).
Programming LINGO 475
To illustrate, the following sample code uses if/then/else blocks to as part of a binary search for a key
value in an array:
@IFC( KEY #EQ# X( INEW):
LOC = INEW;
@ELSE
@IF( KEY #LT# X( INEW):
IE = INEW;
@ELSE
IB = INEW;
);
);
@FOR
You’ve encountered the @FOR set looping statement previously as a way to generate constraints in
the model section. @FOR is also allowed in the calc section to perform looping. The main difference
is that @FOR does not generate constraints when used in the calc section. Instead, it immediately
executes any assignment statements contained within its scope. The following example shows a
@FOR loop extracted from a portfolio model. The loop is used to solve the portfolio model for a
number of different levels of desired return. For each level of return, the model minimizes variance,
and the variance is stored for later use.
@FOR( POINTS( I):
! Compute new return level;
RET_LIM = RET_MIN + (I-1)*INTERVAL;
! Re-solve the model;
@SOLVE();
! Store the return value;
YRET( I) = RET_LIM;
! Store the variance too;
XSTD( I) = VARIANCE^0.5;
);
@WHILE
The @WHILE statement is used for looping over a group of statements until some termination
criterion is met. The syntax is as follows:
@WHILE( <conditional-exp>: statement_1[; …; statement_n;]);
As long as the conditional expression is true, the @WHILE function will keep looping over its block of
statements.
476 CHAPTER 13
As an example, the following code uses an @WHILE loop to search for a key value in an array as part
of a binary search procedure:
@BREAK
The @BREAK statement is used to break out of the current loop. Execution resumes at the first
statement immediately following the end of the current loop. The @BREAK statement is valid only
within @FOR and @WHILE loops in calc sections and does not take any arguments. As an example,
we extend the @WHILE loop in the binary search example above to include an @BREAK statement
that will be executed when the key value can’t be found:
If a text argument is included in the @STOP, then it will also be displayed as part of this error
message.
As an example, we extend the @WHILE loop in the binary search example above to include an
@STOP statement that will be executed when the key value can’t be found:
@WHILE( KEY #NE# X( LOC):
!exit if key can’t be found;
@IFC( IE –IB #LE# 1:
@STOP( 'Unable to find key!!!');
);
INEW = @FLOOR( ( IE + IB) / 2);
@IFC ( KEY #EQ# X( INEW):
LOC = INEW;
@ELSE
@IF ( KEY #LT# X( INEW):
IE = INEW;
@ELSE
IB = INEW;
);
);
);
Model Generation
The commands in this category are related to the model generator, i.e., the component in LINGO that
translates your model’s statements into a format compatible with the solver.
The @GEN statement accepts an optional argument of one or more submodel names. If a submodel is
specified, LINGO will only generate the model contained within the specified submodel section. If
multiple submodel names are specified, LINGO will combine them all into a single, larger model. If
submodel names are omitted entirely, then LINGO will generate only those model statements
occurring before the @GEN statement and not contained in any submodels.
As an example, below is a small staffing model that uses the @GEN statement in a calc section:
MODEL:
SETS:
DAY / MON, TUE, WED, THU,
FRI, SAT, SUN/ :
NEED, START, COST;
ENDSETS
DATA:
NEED = 18 15 12 16 19 14 12;
COST = 400;
ENDDATA
CALC:
@GEN();
ENDCALC
END
Programming LINGO 479
Running this sample model yields the following generated model report:
MODEL:
[OBJ] MIN= 400 * START_MON + 400 * START_TUE +
400 * START_WED + 400 * START_THU + 400 * START_FRI +
400 * START_SAT + 400 * START_SUN ;
[CONS_MON] START_MON + START_THU + START_FRI +
START_SAT + START_SUN >= 18 ;
[CONS_TUE] START_MON + START_TUE + START_FRI +
START_SAT + START_SUN >= 15 ;
[CONS_WED] START_MON + START_TUE + START_WED +
START_SAT + START_SUN >= 12 ;
[CONS_THU] START_MON + START_TUE + START_WED +
START_THU + START_SUN >= 16 ;
[CONS_FRI] START_MON + START_TUE + START_WED +
START_THU + START_FRI >= 19 ;
[CONS_SAT] START_TUE + START_WED + START_THU +
START_FRI + START_SAT >= 14 ;
[CONS_SUN] START_WED + START_THU + START_FRI +
START_SAT + START_SUN >= 12 ;
END
Running the modified model will generate the following matrix picture:
S S S S S S S
T T T T T T T
A A A A A A A
R R R R R R R
T T T T T T T
( ( ( ( ( ( (
M T W T F S S
O U E H R A U
N E D U I T N
) ) ) ) ) ) )
OBJ: C C C C C C C MIN
CONS(MON): 1 1 1 1 1 > B
CONS(TUE): 1 1' ' 1'1 1 > B
CONS(WED): 1 1 1 ' 1 1 > B
CONS(THU): 1 1 1 1 1 > B
CONS(FRI): 1 1'1 1 1' ' > B
CONS(SAT): ' 1 1 1 1 1 ' > B
CONS(SUN): ' 1 1 1 1 1 > B
Refer to the PICTURE command for a discussion on how to interpret matrix picture reports.
@RELEASE ( VARIABLE_NAME)
When a variable is assigned a value in a calc section, that variable is marked by LINGO as being
permanently fixed at the value, i.e., subsequent optimization via @SOLVE will not affect the variable’s
value. The @RELEASE statement is used to release such a fixed variable so that it may once again
become optimizable.
The following calc section, developed for the staff scheduling model presented above, will help to
illustrate the use of @RELEASE:
CALC:
@SOLVE();
@FOR( DAY( D):
! Force starts to 0 for today;
START( D) = 0;
@SOLVE();
OBJ0( D) = OBJ;
! Allow START( D) to be optimizable again;
@RELEASE( START( D));
);
@FOR( DAY( D):
@WRITE( DAY( D), 10*' ', OBJ0( D), @NEWLINE( 1));
);
ENDCALC
This calc section loops through each day of the week and does the following:
2. Solves for the low-cost staffing pattern given that the current day will have no starting
employees: @SOLVE()
4. Releases the number of employee starts for the current day so that it can be optimized in
the next solve: RELEASE( DAY( D))
482 CHAPTER 13
The last part of the calc section writes the following report summarizing the results:
MON 9800
TUE 9000
WED 8600
THU 9600
FRI 9200
SAT 8800
SUN 8600
Output Statements
The features in this section are for displaying output from within the code in calc sections. Unlike the
data section, calc sections can send output only to the screen and text files. If you need to send output
to Excel or to databases, then you will need to do so from a data section.
Dual analysis:
Chassis1: 50
Chassis2: 0
Drives: 150
Note that in addition to the @WRITE statement, we also made use of both the @NEWLINE report
function to produce line feeds and the @DUAL report function to return the dual values, or shadow
prices, on the constraints. Additional information on these and other report functions may be found in
section Report Functions of Chapter 7.
484 CHAPTER 13
The user has the option of pressing either the Resume button to continue normal processing, or the
Interrupt button to immediately terminate execution of the model. On platforms other than Windows,
the output will be routed to the terminal and LINGO will wait for the user to press the Enter key.
Programming LINGO 485
@DIVERTError! Bookmark not defined.( [‘FILE_NAME’])
By default, output generated by the @WRITE statement will be sent to the screen. However, you may
wish to capture this output in a file. @DIVERT allows you to do this. As an example, we modified the
product mix example from above to route its custom report to file MYREPORT.TXT as follows:
CALC:
@SOLVE();
@DIVERT( 'MYREPORT.TXT');
@WRITE( 'Total profit = ', PROFIT, @NEWLINE( 1));
@WRITE( 'Production:', @NEWLINE( 1));
@WRITE( ' WS = ', WS, @NEWLINE( 1),
' NC = ', NC, @NEWLINE( 1),
'Total units = ', WS + NC,
@NEWLINE( 2)
);
@WRITE( 'Dual analysis:', @NEWLINE( 1),
' Chassis1: ', @DUAL( CHASSIS1),
@NEWLINE( 1),
' Chassis2: ', @DUAL( CHASSIS2),
@NEWLINE( 1),
' Drives: ', @DUAL( DRIVES),
@NEWLINE( 1)
);
@DIVERT();
ENDCALC
Note the two occurrences of @DIVERT. In the first instance, we specify the file we want to open, and
subsequent @WRITE output is diverted to that file. The second reference to @DIVERT closes the file
and reverts output back to the terminal device.
@DIVERT also accepts an optional second argument of the letter ‘A’. If this argument is present,
LINGO will append output to the end of the file if it already exists. If the argument is omitted, LINGO
will overwrite the file if it already exists
Note: @DIVERT statements can be nested so that multiple levels of output files may be
simultaneously in use.
Setting Parameters
LINGO has many optional settings that can be controlled with the LINGO|Options command. At times, you
may find it necessary to alter these parameters dynamically in your model’s calc sections. For this reason,
LINGO provides the @SET statement, which gives you access to the entire set of system parameters. There
is also and additional function, @APISET, for setting more obscure parameters in the LINDO API
(LINGO’s solver library) that aren’t available through the standard LINGO options set.
486 CHAPTER 13
Note: Parameters whose values have been changed with @APISET will be restored to their original
settings once the model run is completed.
Programming LINGO 487
Note: A binary search is not typically something you would do in LINGO, and we use it here
merely as an example of a simple algorithm for illustrative purposes.
The Model
The following model is an example of implementing a binary search in LINGO:
MODEL:
! Illustrates programming looping
capabilities of LINGO by doing a
binary search;
SETS:
S1: X;
ENDSETS
DATA:
! The key for which we will search;
KEY = 16;
! The list (must be in sorted
increasing order);
X = 2 7 8 11 16 20 22 32;
ENDDATA
! Do a binary search for key;
CALC:
! Set begin and end points of search;
IB = 1;
IE = @SIZE( S1);
! Loop to find key;
@WHILE( IB #LE# IE:
! Cut bracket in half;
LOC = @FLOOR((IB + IE)/2);
@IFC( KEY #EQ# X(LOC):
@BREAK; ! Do no more loops;
@ELSE
@IFC( KEY #LT# X( LOC):
IE = LOC-1;
@ELSE
IB = LOC+1;
);
);
);
@IFC( IB #LE# IE)
! Display key's location;
488 CHAPTER 13
The Details
Our first step is to define a set, S1, and give it the attribute X using the following sets section:
SETS:
S1: X;
ENDSETS
The X attribute will store the list that we will search for the key value. We populate X and input our
key in the data section:
DATA:
! The key for which we will search;
KEY = 16;
! The list (must be in sorted
increasing order);
X = 2 7 8 11 16 20 22 32;
ENDDATA
Note that for the purposes of this example, the list of values must be input in sorted order. An
interesting exercise would be to extend this model so that it tests to determine if the data is sorted. Or,
if you are really ambitious, extend the model so that it will sort a list with arbitrary ordering.
Next comes the calc section, which contains our code to do the binary search. In the first part of the
calc section:
CALC:
! Set begin and end points of search;
IB = 1;
IE = @SIZE( S1);
we bracket the key by pointing to the beginning of the list, IB, and the end of the list, IE. Note that we
make use of the @SIZE function to dynamically compute the size of the list.
Next, we have the @WHILE loop to search for the key value:
! Loop to find key;
@WHILE( IB #LE# IE:
! Cut bracket in half;
LOC = @FLOOR((IB + IE)/2);
@IFC( KEY #EQ# X(LOC))
@BREAK; ! Do no more loops;
@ELSE
@IFC( KEY #LT# X( LOC):
IE = LOC-1;
@ELSE
IB = LOC+1;
Programming LINGO 489
);
);
);
The @WHILE loop tests if the candidate range has become empty at the start of each iteration:
@WHILE( IB #LE# IE:
If not, we continue by dissecting the bracket in half:
!cut bracket in half;
INEW = @FLOOR( ( IE + IB) / 2);
We then determine if the new bracket point resides above or below the key and set IB and IE
accordingly:
@IFC( KEY #EQ# X(LOC):
@BREAK; ! Do no more loops;
@ELSE
@IFC( KEY #LT# X( LOC):
IE = LOC-1;
@ELSE
IB = LOC+1;
);
);
Finally, when we fall out of the @WHILE loop we display the result of our search:
@IFC( IB #LE# IE:
! Display key's location;
@PAUSE( 'Key is at position: ', LOC);
@ELSE
! Key not in list;
@STOP( ' Key not on list!!!');
);
If the eligible range is empty, then we did not find the key. Otherwise, the key was found and report
its location on the list.
490 CHAPTER 13
If you run this model, LINGO should successfully find the key and display:
The Model
Here’s the model we will use to generate points along the efficient frontier:
MODEL:
! Solves the generic Markowitz portfolio
model in a loop to generate the points
on the efficient frontier;
SETS:
ASSET: RATE, UB, X;
Programming LINGO 491
COVMAT( ASSET, ASSET): V;
POINTS: XRET, YVAR;
ENDSETS
DATA:
! Number of points on the
efficient frontier graph;
NPOINTS = 10;
POINTS = 1..NPOINTS;
! The stocks;
ASSET = GOOGLE, YAHOO, CISCO;
! Expected growth rate of each asset;
RATE = 1.3 1.2 1.08;
! Upper bound on investment in each;
UB = .75 .75 .75;
! Covariance matrix;
V = 3 1 -.5
1 2 -.4
-.5 -.4 1;
ENDDATA
! Below are the three objectives we'll use;
SUBMODEL SUB_RET_MAX:
[OBJ_RET_MAX] MAX = RETURN;
ENDSUBMODEL
SUBMODEL SUB_RET_MIN:
[OBJ_RET_MIN] MIN = RETURN;
ENDSUBMODEL
SUBMODEL SUB_MIN_VAR:
[OBJ_MIN_VAR] MIN =
@SUM( COVMAT( I, J): V( I, J) * X( I) * X( J));
ENDSUBMODEL
!and the constraints;
SUBMODEL SUB_CONSTRAINTS:
! Compute return;
RETURN = @SUM( ASSET: RATE * X);
! Must be fully invested;
@SUM( ASSET: X) = 1;
! Upper bounds on each;
@FOR( ASSET: @BND( 0, X, UB));
! Must achieve target return;
RETURN >= RET_LIM;
ENDSUBMODEL
CALC:
! Set some parameters;
! Reset all params;
@SET( 'DEFAULT');
! Minimize output;
@SET( 'TERSEO', 1);
! Suppress status window;
@SET( 'STAWIN', 0);
! Capture unwanted output;
@DIVERT( 'LINGO.LOG');
RET_LIM = 0;
@SOLVE( SUB_RET_MAX, SUB_CONSTRAINTS);
The Details
First consider the sets section:
SETS:
ASSET: RATE, UB, X;
COVMAT( ASSET, ASSET): V;
POINTS: XRET, YVAR;
ENDSETS
It defines three sets: ASSET, COVMAT and POINTS.
Programming LINGO 493
The ASSET set will contain the set of assets available for investment. Each asset has an expected rate
of return (RETURN), an upper bound on the amount of the asset we’ll allow in the portfolio (UB), and
the fraction of the portfolio devoted to the asset (X). Note that the fraction of the portfolio devoted to
each asset, X, constitutes our decision variables.
The COVMAT set is a cross of the ASSET set on itself. We create this set for the attribute V, which
will store the covariance matrix for all the assets.
The POINTS set is used to represent the points on the efficient frontier that we will be generating. For
each point, we will determine its x-coordinate, XRET, and its y-coordinate, YVAR. Note that the x-
coordinate will represent risk, while the y-coordinate will represent return. What we intend to do is to
solve a portfolio model once for each member of the POINTS set to get a new point on the efficient
frontier. These points will be stored in XRET and YVAR. Once the loop is completed, the list of points
in XRET and YVAR will give us a glimpse of what the efficient frontier looks like.
Next we have the data section:
DATA:
! Number of points on the
efficient frontier graph;
NPOINTS = 10;
POINTS = 1..NPOINTS;
! The stocks;
ASSET = GOOGLE, YAHOO, CISCO;
! Expected growth rate of each asset;
RATE = 1.3 1.2 1.08;
! Upper bound on investment in each;
UB = .75 .75 .75;
! Covariance matrix;
V = 3 1 -.5
1 2 -.4
-.5 -.4 1;
ENDDATA
In this data section we are setting the number of points that we will generate along the efficient frontier
(NPOINTS) to 10. Once we have the number of points established, we dynamically create the POINTS
set. Next, we input the set of assets, their expected rate of return, their upper bounds, and their
covariance matrix.
The next section of the model:
! Below are the three objectives we'll use;
SUBMODEL SUB_RET_MAX:
[OBJ_RET_MAX] MAX = RETURN;
ENDSUBMODEL
SUBMODEL SUB_RET_MIN:
[OBJ_RET_MIN] MIN = RETURN;
ENDSUBMODEL
SUBMODEL SUB_MIN_VAR:
[OBJ_MIN_VAR] MIN =
@SUM( COVMAT( I, J): V( I, J) * X( I) * X( J));
ENDSUBMODEL
494 CHAPTER 13
makes use of the SUBMODEL and ENDSUBMODEL statements to establish three different objectives.
The first two objectives respectively maximize and minimize portfolio return, which are used later on
in the model to determine that maximum and minimum possible returns that can be generated with out
basket of available stocks. The third objective, SUB_MIN_VAR, minimizes portfolio risk as measured
by its variance.
Following the three submodels containing our various objectives, we have another submodel that
contains our three constraints:
!and the constraints;
SUBMODEL SUB_CONSTRAINTS:
! Compute return;
RETURN = @SUM( ASSET: RATE * X);
! Must be fully invested;
@SUM( ASSET: X) = 1;
! Upper bounds on each;
@FOR( ASSET: @BND( 0, X, UB));
! Must achieve target return;
RETURN >= RET_LIM;
ENDSUBMODEL
The first constraint of our constraint section computes portfolio return. The second constraint says that
we must invest 100 percent of our capital. The third constraint puts an upper bound on the percentage
of the portfolio invested in each asset. Finally, the fourth constraint forces total portfolio return to
achieve some desired level.
The next section, the calc section, is of primary interest. It contains the logic to solve the model
multiple times to generate and store the points on the efficient frontier. First, we make use of @SET to
set some parameter values:
! Set some parameters;
! Reset all params to their defaults; @SET( 'DEFAULT');
! Minimize output;
@SET( 'TERSEO', 1);
! Suppress status window;
@SET( 'STAWIN', 0);
The first call to @SET restores all parameters to their default values, the second call minimizes the
level of LINGO’s output to improve performance, while the third call suppresses the status window
that would normally pop up when solving a model. We suppress the status window so it does not
obscure the custom report we are creating at the end of the run.
Next, we use @DIVERT to route LINGO’s output to a log file:
! Capture spurious output;
@DIVERT( 'LINGO.LOG');
We do this to capture the output LINGO would normally generate while it’s solving the various
portfolio models. We capture this output so it doesn’t distract from the custom report we will display
at the end of the run.
Programming LINGO 495
Our next task is to find out the possible range of portfolio returns. We want this information so that
we don’t try to run our model for returns lying outside this range, which would result in an infeasible
model. This requires two runs: one where we maximize return and one where we minimize return.
Here is the code that performs the runs:
! Solve to get maximum return;
RET_LIM = 0;
@SOLVE( SUB_RET_MAX, SUB_CONSTRAINTS);
Gilmore and Gomory published a paper in 1961 titled A Linear Programming Approach to the
Cutting-Stock Problem. In this paper they outline a two-stage, iterative approach for solving cutting-
stock problems that dramatically reduced the number of patterns one must generate to get good
solutions. The basic idea involves solving a master problem containing a limited number of patterns.
The dual prices on the finished goods are then passed to a small knapsack subproblem that selects a
new cutting pattern that maximizes the sum of the dual values of all the finished goods contained in the
pattern subject to not exceeding the length of the raw material. This pattern is then appended to the
previous master problem, which is then re-solved. This iterative process continues until no further
beneficial patterns can be generated by the knapsack subproblem. In which case, we have the optimal
solution to the original, linear cutting-stock problem. The remarkable feature of this algorithm is that
it typically takes relatively few passes to reach the optimal solution, thereby making it possible to
solve very large cutting-stock models in an extremely reasonable amount of time. This approach of
iteratively appending new columns to models is also referred to as column generation.
The Model
For our example, we will be cutting 45 foot wide rolls of paper into smaller rolls of widths: 34, 24, 15,
10 and 18. We use Lingo’s programming capability to iteratively solve the master and subproblem
until no further beneficial cutting patterns remain to be appended to the master problem.
MODEL:
! Uses Lingo’s programming capability to do
on-the-fly column generation for a
cutting-stock problem;
SETS:
PATTERN: COST, X;
FG: WIDTH, DEM, PRICE, Y, YIELD;
FXP( FG, PATTERN): NBR;
ENDSETS
DATA:
PATTERN = 1..20; ! Allow up to 20 patterns;
RMWIDTH = 45; ! Raw material width;
FG = F34 F24 F15 F10 F18;!Finished goods...;
WIDTH= 34 24 15 10 18;!their widths...;
DEM = 350 100 800 1001 377;!and demands;
BIGM = 999;
ENDDATA
SUBMODEL MASTER_PROB:
[MSTROBJ] MIN= @SUM( PATTERN( J)| J #LE# NPATS:
COST( J)*X( J));
@FOR( FG( I):
[R_DEM]
@SUM( PATTERN( J)| J #LE# NPATS:
NBR( I, J) * X( J)) >= DEM( I);
);
ENDSUBMODEL
SUBMODEL INTEGER_REQ:
@FOR( PATTERN: @GIN( X));
ENDSUBMODEL
Programming LINGO 499
SUBMODEL PATTERN_GEN:
[SUBOBJ] MAX = @SUM( FG( I): PRICE( I)* Y( I));
@SUM( FG( I): WIDTH( I)*Y( I)) <= RMWIDTH;
@FOR( FG( I): @GIN(Y( I)));
ENDSUBMODEL
CALC:
! Send unwanted output to log file;
@DIVERT( 'LINGO.LOG');
! Set parameters;
@SET( 'DEFAULT');
@SET( 'TERSEO', 1);
@SET( 'STAWIN', 0);
The Details
First, we have the sets section:
SETS:
PATTERN: COST, X;
FG: WIDTH, DEM, PRICE, Y, YIELD;
FXP( FG, PATTERN): NBR;
ENDSETS
500 CHAPTER 13
The PATTERN set is used to represent the cutting patterns we will be generating. Each pattern has a
cost, which, with one exception (discussed below), will be 1, i.e., each pattern uses 1 raw material
piece. We also assigned an attribute called X to the patterns set. X( p) will be used to store the
number of times each pattern p is to be cut and is one of the decision variables.
The FG set is used to represent the set of finished goods. As input, each finished good has a width and
customer demand. The PRICE attribute will be used to store the dual prices on the finished goods.
These prices will be updated each time we solve the master problem. Y will be an integer, decision
variable that we will use in the knapsack subproblem to represent the number of pieces of each
finished good to use in the next generated pattern. YIELD will be used to store the number of pieces of
each finished good that gets produced.
The derived set, FXP, is derived on the finished goods and patterns sets. The attribute NBR( i, j) will
store the number of finished goods I contained in pattern j.
Next, we have the data section
DATA:
PATTERN = 1..20; ! Allow up to 20 patterns;
RMWIDTH = 45; ! Raw material width;
FG = F34 F24 F15 F10 F18;!Finished goods...;
WIDTH= 34 24 15 10 18;!their widths...;
DEM = 350 100 800 1001 377;!and demands;
BIGM = 999;
ENDDATA
We initialize the pattern set to have 20 members. This will only allow for generation of up to 20
patterns; however, this should be more that adequate for this small example.
After inputting the raw material width of 45, we input the set of five finished goods and their widths.
After that, we input the demands for the finished goods. Finally, we input a parameter called BIGM,
the purpose of which is discussed below.
Next, we have our first submodel:
SUBMODEL MASTER_PROB:
[MSTROBJ] MIN= @SUM( PATTERN( J)| J #LE# NPATS:
COST( J)*X( J));
@FOR( FG( I):
[R_DEM]
@SUM( PATTERN( J)| J #LE# NPATS:
NBR( I, J) * X( J)) >= DEM( I);
);
ENDSUBMODEL
This is the master problem we’ve been mentioning. The objective states that we wish to minimize the
total cost of all the patterns used. The constraint says that we must meet, or exceed, demand for all the
finished goods.
The next submodel:
SUBMODEL INTEGER_REQ:
@FOR( PATTERN: @GIN( X));
ENDSUBMODEL
Programming LINGO 501
will be used in conjunction with the master problem to force the variables to take on integer values.
Given that it’s not possible to cut a fractional number of a particular pattern, for our final solution we
need the X variables to be integer valued.
Our final submodel:
SUBMODEL PATTERN_GEN:
[SUBOBJ] MAX = @SUM( FG( I): PRICE( I)* Y( I));
@SUM( FG( I): WIDTH( I)*Y( I)) <= RMWIDTH;
@FOR( FG( I): @GIN(Y( I)));
ENDSUBMODEL
is the pattern-generation subproblem we’ve mentioned. This is a knapsack problem that finds the best
pattern that can be cut given the dual prices on the finished goods. Recall that we get the dual prices
on the finished goods from the demand constraints in the master problem.
We then enter the calc section, which contains the programming logic to coordinate the iterative
algorithm we’ve chosen. As with the previous Markowitz model, the start of the calc section is
devoted to diverting output to a file and setting of parameters. You may refer to the previous
Markowitz model for a discussion of why these steps are being performed.
One of the features of this model most likely to change in the future would be the maximum number of
patterns to generate. It’s probably not wise to assume that this number will always be fixed at 20
patterns. For this reason, we use the @SIZE function to get the current number of allowed patterns:
! Max number of patterns we'll allow;
MXPATS = @SIZE( PATTERN);
Next, we construct what we refer to as a “super pattern”:
! Make first pattern an expensive super pattern;
COST( 1) = BIGM;
@FOR( FG( I): NBR( I, 1) = 1);
The supper pattern is a device to jumpstart our algorithm by guaranteeing that the model will be
feasible from the start. We need the model to always be feasible in order to obtain a valid set of dual
prices to feed into the pattern generation submodel. The super pattern is an artificial pattern that can
create one piece of each finished good. In real life, such a pattern would not be possible because we
can’t physically fit one of each finished good on a single raw material piece. Given that the super
pattern is not physically possible, we assign it a high cost, BIGM, so it will not be used in the final
solution
Recall that NBR( i, j) represents the number of finished good i in pattern j. So, for our super pattern,
we use a for loop over all the finished goods i, setting NBR( i, 1) to 1 for each of the finished goods.
The second index of NBR is set at 1 given that the super pattern is the first pattern generated.
Next, the main loop that will alternate between solving the master and subproblem until an optimal
solution is found:
502 CHAPTER 13
Pattern:
FG Demand Yield 1 2 3 4 5 6 7 8 9
=====================================================
F34 350 350 1 . . . 1 . . . .
F24 100 100 1 . . . . 1 . . 1
F15 800 801 1 . 3 . . . 1 1 .
F10 1001 1002 1 4 . . 1 2 1 3 .
F18 377 377 1 . . 2 . . 1 . 1
=====================================================
Usage: 0 0 133 0 350 0 277 125 100
A total of 985 raw materials were used, with only 2.7% of the raw material input lost to trim waste.
Note that only 9 patterns were needed to solve the model. Also note that Pattern 1, the super pattern, is
not used in the final solution as per our design.
Summary
There are many instances where you will need to solve a model multiple times and/or use the results of
one model as input into another. The programming capabilities in LINGO allow you to automate these
classes of problems. The three examples presented in this chapter introduce the programming
constructs available in LINGO.
In addition to the three examples presented in this chapter, several more examples of solving models in
a loop may be found in the samples folder of your LINGO installation. These additional examples
include:
Loopdea.lg4 – A Data Envelopment Analysis model with a loop over each of the
decision-making units to compute their individual efficiency scores.
Loopts.lg4 – A traveling salesman model that loops to append subtour breaking
constraints to the formulation, continuing until a complete tour of the cities is generated.
Loopttt.lg4 – A looping model that implements a game of tic-tac-toe.
14 On Mathematical
Modeling
When developing a model in LINGO, it helps to understand how the model is processed internally by
the LINGO solver. The relationships in your model influence the computation time, the solution
methods used by LINGO, and the type of answer returned. Here we’ll explain some of the different
types of relationships in a LINGO model and how each type affects the solution search. An
understanding of these topics is not required to use LINGO, but it can help you use the software more
effectively.
505
506 CHAPTER 14
The linear solver in LINGO uses the revised simplex method with product form inverse. A barrier
solver may also be obtained, as an option, for solving linear models. LINGO’s nonlinear solver
employs both successive linear programming (SLP) and generalized reduced gradient (GRG)
algorithms. Integer models are solved using the branch-and-bound method. On linear integer models,
LINGO does considerable preprocessing, adding constraint “cuts” to restrict the noninteger feasible
region. These cuts will greatly improve solution times for most integer programming models.
Type of Constraints
Through the use of the direct solver, LINGO substitutes out all the fixed variables and constraints from
the model. The remaining reduced set of constraints and variables are then classified as being either
linear or nonlinear. LINGO’s solver status window, which by default opens every time you solve a
model, gives a count of the linear and nonlinear variables and constraints in a model. If any nonlinear
variables or constraints are found in the model, the entire model is considered nonlinear and the
relatively slower nonlinear solver must be invoked in place of the linear solver.
Linear Constraints
If all the terms of a constraint are of the first order, the constraint is said to be linear. This means the
constraint doesn’t contain a variable squared, cubed, or raised to any power other than one, a term
divided by a variable, or variables multiplied by each other. Also, proportionality must exist. In other
words, for every unit increase or decrease in a variable, the value of the constraint increases or
decreases by a fixed amount.
Linear formulas are “straight line” relationships. The basic form of a linear formula is:
Y=mX+b
where m and b are constants.
For example, suppose you’re buying tomatoes for $1.50 per pound. The expression or function used to
calculate the cost (C) in terms of the amount of tomatoes purchased (T) is:
C = 1.5 * T.
ON MATHEMATICAL MODELING 507
As you might expect, a graph of this function for cost is a straight line:
$60
$50
$40
Cost $30
$20
$10
$0
0 10 20 30 40
Tomatoes Purchased (lbs.)
Linear expressions can have multiple variables. For example, if you added potatoes (P) at $0.75 per
pound and apples (A) at $1.25 per pound, your cost function would become:
C = 1.5 * T + 0.75 * P + 1.25 * A
This new cost expression is also linear. You could think of it as the sum of three simpler linear
expressions.
Because linear models can be solved much faster and with more accuracy than nonlinear models, it’s
preferable to formulate your models using linear expressions whenever possible. When the
LINGO|Solve command is issued, LINGO analyzes the relationships in the model. If all the
expressions are linear, LINGO will recognize and take advantage of this fact.
Relative to other types of models, problems expressed using exclusively linear relationships can be
solved quickly. If allowed to run to completion, LINGO will return the answer that yields the highest
value for a maximization objective, or the lowest value for a minimization objective.
One way to learn whether or not all expressions in your model are linear is to note the classification
statistics displayed during the solution process in the solver status window. The Nonlinear categories
of the Variables and Constraints boxes display the number of nonlinear relationships in the model. If
zeros appear in both these categories, the model is linear.
If LINGO displays a number greater than zero for the nonlinear relationships, you may want to
investigate whether the constraints and variables in your model could be reformulated in a linear
manner. For example, consider the following constraint:
X / Y = 10;
As written, this constraint is nonlinear because we are dividing by Y. By simply multiplying both sides
of the equation through by Y, we can convert it to the equivalent linear constraint:
X = 10 * Y;
508 CHAPTER 14
Nonlinear Constraints
By definition, all constraints that are not linear are nonlinear. Nonlinear expressions include
relationships with variables that are squared, cubed, taken to powers other than one, or multiplied or
divided by each other.
Models with nonlinear expressions are much more difficult to solve than linear models. Unlike linear
models, nonlinear models may prevent LINGO from finding a solution, though one exists. Or LINGO
may find a solution to a nonlinear model that appears to be the “best”, even though a better one may
exist. These results are obviously undesirable. For more on what you can do to help minimize the
occurrence of these undesirable results, see the Guidelines for Nonlinear Modeling section below.
Convexity
The characteristic of an expression called convexity along with a closely related feature called
concavity are the most useful features of a model for guaranteeing that a local optimum is actually a
global optimum. Let’s consider the convexity property of a minimization problem. The mathematical
definition of a convex model is as follows:
f(y) ≤ a f(x) + (1-a) f(z), where y=a*x + (1-a)*z
In words, a function is convex if, for any two points on the function, a straight line connecting the two
points lies entirely on or above the function.
Determining the convexity of a multiple variable problem is no easy task. Mathematicians call a
function convex if the matrix of the second derivatives is positive definite or has positive Eigen values.
However, if you can identify your LINGO model as being convex for a minimization problem, you can
ensure that any solution you reach is a global optimum (including nonlinear models).
Strictly Convex
A strictly convex function with no constraints has a single global minimum. Therefore, minimizing a
strictly convex function will yield the unique global optimal solution regardless of the initial value of
the variables. The graph below shows a convex function of a single variable:
40000
35000
30000
25000
20000
15000
10000
5000
0
100
-250
150
200
-150
50
250
-100
300
-200
0
-50
-300
Loosely Convex
A loosely convex function with no constraints has multiple local minima, which are also global
minima. Therefore, minimizing a loosely convex function may yield different solutions for different
initial values of the variables. However, if you know the function is loosely convex, you will know the
solution is a global minimum. For example, note the flat section (i.e., slope is zero) from
approximately –200 to 150 in the following function that makes it loosely convex:
40000
35000
30000
25000
20000
15000
10000
5000
0 100
-250
150
200
-150
50
250
-100
300
-200
0
-50
-300
Concavity
While convexity applies to minimization problems, concavity ensures the corresponding attribute of
global optimality in maximization problems. Concavity can be defined as the negative of convexity
(see above). In other words, a function is concave if, for any two points on the function, a straight line
connecting the two points lies entirely on or below the function.
512 CHAPTER 14
The following function is strictly concave:
-300
-250
-200
-150
-100
100
150
200
250
300
-50
50
0
0
-10000
-20000
-30000
-40000
-50000
-60000
-70000
-80000
-90000
100
90
80
70
60
50
40
30
20
0
-100 -50 0 50 100
Graph of ABS(X)
Here, there is an abrupt bend in the graph at zero. This can dramatically increase solution times as well
as affect the accuracy of the solution. Additional nonsmooth functions are @MAX, @MIN, @SMAX,
@SMIN, and any of the probability functions that use linear interpolation to return results for
nonintegral arguments.
Perhaps even more confounding than functions with sharp bends are discontinuous functions that have
actual breaks in their graphs. Discontinuous functions in LINGO include @SIGN and @FLOOR.
In simplified terms, LINGO searches along the path of a function to find a maximum or minimum
point representing an optimal solution. As LINGO is searching along a function, it uses the function’s
derivatives to help guide the search. When derivatives vanish, as they do at sharp corners, LINGO is
“blinded” and is unable to “see” around the sharp corner. Thus, dealing with functions containing
breaks and sharp bends is considerably more difficult than dealing with smooth, continuous functions.
Where possible, nonsmooth functions should be replaced with linear constraints or some combination
of linear constraints and integer variables. As an example, refer to the assembly line balancing
example (ASLBAL.LNG) on page 517 to see how we constructed a model to avoid the use of the
@MAX function in order to maintain linearity.
514 CHAPTER 14
Simplify Relationships
When practical, use linear rather than nonlinear relationships. Some nonlinear expressions can be
reformulated in a linear manner. A simple example is a constraint on the ratio of two variables. Using
the example from earlier in the chapter, consider the following constraint:
X / Y < 10;
This constraint is nonlinear because we are dividing by Y. To linearize the constraint, you can multiply
both sides by Y. The equivalent, linear constraint becomes:
X < 10 * Y;
Avoid nonsmooth relationships when possible. Models with nonsmooth constraints are generally much
more difficult to solve. When possible, approximate the nonsmooth relationship with smooth
expressions and, perhaps, integer variables.
517
518 APPENDIX A
DATA:
! Data taken from Chase and Aquilano, POM;
! Each task has an estimated time required:
A B C D E F G H I J K;
T = 45 11 9 50 15 12 12 12 12 8 9;
ENDDATA
! The model;
! *Warning* may be slow for more than 15 tasks;
! For each task, there must be one assigned
station;
@FOR(TASK(I): @SUM(STATION(K): X(I, K)) = 1);
! Precedence constraints;
! For each precedence pair, the predecessor task
I cannot be assigned to a later station than
its successor task J;
@FOR(PRED(I, J):
@SUM(STATION(K):
K * X(J, K) - K * X(I, K)) >= 0);
! For each station, the total time for the
assigned tasks must be less than the maximum
cycle time, CYCTIME;
@FOR(STATION(K):
@SUM(TXS(I, K): T(I) * X(I, K)) <= CYCTIME);
! Minimize the maximum cycle time;
MIN = CYCTIME;
! The X(I,J) assignment variables are binary integers;
@FOR(TXS: @BIN(X));
END
Model: ASLBAL
ADDITIONAL EXAMPLES 519
END
Model: BLEND
522 APPENDIX A
DATA:
! The plant, their fixed costs
and capacity;
PLANTS, FCOST, CAP =
P1 91 39
P2 70 35
P3 24 31;
! Customers and their demands;
CUSTOMERS, DEM =
C1 15
C2 17
C3 22
C4 12;
! The plant to cust cost/unit
shipment matrix;
COST = 6 2 6 7
4 9 5 3
8 8 1 5;
ENDDATA
! The objective;
[TTL_COST] MIN = @SUM( ARCS: COST * VOL) +
@SUM( PLANTS: FCOST * OPEN);
DATA:
! Parts list;
PART = @FILE( 'MRP.LDT');
! Time periods;
TIME = @FILE( 'MRP.LDT');
DATA:
DATA:
TIME = 10, 14, 3, 3, 7, 4, 10;
ENDDATA
ES( 1) = 0;
LTASK = @SIZE( TASKS);
LS( LTASK) = ES( LTASK);
DATA:
!Use @TABLE() to display the precedence relations set, PRED;
@TEXT() = @TABLE( PRED);
END
Model: PERT
572 APPENDIX A
where,
St = predicted sales, or signal, in period t,
Xt = observed sales in period t, and
α = a constant term between 0 and 1.
From this equation, we can see the closer α is to 1, the more the current observation affects our signal
and, subsequently, the less “memory” our equation has. Frequently, a value for α is chosen in the
range of .01 to .3. In this example, we will solve for an α that minimizes the sum of the squared
prediction errors.
For more information on exponential smoothing, see Winston (1995).
MODEL:
SETS:
PERIODS /1..8/: OBSERVED, ERROR, PREDICT;
ENDSETS
DATA:
! The degree of the objective. N may be changed
to 1 to minimize absolute deviation;
N = 2;
! The observed values of the time series;
OBSERVED = 10 14 12 19 14 21 19 26;
ENDDATA
! Force Period 1 prediction to 10;
PREDICT(1) = 10;
! The objective function;
[OBJ] MIN= @SUM(PERIODS: @ABS(ERROR) ^ N);
! Calculate the forecasts;
@FOR(PERIODS(T) | T #GT# 1:
PREDICT(T) = ALPHA * OBSERVED(T - 1) +
(1 - ALPHA) * PREDICT(T - 1));
! Calculate forecast errors;
@FOR(PERIODS: ERROR = PREDICT - OBSERVED);
! Error terms may be negative as well as positive;
@FOR(PERIODS: @FREE(ERROR));
! Exclude meaningless Alphas of zero or one;
@BND(.01, ALPHA,.9999);
END
Model: SIMXPO
ADDITIONAL EXAMPLES 593
603
604 APPENDIX B
5. THERE IS NO CURRENT MODEL.
Any command that makes sense only if there is a current model in memory will print this
message if invoked without the presence of a model. You need to load a model with the
File|Open command in Windows or the TAKE command on other platforms, or enter a new
model with the File|New command in Windows or the MODEL command on other platforms.
6. TOO MANY NESTED TAKE COMMANDS.
You have exceeded LINGO’s limit of ten nested TAKE commands within a command script.
If possible, try combining some commands into a single file.
7. UNABLE TO OPEN FILE: FILENAME.
The file you tried to read doesn’t exist, or you misspelled its name. Try opening the file again.
8. TOO MANY CONSECUTIVE COMMAND ERRORS. REVERT TO TERMINAL
INPUT.
LINGO prints this message after having encountered a number of consecutive errors in a
command script. LINGO assumes that something has gone seriously awry, closes the script
file, and returns you to command level.
9. NOT USED.
10. NOT USED.
11. INVALID INPUT. A SYNTAX ERROR HAS OCCURRED.
This is the generic error issued by the LINGO compiler when it detects a syntax error. In
Windows, when you close the error box, the cursor will be on the line where the error
occurred. Other versions of LINGO will try print out the general area where the error has
occurred, but LINGO cannot always pinpoint the exact line. Examine this area for any
obvious syntax errors. If you are unable to find any obvious errors, a useful technique is to
comment out small sections of the model until the error goes away. This should give you a
good idea of exactly where the error is occurring.
Syntax errors may also occur if you are not invoking the correct compiler in LINGO. Most
users will choose to build models using the native LINGO syntax, however, some users may
prefer building their models using LINDO systax. LINGO can compile models written in
either native LINGO syntax or LINDO syntax. LINGO chooses the compiler based on a
model's file extension. LINGO models must have an extension of lg4 (the default) or lng.
LINDO models must have an ltx extension. The default model extension may be set by
clicking on: LINGO | Options | Interface | File Format. Each model window's title bar
displays whether it is a LINGO or LINDO model.
No. Interpretation
2 attempt to export an invalid variable
3 ran out of working memory
4 requested duals in primals only mode
5 range values were requested on fixed rows
6 unexpected error, call LINDO Systems Technical
Support
146. THE FOLLOWING NAMES APPEARED IN THE MODEL AND THE DATA:
NAME1 NAME2 NAME3.
If you go to the Model Generator tab on the LINGO|Options command’s dialog box, you will
see a Check for duplicates names in data and model checkbox. When this option is enabled,
LINGO will compare primal set member names to all the variable names used in the model’s
equations. If any duplicates are found, LINGO will print the first three and print this error
message. To enable this option in command-line versions, use the SET CHKDUP 1
command.
147. UNABLE TO EVALUATE ALL @WARN FUNCTIONS.
Conditional expressions contained in @Error! Bookmark not defined.WARN functions
must contain fixed variables only. When this is not the case, LINGO can’t evaluate the
@WARN functions and you will receive this error message.
148. @OLE FUNCTION NOT SUPPORTED ON THIS PLATFORM.
At present, the @OLE function is supported only in Windows versions of LINGO. If you
don’t have a Windows version of LINGO, you can export the data from your spreadsheet to a
file and use the @FILE function in your LINGO model to import the data.
ERROR MESSAGES 619
149. NOT USED.
150. ODBC INTERFACE NOT SUPPORTED ON THIS PLATFORM.
LINGO’s ODBC link to databases is supported only in Windows versions. If you don’t have a
Windows version of LINGO, you can use text files to move data in and out of LINGO. See
Chapter 8, Interfacing with External Files, for more details.
151. @POINTER NOT SUPPORTED ON THIS PLATFORM.
LINGO’s @POINTER function for interfacing with calling applications is supported only in
Windows versions. If you don’t have a Windows version of LINGO, you can use text files to
move data in and out of LINGO. See Chapter 8, Interfacing with External Files, for more
details.
152. COMMAND NOT SUPPORTED ON THIS PLATFORM.
You have selected a command that is not supported on your platform.
153. SET DEFINITIONS NOT ALLOWED IN INIT SECTIONS.
Sets can’t be initialized in an INIT section. You must change the model, so the set is
initialized in either a sets section or in a data section.
154. ATTEMPT TO REDEFINE A PREVIOUSLY DEFINED SET.
You have attempted to define a set twice in the same model. A set name can only be used
once. Choose a different name for the two sets and try again.
155. SET MEMBER NAMES MAY NOT BE OMITTED IN DATA STATEMENTS.
When initializing a set in a model’s data section, you must explicitly list each member. You
may not skip elements as you can with set attributes.
156. INCORRECT NUMBER OF ARGUMENTS IN A DATA SECTION.
ARGUMENT MUST BE A MULTIPLE OF: N
NUMBER OF ARGUMENTS FOUND: M
You have a data statement in your model that doesn’t have the correct number of values to
initialize a specified list of attributes and/or sets. LINGO will let you know how many values
it found and what the number of arguments should be a multiple of. Add or subtract values in
the data statement and try again.
157. ATTEMPT TO USE @INDEX ON AN UNDEFINED SET.
LINGO can’t compile an instance of the @INDEX function without knowing of the existence
of the set that @INDEX is referencing. Move the expression with the @INDEX function after
the set definition to correct this problem.
158. A SET MEMBER NAME WAS EXPECTED.
You have used a function that was expecting a set member name as an argument. Correct the
arguments to the function and try again.
620 APPENDIX B
159. THE FOLLOWING DERIVED SET MEMBER IS NOT CONTAINED IN ITS
PARENT SET.
You have specified a set member in a derived set that is not contained in the parent set. Each
member in a derived set must be derived from a member of its parent set(s). You have
probably misspelled the name of the derived set member. Check the spelling and try again.
160. ONLY ONE SET MAY BE DEFINED IN EACH DATA STATEMENT.
You have attempted to define more than one set in an individual data statement. Break the
data statement into multiple statements with no more that one set per statement.
161. NOT USED.
162. ATTEMPT TO INITIALIZE MIXED DATA TYPES (TEXT AND NUMERIC) FROM
A SINGLE WORKBOOK RANGE.
When specifying a single range for multiple model objects, all the objects must be of the
same data type—either text (set members) or numeric (set attributes). LINGO’s spreadsheet
interface can’t handle ranges with mixed data types. Break the data statement up into two—
one containing text model objects and the other containing numeric objects.
This constraint introduces some large coefficients into the model. If we rewrote the
constraint, so that it is in units of millions of dollars, then we would have:
X + 2.2 * Y + 2.9 * Z <= 5;
The coefficients in this new constraint are much less likely to present a problem.
206. A MODEL MAY NOT BE SOLVED WITH LINEARIZATION AND THE GLOBAL
SOLVER SIMULTANEOUSLY ENABLED. LINEARIZATION WILL BE
TEMPORARILY DISABLED.
The linearization and global solver options may not be simultaneously selected when solving
a model. LINGO will default to using the global solver. You can set the linearization option
on the General tab of the LINGO|Options command, while the global solver option is
controlled on the Global Solver tab.
207. MISSING LEFT PARENTHESIS.
A unmatched right parenthesis was encountered in the model. Use the Edit|Match
Parenthesis command to help pair up parentheses.
208. @WRITEFOR() MAY ONLY APPEAR IN A DATA SECTION.
The @WRITEFOR function is permitted only in the data section of a model. You will need to
move the expression to a data section.
209. RELATIONAL OPERATORS NOT ALLOWED IN @WRITEFOR() STATEMENTS.
The @WRITEFOR function is used to display output. Constraint relational operators may not
be used inside the @WRITEFOR function.
210. INVALID USAGE OF @WRITEFOR() FUNCTION.
The @WRITEFOR function is being used incorrectly. @WRITEFOR may only be used in the
data and calc sections for the purpose of creating reports. Refer to the documentation for the
correct usage of @WRITEFOR.
211. ARITHMETIC ERROR IN OUTPUT OPERATION.
The @WRITEFOR function is being used incorrectly. @WRITEFOR may only be used in the
data section for creating reports. Refer to the documentation for the correct usage of
@WRITEFOR.
626 APPENDIX B
212. SUBSCRIPT OUT OF RANGE ON SET NAME: SET_NAME
A subscript was found to be out of range while attempting to output a set member, . Check
all output operations that refer to the set SET_NAME for correctness.
213. TEXT OPERAND NOT PERMITTED HERE.
A text argument to a function was encountered where something other than text was
expected. Please check all function arguments to be sure they are correct.
214. DUPLICATE INITIALIZATION OF A VARIABLE.
A variable has been initialized more than one time in the model's data sections. Eliminate all
duplicate variable initializations.
635
636 BIBLIOGRAPHY
Lin, S., and B. Kernighan (1973), “An effective Heuristic Algorithm for the Traveling Salesman
Problem.”, Operations Research, vol. 10, pp. 463-471.
Markowitz, H. M., Portfolio Selection, Efficient Diversification of Investments, John Wiley & Sons,
1959.
Nemhauser, G., and L. Wolsey, Integer and Combinatorial Optimization. New York : John Wiley &
Sons, 1988.
Moder, Joseph J., and Salah E. Elmaghraby (editors), Handbook of Operations Research. New York:
Van Nostrand Reinhold Company, 1978.
Schrage, L., Optimization Modeling with LINDO, 5th. ed. Belmont, CA: Duxbury Press, 1997.
_______________, Optimization Modeling with LINGO, 6th. ed. Chicago, IL: LINDO Systems Inc.,
2006.
Wagner, H.M., Principles of Management Science with Applications to Executive Decisions, 2nd ed.
Englewood Cliffs, N.J. : Prentice-Hall, Inc., 1975.
_________, Principles of Operations Research, 2nd ed. Englewood Cliffs, N.J. : Prentice-Hall, Inc.,
1975.
Winston, Wayne L., Introduction to Mathematical Programming: Applications and Algorithms, 3rd
ed. Belmont, CA: Duxbury Press, 1995.
_____________, Operations Research: Applications and Algorithms, 2nd ed. Belmont, CA: Duxbury
Press, 1995.
Index
data envelopment, 527
range, 622
! regression, 542, 584
!, 6, 284 What If, 93
Analytical derivatives, 172, 513
Anderson, D., 439
~ Annuity, present value, 290
~ (End-of-record mark), 315 APISET command, 217, 627
Append output checkbox, 115
Application drop-down box, 177
A AppWizard, 391, 415
Arithmetic operators, 286, 608, 612
About LINGO command, 32, 213
Arrange Icons command, 209
ABS function, 289, 513
ASCII format, 19, 106, 233, 313, 315, 328
ABSINT parameter, 256
Assembly line balancing, 429, 517
Absolute Integrality, 182
Assume Model Is Linear, 163
Absolute integrality tolerance, 182, 256
Attributes
Absolute optimality tolerance, 184, 257
definition, 35
Accelerator key, 102
dynamic link library, 382
Acceptance sampling, 586, 590
examples
Access
knapsack model, 73
datasources, 361–66
marketing models, 467
importing, 321, 369–71
portfolio models, 442, 454
interfacing, 133
production models, 54, 57, 471
Administrator, ODBC, 361–66
exporting, 371
Airline overbooking, 563
importing, 367
Algebraic formulation, 118, 234
initializing, 91
Algebraic reformulation, 279
names, 605
Algorithm
solution report, 27
barrier
AUTOLG.DAT, 121, 325
cold start, 184, 268
AutoUpdate command, 212
error messages, 622
options, 166
parameters, 264 B
warm start, 183, 268
branch-and-bound, 68, 181, 266 Backorders, 547
generalized reduced gradient, 506 Balof, J., 542
simplex Barrier algorithm
cold start, 184, 268 cold start, 184, 268
options, 166 error messages, 622
parameters, 264 options, 166, 183
warm start, 183, 268 parameters, 264
Allowable Increase/Decrease, 140, 240 warm start, 183, 268
ALTER command, 226, 244 BAS file, 410, 411
Analysis BASCTS, 269
conjoint, 469, 525 BASCTS parameter, 277
637
638 INDEX
BASIC, 384 direction, 266, 279
Basis cuts, 179, 269, 277 global solver, 188
Bayes rule, 519 heuristics, 176, 189, 278
BCROSS parameter, 252, 281 optimality tolerance, 184, 257
Best bound, 187, 258 solver, 13, 175
Best Obj field, 13 strategy, 181
Big M coefficients, 160, 275 BRANDPR parameter, 267
BigM Threshhold., 182 BRANDR parameter, 266
BIGMVL parameter, 252, 283 BREAK statement, 476
BIN function
definition, 67, 293
dual values, 82
C
example, 64 C# .NET, 380
options, 180 C/C++, 379, 380, 390–97
syntax, 72 callback function, 404
Binary pointer function, 384
format, 329 user defined function, 414, 415
operators, 286 Calc Section, 96
Binary integer variables Call option, 449, 565
examples, 431 Callable Library, 379–414
functions, 67, 293 Callback functions, 387, 388, 389, 403–14, 410
priority, 181, 267 Capacitated EOQ, 533
syntax, 72–82 Capacitated lot sizing, 551
Binary Search, 487 Capacitated Plant Location, 434, 522
Binding constraints, 18, 204, 231 Capacity, scarce, 551
Binomial distribution, 290, 563, 567 Cartesian product, 41, 56
Black & Scholes pricing model, 451, 565 Cascade command, 209
Black, F., 449, 565 Case sensitivity, 6
Blending problems, 53, 420, 520, 523 Cash flow matching, 453, 569
BND function CAT command, 215, 217
definition, 67, 293 Chemical equilibrium, 524
error messages, 611 CHESS model, 54, 239, 243, 523
examples, 423, 443 Chi-squared distribution, 290
syntax, 88 CHKDUP parameter, 262
Bond portfolio optimization, 453, 569 Class of a model, 11
Bounds Classification statistics, 203, 230, 507
allowable limit, 611 ClassWizard, 392, 406, 407
conflicting, 611 Clipboard, 123, 200
objective, 13 Close All command, 208
variables Close command, 111
functions, 67, 293 Coefficient reduction, 269
global, 278 Coefficients
guidelines for, 514 big M, 160, 275
probing, 176 current, 140, 240
syntax, 88 delta, 160
Bracken, J., 588 matrix of, 169, 199, 229
Branch-and-bound nonzeros, 10, 204, 231
constraint cuts objective, 17
options, 177, 180 objective ranges, 140, 241
parameters, 256, 269, 277 reduction, 179, 269
reports, 204 right-hand side, 241
definition, 68, 473, 505 unity, 204, 231
INDEX 639
COFCTS, 269 strict, 510
Cold start option, 184, 268 Copy command, 123, 313–14
Colon prompt, 15, 215, 221 COS function, 289
Coloring, 130, 148, 270 Covariance, 441, 442
Column length, 376 Covering model, 531
COM command, 217, 615 Crashing, 173, 263, 572
Command scripts, 323–25 Create from File, 355
command-line, 221 Critical path, 56, 61, 571
DLL, 386, 396, 402 Cumulative distribution, 291
OLE automation Current coefficients, 140, 240
importing, 344–48 Cut command, 123
Windows, 115–18, 206 CUTAPP parameter, 256
Command Window command, 149, 206, 261 Cutoff level, 384
Command-line interface, 15, 16, 115, 149 CUTOFF parameter, 267
Commands Cuts
Command-line, 215–84 options, 177, 180, 269, 277
Windows, 102–4, 105–214 parameters, 256
Comments, 6, 284 reports, 204
Compiling the model Cutting Stock, 497
generating, 46 Cycle time, 429, 431
solving, 108, 135, 238
CompuQuick Corporation, 4, 14, 16, 69
Concavity, 511
D
loose, 512 Dantzig pricing, 168, 265
strict, 512 Data envelopment analysis, 527
Conditional qualifier, 43, 47, 63 Data independence, 19, 59, 76
Conjoint analysis, 469, 525 Data section
Constraints definition, 19
binding, 18, 204, 231 error messages, 619
cuts attributes, 605
options, 177, 180 exporting, 617
parameters, 256, 269, 277 importing, 609
reports, 204 names, 609, 619
defining, 4, 5, 22 question marks, 613, 621
examples, 423, 436, 443, 471 examples, 55, 316
fixed, 10 pointer function, 383
forcing, 160 syntax, 91–94
in solutions, 10 Data type, 412
limits, 89 Database User Info command, 121
modeling language, 35 Databases, 359–77
names, 29, 109 exporting
nonlinear, 10 functions, 371–77
options, 177 reports, 247
selective evaluation, 174, 264 importing, 332, 367–71
two-sided, 605 Datasources, 360–66
types, 506–8 linking, 121, 131, 249
Constraints Cuts box, 177 ODBC function, 131, 371
Contacting LINDO, 33, 214 DBGCLD parameter, 281
Continuous variables, 64, 67 DBGLVL parameter, 252, 282
convexity, 188, 279 DBGWRM parameter, 281
Convexity, 510 DBPWD command, 121, 249
loose, 511 DBUID command, 121, 249
640 INDEX
Debt defeasance, 453, 569 syntax, 232
Debug command DIVERT function, 485
command-line, 236 DLLs, 379–414
Windows, 201–3 Domain functions, 64, 67–89, 611
Debugging, 199, 229 Double data type, 412
Decision variables Downloading, 212
defining, 4 Downside risk, 199, 445, 448, 529
examples, 63, 431, 435, 446 Dual computations box, 154
ranges, 139, 240 Dual function, 298
DECOMP parameter, 278 Dual values, 18
Decomposition, 169 enabling, 139, 154, 617
Default button, 142 examples, 56
Degree, 160 integer programming, 82
Delay field, 149 nonzeros, 612
DELETE command, 245 ranges, 139, 140, 241
Delphi, 379, 380 DUALCO, 265
Delta coefficient, 160 DUALCO parameter, 240
Demand backlog model, 547 DUALPR parameter, 265
Dense set, 39 Duplicate names, 164, 607, 618
Depth first, 187, 258 DYNAMB model, 438
DERCMP parameter, 254 Dynamic Link Library, 379–414
Derivatives, 172, 254, 510, 513 functions, 384–89
Derived sets Dynamic programming, 438, 532, 556, 567
defining, 36, 38–41, 622
dense, 54
examples
E
blending models, 53–56, 422 Echo Input, 150
financial models, 446 Echo to screen checkbox, 115
marketing models, 470 ECHOIN parameter, 262
production models, 427, 431 Economic order quantity, 533, 577
exporting, 371 Edit menu, 103, 122–34
sparse, 56–65 Efficient Frontier, 490
Designing a computer cabinet, 583 Eigen values, 510
Desrochers, M., 595 Elapsed Runtime, 10
Devex pricing, 168, 265 ELSE statement, 474
Direct solver, 473, 505 Embedding files, 348–57
Direction box, 181 END command, 16
Disaggregation, 179, 269 End-of-file, 318
DISCTS, 269 End-of-record Mark (~), 315, 318, 427, 608
Display As Icon button, 132 ENDSUBMODEL, 473
Distribution, 291, 528 Entering a Model
binomial, 290, 563, 567 Command-line, 15–16
Chi-squared, 290 Windows, 3–15
cumulative, 291 Environment object, 385
hypergeometric, 291, 586 Eppen/Martin model, 551
Multi-level, 558 Equality relation, 18, 287
Poisson, 291, 457, 535 Equilibrium
Diversification, 441 network, 560
DIVERT command Erlang busy probability, 290, 579
error messages, 617 Erlang loss probability, 291, 457, 536
printing, 17 ERRDLG parameter, 262
reverting, 233 Error messages, 7, 109, 262, 388, 390, 603–33
INDEX 641
Error Text box, 135 linked objects, 133
Errors In Dialogs, 144 saving, 15, 111, 328
Excel LNG, 106, 328
exporting, 343 LRP, 328
linking with LINGO, 125, 331, 385 LTF, 106, 328
OLE automation, 344–58 MPI, 120, 234
OLE function, 331, 342 MPS
TEXT function, 321 definition, 103
Exclamation point, 6, 284 exporting, 118–19, 234
Exercise price, 449, 565 importing, 109, 218
Exit, 284 header, 392, 409
Exit command, 121 importing, 108
EXP function, 289 log files, 632
Expected rate of return, 447 names, 6, 29, 111
Explicit listing, 39, 56, 58 script files, 323–25
Exponential smoothing, 457, 592 types, 106
Export summary report File Format box, 148
ODBC based, 376 FILE function, 106, 296, 315–19
OLE based, 343 examples, 325, 426
Exporting File menu, 103, 105–21
attributes, 371 FILFMT parameter, 261
databases Fill Out Ranges and Tables, 147
functions, 371–77 FILOUT parameter, 281
reports, 247 Filter, 39, 61
MPI files, 120, 234 Final linear feasibility tolerance, 167, 253
MPS files, 118–20 Final nonlinear feasibility tolerance, 171, 253
pasting out, 314 Financial models, 290, 441–56
spreadsheets Find/Find Next command, 128
functions, 336–44 Finite source, Poisson, 460
reports, 247 Fixed constraints, 10
Expressions, 513 Fixed costs, 77
EXTEND command, 246 Fixed MPS format, 218
Extended Solver Status, 13 Fixed ordering charge, 562
EZCOUNT model, 597 Fixed Var Reduction, 158
EZMREPAR model, 460, 534 Fixed variables, 9, 141, 506, 610
EZNEWS model, 535 FLFTOL parameter, 253
Floating point operations, 167, 253
FLOOR function, 289
F Flow Cover, 179, 269
Factorial function, 597 Flow cuts, 269
Feasibility tolerances, 167, 171 FLWCTS, 269
Feasible solution, 167 FNFTOL parameter, 253
Features, new, vii–ix Font selecting, 130
File FOR function
data, 328 constraints, 31, 46
format error messages, 608, 610
ASCII, 19, 106, 233, 313, 315, 328 nesting, 48, 129
BAS, 410, 411 sets, 37, 296
binary, 329 syntax, 43
LDT, 106, 328 variables, 71, 85
LG4 FOR statement, 475
definition, 106 Forcing constraints, 160
642 INDEX
Forecasting, 57, 83, 584, 592 GLBUBD parameter, 278
FORMAT function, 298 GLBVBD parameter, 278
Formatting, 106 Global delta tolerance, 278
FORTRAN Global optimality tolerance, 278
pointer functions, 384 Global optimum, 271, 384, 508, 510
user defined functions, 414, 415 Global solver, 13, 173, 273, 624
FPA function, 290 Global Solver, 143, 173, 188, 189, 191, 193, 310,
FPL function, 290 625
Fragmentation ratio, 218 Global Solver tab, 188
Free MPS format, 218 GO command, 16, 26, 238
Free variables, 67, 83–88, 293 Go To Line command, 129
FREEZE command, 249, 255, 616 GOMCTS, 269
FRMPS command, 218 Gomory, 269
Functions, 35, 285–311 Gomory cuts, 179, 269
callback, 403–14 Gradient, 171, 254, 506, 513
DLL, 384–89 Graphic reports, 139
factorial, 597 Graves, G., 558
interface functions, 315, 618 Greatest common divisor, 269
non-differentiable, 513 Growth rate, 441
pasting, 130 GUB cuts, 179, 204, 231, 269
probability, 290–92 GUBCTS, 269
set-looping, 41–48, 295
user defined, 414–18
utility, 469
H
variable domain, 64, 67–89, 611 Handle usage, 218
Hansen, P., 560
G Hayes, J.M., 524
Header file, 392, 409
GCD cuts, 179, 269 Heap compacts, 218
GCDCTS, 269 HELP command, 217
GEN, 222 Help menu, 104, 209–14
GEN command, 46 Help Topics command, 210
GEN statement, 477 HEURIS parameter, 256
General box, 144 Heuristics, 176, 278, 556, 595, 598
General equilibrium of an economy, 537 HIDE command, 226, 615
General integer variables, 67, 68–72, 293 Hieftje, G.M., 524
General Solver tab, 152–65 Holding costs, 533, 551
Generate command, 46 Home mortgage calculation, 600
Generating a model, 159, 266 HUMNTM parameter, 278
Generator memory used, 10, 218 Hurdle, 186
Geoffrion, A., 558 HURDLE parameter, 257
GIN function Hypergeometric distribution, 291, 586
definition, 67, 293 Hyperplanes, 512
dual values, 82
examples, 69, 71
options, 180
I
syntax, 68 IF function, 309
GLBBRN parameter, 279 IFC statement, 474
GLBBXS parameter, 279 ILFTOL parameter, 253
GLBDLT parameter, 278 ILP, 11
GLBOPT parameter, 278 Implicit set members, 623
GLBREF parameter, 279
INDEX 643
Implied indexing, 44 Integer variables
Importing binary (0/1)
attributes, 367 examples, 431
command scripts, 323–25 functions, 67, 293
databases, 332, 367–71 priority, 181, 267
FILE function, 106, 315–19 syntax, 72–82
MPS files, 108–10, 218 examples, 423
pasting in, 313–14 general, 67, 68–72, 293
set members, 163 limits, 32
spreadsheets, 331–36 solutions, 9, 515
IN function, 293, 608, 612 variable domain functions, 67–82
Include file, 318 Integrality tolerance, 182, 253
Index, 227, 605, 606, 616 Interest rate, 449
implied, 44 Interface functions, 315, 618
internal, 29, 141 Interface tab, 144–51
placeholder, 40, 63 Interfacing with other applications, 379–418
INDEX function Interior point algorithm, 166, 264
error messages, 605, 610 Interrupting the solver, 8, 613, 632
examples, 74 Invalid switch, 613
parameters, 263 Inventory system
set names, 163, 164 demand backlog, 547
Inequality relation, 287 newsboy model, 535, 562
Infeasible model, 18, 201, 236, 384, 613 PERT, 56, 369–71, 374–77, 571
INFTOL parameter, 253 IPTOLA parameter, 257
Init section, 95–99 IPTOLR parameter, 257
importing, 332 IQP, 11
Initial linear feasibility tolerance, 167, 253 Iterations, 167
Initial nonlinear feasibility tolerance, 171, 253 definition, 154, 254
Initializing data dynamic link library, 406
error messages, 623 limit, 254
variables, 514 max passes, 178
INLP, 11 SLP direction, 174, 264
Insert New Object command, 131–33, 355 steepest edge, 168, 174, 263, 265
Installing LINGO ITERS function, 302
command-line, 15 ITRLIM parameter, 254
Windows, 1 ITRSLW parameter, 254
Integer pre-solver tab, 175–79
Integer programming, 616
branch-and-bound
J
constraint cuts Job shop scheduling, 540
options, 177, 180 Just-in-time production, 424, 554
parameters, 256, 269, 277
reports, 204
definition, 68, 473, 505 K
direction, 266 Kehoe, T.J., 537
heuristics, 176, 278 Kernighan, B., 595
optimality tolerance, 184, 257 Knapsack cover, 269
solver, 175 Knapsack cover cuts, 179, 269
strategy, 181 Knapsack problem, 72–77, 245, 541
dual values, 82 KNPCTS, 269
probing, 176
Integer Solver tab, 180–87
644 INDEX
O P
Obj Bound field, 13 PAGE command, 240, 243, 246
OBJCTS, 269 Page size limits, 150
OBJCTS parameter, 255 Parameters
Object Linking and Embedding, 329 error messages, 614, 616
Object Properties command, 134 pre-defined, 240, 247, 253–66
Object, Insert New, 131–33, 355 user defined, 92
Objective coefficient ranges, 140, 240, 242 Paren Match checkbox, 149
Objective coefficients, 17 Parent sets, 38
Objective cuts, 269 Parentheses, 129, 149, 631
Objective function Passwords
bounds, 13 databases, 121, 249
cuts, 179, 269 error messages, 615
definition, 4 HIDE command, 226
developing, 5, 21 licenses
examples Windows, 120
financial models, 443, 447, 455 Paste command, 123, 314
INDEX 647
Paste Function command, 130 Primal/dual interior point algorithm, 166, 264
Paste Special command, 124 Primitive set
PAUSE command, 247 defining, 36–38
PAUSE statement, 484 examples, 36, 49–53, 54, 57
PBN function, 290 exporting, 371
PCX function, 290 names, 162, 607
PEB function, 290, 579 PRIMPR parameter, 265
PEL function, 291, 457, 536, 580 Print command, 15, 112
PERT, 56, 369–71, 374–77, 571 Print Preview command, 114
Peters, D.G., 524 Print Setup command, 113
PFD function, 291 Priority option, 181, 267
PFS function, 291, 460 PRNCLR parameter, 270
PHG function, 291, 586 Probabilities, 446
PIC statement, 479 Probability distribution
Picture command, 104, 199 binomial, 290, 563, 567
PICTURE command, 229 Erlang busy, 290, 579
PILP, 11 Erlang loss, 291, 457, 536
PINLP, 11 Erlang queuing, 457
PIQP, 11 exponential, 457
Plant location, 179, 269, 434, 522 hypergeometric, 291, 586
Platforms, 215 normal, 291, 528
PLOCTS, 269 Poisson, 291, 457, 460, 535
pLSenvLINGO, 385, 395 steady state, 465, 544
Pluto Dogs, 49, 70 t distributions, 528
POINTER function, 297 transition, 465
data section, 382, 384 Probability functions, 290–92
DLL, 385, 389, 395 Probing, 176, 264
Poisson PROD function, 296
finite source, 460 Product form inverse, 506
linear loss function, 291 Production, 551
probability distribution, 291, 457, 535 planning, 551
random variable, 291 product mix, 4, 77–82, 116, 570, 574
Polaris project, 56 rates, 533
Portfolio selection schedules, 424, 554
Markowitz, 441, 445, 448, 539 Production management models, 420–33
scenario based, 529, 575 Programming Example
Portfolio Selection, Markowitz, 490 Binary Search, 487
POW function, 289 Cutting Stock, 497
Powerstation FORTRAN Markowitz Efficient Frontier, 490
pointer functions, 384 Project Evaluation and Review Technique, 56, 369–
user defined functions, 414, 415 71, 374–77, 571
PPL function, 291 Prompts, 15, 215, 221
PPS function, 291 Properties command, 327
PRBLVL parameter, 264 Proportionality, 506
Precedence relations, 429, 431 PSL function, 291
Precision, Numeric, 167 PSN function, 291, 528
Present value, 290 PTD function, 291, 528
Prices Purchase transition matrix, 466
Black & Scholes, 451, 565
computing, 155
options, 449, 567
Q
strategies, 167, 265 QP, 11
648 INDEX
QRAND function, 292, 623 Relationships, 515
Quadratic functions, 9, 576 Relative Integrality, 182
Quadratic MPS format, 218, 622 Relative integrality tolerance, 182, 253
Quadratic recognition, 174 Relative limit box, 178
Quasi-random numbers, 292 Relative optimality tolerance, 184, 185, 257
Question mark prompt, 15, 221 RELEASE statement, 481
Queuing models RELINT parameter, 253
examples REOPTB parameter, 268
Erlang, 457–64, 536, 579–80 REOPTX parameter, 268
machine repairman, 534 Replace command, 128
steady state, 581 Reports, 261
probabilities, 290, 291 export summary report
QUIT command, 284 ODBC based, 376
OLE based, 343
generating, 232
R graphic, 139
R/C format names nonzeros only, 27, 138
options, 110, 155 range report
parameter, 220, 266 command-line, 240
RAND function, 292, 528 error messages, 616
Random numbers, 292, 528, 544, 623 Windows, 139
Range send to command window, 150
analysis, 622 solution report
bounding, 188 command-line, 16, 238, 242
dual values, 139, 140, 241 examples, 26, 56
names exporting, 376
defining, 342 naming constraints, 30
exporting (OLE), 338 tolerances, 147, 167, 253, 267
exporting (WKX), 344 width, 247
importing (OLE), 336 Windows, 14, 137, 138
OLE automation, 345 statistics, 104, 203, 230
reports Resource limits, 4, 5, 19, 78
command-line, 240 Right-hand side
error messages, 616 probing, 176
Windows, 139 range report, 140, 242
Range command, 18, 139 Risk
RANGE command, 240 downside risk, 199, 529
RANGED function, 302 portfolio selection, 441–48, 539, 575
RANGEU function, 303 sampling, 586, 590
Rate of return, 443, 447 RMPS command, 218
RCMPSN parameter, 266 Root filed, 178
RCTLIM parameter, 269 Rounding, 67, 71, 147, 169, 182, 267
Read error, 609 Row name, 27, 606
Redirecting I/O, 328 Runtime, 10, 184
Redo command, 122 Runtime Limits box, 154
REDUCE parameter, 265 RVRT command, 233
Reduced cost, 17
examples, 56 S
ranges, 139, 140, 241
Register command, 211 Sales forecasting, 591
Regression Analysis, 542, 584 Sampling, 292, 586, 588, 590
Relational operators, 287, 605 SAVE command, 17, 233
INDEX 649
Save/Save As command, 15, 110, 111 exporting, 371
Scalar variables names, 607
examples, 19, 22, 431, 570 sufficient, 201, 203, 236, 238
initializing, 92 Sets section, 36–41, 316
Scale Model, 169 Setup cost, 533, 551
SCALEM parameter, 265 Shack4Shades, 83
Scaling, 167, 253, 514, 625 Shadow prices, 18, 140, 241
Scarce capacity, 551 Shipping problem, 19, 35, 434, 558
Scenario portfolio selection, 445, 529, 575 Shortcut, 326
Scholes, M., 449, 565 Shortest route problem, 438
Schrage, L., 82, 419, 527 SIGN function, 289
SCLRDL parameter, 270 Simplex method, 506
SCLRLM parameter, 270 cold start, 184, 268
Script files, 323–25, 402 options, 166, 183
command-line, 221 parameters, 264
DLL, 386, 396 warm start, 183, 268
OLE automation SIN function, 289
importing, 344–48 SIZE function, 59, 295
Windows, 115–18, 206 Slack or Surplus, 18, 167, 253, 433
Script processor, 382 Slack variables, 204, 231, 423
Scripting, 474 Slope, 84, 513
Second order derivatives, 172 Slow progress iteration limit, 171, 254
SECORD parameter, 252, 283 SLP directions, 174, 264
Seed value, 623 SMAX function, 289
SELCON parameter, 264 SMIN function, 289
Select All command, 128, 314 Smooth expressions, 457, 513, 592
Select Font command, 130 SMPI command, 234
Selective constraint evaluation, 174, 264 SMPI statement, 480
Semi-variance risk, 445, 448 SMPS command, 234
Send To Back command, 208 SOLU, 242
SET command, 249, 616 Solution command, 27, 138
SET function, 486 SOLUTION command, 242
Set-looping functions, 41–48, 295 Solution cutoff, 147
Sets, 19–29, 35–65, 623 Solution report
derived sets command-line, 16, 238, 242
defining, 36, 38–41, 622 examples, 26, 56
dense, 54, 431, 623 exporting, 376
examples naming constraints, 30
blending models, 53–56, 422 tolerances, 147, 167, 253, 267
financial models, 446 width, 247
marketing models, 470 Windows, 14, 137, 138
production models, 427, 431 Solutions
exporting, 371 feasible, 167
sparse, 56–65 infeasible, 18, 201, 236, 384, 613
errors, 621 non-optimal, 613
implicit, 623 optimal, 271, 384, 508
importing, 163 unbounded, 201, 236, 384, 612
names, 605 undefined, 612
necessary, 201, 236 undetermined, 384
primitive set Solve command, 7–14, 26, 108, 135, 208, 357
defining, 36–38 SOLVE statement, 474
examples, 36, 49–53, 54, 57 SOLVEL parameter, 264
650 INDEX
Solver, 8, 473, 505 Steady state, 463, 581
branch-and-bound equations, 462
constraint cuts probability, 465, 544
options, 177, 180 queuing model, 462, 581
parameters, 256, 269, 277 Steepest edge, 168, 174, 263, 265
definition, 68, 473, 505 Steps field, 14
direction, 266 Stock options, 449
heuristics, 176, 278 STOP statement, 477
optimality tolerance, 184 Straight line relationships, 506
solver, 13, 175 Strategies, 173
strategy, 181 Strategies box, 192
global, 13, 173, 273 Stratified sampling design, 588
interrupting, 8, 613, 632 STRLEN function, 305
linear, 165, 183, 473, 505 Strong branch, 187, 267
method, 183, 264, 622 STRONG parameter, 267
multistart, 13, 270 SUBMODEL, 473
nonlinear, 473, 505 SUBOUT parameter, 279
quadratic, 174 Successive linear programming, 174, 264, 506
type, 13 Sufficient set, 201, 203, 236, 238
Solver status window, 136 SUM function, 43, 74, 296
error messages, 632 Sum of a set, 296
field definitions, 8–14 Sum of squared prediction, 591
model classification, 506, 507 Sweeney, D., 439
options, 146, 260 Syntax, 6, 7, 91
SONGS model, 593 coloring, 130, 148, 270
Sorting, 594
Spanning tree model, 556
Sparse derived set, 56–65, 607
T
Sparse set, 39 t distribution, 528
Splash screen, 145, 260 TABLE function, 305
Spreadsheets Table of Contents, iii
data section, 19 TAKE command, 221
exporting command scripts, 323
functions, 336–44 Take Commands command, 106, 115–18, 323
reports, 247 TAN function, 289
importing, 331–36 Tanner, K., 558
interfacing with, 125, 313, 315, 331–58 Technical support, 33
SQL, 366 Terminal width parameter, 247, 614
SQR function, 289 TERSE command, 247
SQRT function, 289 Terse output, 147
STABAR parameter, 261 TERSEO parameter, 259
Stack space, 605 Text files, 19, 233, 313, 315, 328
Staff scheduling, 49, 320, 324, 390–403, 531 TEXT function, 297
DLL, 379–99, 382 command scripts, 325
State of the model, 10, 11, 12, 612 Tiered distribution, 558
Statistics, 104, 203, 230, 507 Tile command, 209
STATS command, 230 TIM2RL parameter, 258
Status bar, 3, 102, 145, 261 TIME command, 284
STATUS function, 304 Time to relative tolerance, 184, 185, 258
pointer function, 383 Timesharing system, 460, 534
Status Window command, 146, 208 TIMLIM parameter, 255
STAWIN parameter, 260 TITLE command, 31
INDEX 651
Tolerances, 167, 171, 182, 184 Variable Upper Bound Box, 191
command-line, 217, 248, 249, 253–58 Variables
examples, 467 bounds, 278
Tolerances box, 186, 192 functions, 67, 293
Tool tips, 102, 261 guidelines for, 514
Toolbar, 3, 101, 147, 207, 262, 351 probing, 176
TOOLBR parameter, 262 syntax, 88
Traffic congestion, 561 continuous variables, 64, 67
Transfer method decision variables
command line, 343, 376 defining, 4
Transition probability, 465, 544 examples, 63, 431, 435, 446
Transportation model, 601 ranges, 139, 240
datasources, 360 defining, 5, 21
exporting, 341 fixed variables, 9, 141, 506, 610
importing, 334 free variables, 67, 83–88, 293
linking, 124 initializing, 514
sets, 19 integer variables
Traveling salesman problem, 595 binary
Tree field, 178 examples, 431
Truncating, 68, 119, 151, 235 functions, 67, 293
Two-sided constraints, 605 priority, 181, 267
syntax, 72–82
examples, 423
U general, 67, 68–72, 293
Unary Minus Priority, 157 solutions, 9, 515
UNARYM parameter, 252 variable domain functions, 67–82
UNARYM paramter, 282 limits, 32
Unbounded solution, 201, 236, 384, 612 names, 6, 109, 220
Undefined arithmetic operation, 264, 611, 612 negative, 67, 83–88, 293
Undefined solution, 612 nonlinear, 1, 9
Undetermined solution, 384 production models, 422, 427
Undo command, 122 random, 291
Unit normal linear loss, 291 scalar
Unity coefficients, 204, 231 examples, 19, 22, 431, 570
Unix, 328 initializing, 92
Update interval, 9, 146, 260 slack, 204, 231, 423
Updating LINGO, 212 solver status window, 9
Updating links, 134 transportation models, 440
USEPNM parameter, 263 Variables Assumed Non-Negative, 155
USEQPR parameter, 273 Variance, 441, 448, 449
User defined functions, 414–18 Vehicle routing problem, 598
USER function, 311 VERBOSE command, 247
User ID, 121, 249 Verbose output, 147
User interrupt, 8, 613, 632 Virtual memory, 256
Utility functions, 469 Visual Basic, 380, 397–99
callback function, 410
dynamic link library, 379
V macros in Excel, 344, 347
Values transferred Visual Basic .NET, 380
command-line, 344 Visual C++, 379, 380, 390–97
Variable domain functions, 64, 67–89, 611 callback function, 404
user defined function, 415
652 INDEX
Volatility, 449 sets, 36, 57
WNLINE parameter, 261
WNTRIM parameter, 261
W Wolsey, E., 179
Warm start option, 183, 268 Working memory, 608
WARN function, 311, 467, 610, 618 Workstations, 429, 462, 464
What If analysis, 93 Worst bound, 187, 258
WHILE statement, 475 WRAP function, 85, 295
WIDTH command, 247, 614 Write access, 616, 617, 632
Williams, T., 439 WRITE function, 307
Window menu, 104, 205–9 WRITE statement, 482
Windows commands WRITEFOR function, 308
in brief, 102–4 Writing to files, 319
in depth, 105–214
Winston, W., 82, 592 Z
Wireless Widgets
examples, 19 Zoom, 200
importing, 316, 334