AL control statements
Note
We're working on improving the onboarding experience for AL developers. If you've input for this article, we'd love to hear from you. Please use the Feedback section at the bottom of this article to help us make improvements that you’d like to see.
We also welcome contributions to our docs. New to contributing to docs? Don't worry, read more about contributing here Contribute to the help.
AL code consists of one or more statements, which are executed sequentially in a top-down order. However, you'll often need to control the direct top-down flow of the execution. One or more statements might have to be repeated more than once, or you might have to make the execution of a certain statement conditional. To do so, you use control structures.
The control structures in AL are divided into the following main groups, as described in this article:
Note
In the following sections, conventions for how to structure and align AL code are presented to introduce best practices. In many cases, the structure isn't necessary to get the code to compile, but rather to improve readability.
AL compound statements
In some cases, the AL syntax only lets you use a single statement. However, if you have to run more than one simple statement, the statements can be written as a compound statement by enclosing the them between the begin
and end
keywords.
begin
<Statement 1>;
<Statement 2>;
..
<Statement n>;
end;
The individual statements are separated by a semicolon. In AL, a semicolon is used to separate statements and not to terminate them, as in other programming languages. Nevertheless, an extra semicolon before an end
doesn't cause an error because it's interpreted by the compiler as an empty statement.
Blocks
The begin-end
structure is also called a block. Blocks can be useful to refer to the other control structures in AL.
When begin
follows, then
, else
, or do
it should be on the same line and preceeded by one space character.
Example 1
if (x = y) and (a = b) then begin
x := a;
y := b;
end;
Example 2
if (xxx = yyyyyyyyyy) and
< (aaaaaaaaaa = bbb)
then begin
x := a;
x := y;
a := y;
end else begin
y := x;
y := a;
end;
AL conditional statements
You use conditional statements to specify a condition and one or more commands to execute if the condition is evaluated as true or false. There are two types of conditional statements in AL:
if-then-else
, where there are two choicescase
, where there are more than two choices
If-then-else statements
if-then-else
statements have the following syntax.
if <Condition> then
<Statement1>
[else
<Statement2>];
If <Condition>
is true, then <Statement1>
is executed. If <Condition>
is false, then <Statement2>
is executed.
The square brackets around else <Statement2>
mean that this part of the statement is optional. The else
statement is used when different actions are executed depending on how <Condition>
is evaluated.
You can build more complex control structures by nesting if-then-else
statements. The following example is a typical if-then-else
statement.
if <Condition1> then
if <Condition2> then
<Statement1>
else
<Statement2>;
If <Condition1>
is false, then nothing is executed. If <Condition1>
and <Condition2>
are both true, then <Statement1>
is executed. If <Condition1>
is true and <Condition2>
is false, then <Statement2>
is executed.
Note
A semicolon in front of an else
statement isn't allowed.
Reading several nested if-then-else
statements can be confusing but generally, an else
statement belongs to the last if
statement that lacks an else
statement.
Programming conventions
if
andthen
should be on the same line,else
should be on a separate line.- If there are many or long expressions,
then
should be on a new line and be aligned withif
. - When you write
if
expressions withthen
andelse
parts, write them so that thethen
result is more probable than theelse
one. - If the last statement in the
then
part of anif-then-else
statement is anexit
or anerror
, don't continue with anelse
statement.
Example 3
if x = y then
x := x + 1
else
x := -x - 1;
Example 4
if (xxxxxxxxxx = yyy) and
(aaa = bbbbbbbbbb)
then
x := a
else
y := b;
Example 5
if x <> y then
exit(true);
x := x * 2;
y := y - 1;
Incorrect example
if x < y then
exit(true)
else begin
x := x * 2;
y := y - 1;
end;
Example 6
The following example shows an if-then statement
without the optional else
statement.
if Amount < 1000 then
Total := Total + Amount;
Example 7
The following example shows a nested if-then-else
statement.
if Amount < 1000 then begin
if I > J then
Max := I
else
Max := J;
Amount := Amount * Max;
end else
...
Case statements
Case statements have the following syntax.
case <Expression> of
<Value set 1>:
<Statement 1>;
<Value set 2>:
<Statement 2>;
<Value set n>:
<Statement n>;
[else
<Statement n+1>;]
end;
In this definition, the result of <Expression>
is matched against each value set and <Value set>
must be an expression or a range.
Note
<Expression>
can't be an application object variable, since application objects don't have a comparator.
Case statements are also called multiple option statements and are typically used when you must choose between more than two different actions. The method of the case
statement is as follows:
- The
<Expression>
is evaluated, and the first matching value set executes the associated statement, if there's one. - If no value set matches the value of the expression and the optional else part is omitted, then no action is taken. If the optional else part is used, then the associated statement is executed.
The data type of the value sets must be the same as the data type of <Expression>
or at least be convertible to the same data type.
In most cases, the data type of the value sets is converted to the data type of the evaluated expression. The only exception is if the evaluated expression is a Code variable. If the evaluated expression is a Code
variable, then the value sets aren't converted to the Code
data type.
Note
This type conversion can cause an overflow at runtime if the resulting data type can't hold the values of the datasets.
Learn more about Code variables in Code data type.
Programming conventions
When you use a case statement, indent the value sets by four character spaces. If you've two or more value sets on the same line, then separate them by commas without spaces. The last value set on a line is immediately followed by a colon without a preceding space. The action starts on the line after the value set and is further indented by four character spaces. If there's a begin, then it should be put on a separate line unless it follows else. If a begin follows an else, then it should be on the same line as else.
If there are more than two alternatives, use a case
statement. Otherwise, use an if-then-else
statement.
Example 8
case Field of
Field::A:
begin
x := x + 1;
y := -y - 1;
end;
Field::B:
x := y;
Field::C,
Field::D:
y := x;
else begin
y := x;
a := b;
end;
end;
Example 9
The following AL code prints various messages depending on the value of Number. If the value of Number doesn't match any of the entries in the case
structure, then the else
entry is used as the default.
case Number of
1, 2, 9:
message('1, 2, or 9.');
10 .. 100:
message('In the range from 10 to 100.');
else
message('Neither 1, 2, 9, nor in the range from 10 to 100.');
end;
Example 10
The following AL code shows how value sets in a case
statement are evaluated if the expression is a Code
data type.
MyCode := 'ABC';
case MyCode of
'abc':
message('This message is not displayed.');
'def':
message('This message is not displayed.');
else
message('The value set does not match the expression.');
end;
This example requires that you create the following Code
data type variable.
var
MyCode : Code[10];
The value set 'abc' isn't converted because the evaluated expression MyCode is a code variable.
AL repetitive statements
A repetitive statement is also known as a loop. The following table shows the looping mechanisms in AL.
Looping mechanism | Description |
---|---|
for | Repeats the inner statement until a counter variable equals the maximum or minimum value specified. |
foreach | Repeats the inner statement for each statement in a List, XmlNodeList, XmlAttributeCollection, or JsonArray. |
while | Repeats the inner statement as long as the specified condition is true. The statement in a loop of this kind is repeated zero or more times. |
repeat - until | Repeats the inner statements until the specified conditions evaluate to true. The statements in a loop of this kind are always executed at least one time. |
For-to and for-downto control structure
The following syntax shows the for-to and for-downto statement.
for <Control Variable> := <Start Number> to <End Number> do
<Statement>;
for <Control Variable> := <Start Number> downto <End Number> do
<Statement>;
The data type of <Control Variable>
, <Start Number>
, and <End Number>
must be Boolean, number, time, or date.
Use for-to
and for-downto
statements when you want to execute code for a specific number of times. The <Control Variable> controls the number of times that the code of the inner statement is executed according to the following:
In a
for-to
loop statement, the<Control Variable>
value is increased by one after each iteration. The inner<Statement>
is executed repeatedly until the*<Start Number>
* value is greater than the*<End Number>*
value.In a
for-downto
loop statement, the<Control Variable>
value is decreased by one after each iteration. The inner<Statement>
is executed repeatedly until the<Start Number>
value is less than the<End Number>
value.
Note
When the for
statement is executed, <Start Number>
and <End Number>
are converted to the same data type as <Control Variable>
if it's required. This type conversion can cause a runtime error.
Note
If the value of the <Control Variable>
is changed inside the for
loop, then the behavior isn't predictable. Furthermore, the value of the <Control Variable>
is undefined outside the scope of the for
loop.
Example 11
The following code initiates a for loop that uses the integer control variable named Count.
for Count := 1000 to 100000000000000 do
This example requires the following Integer data type variable.
var
Count : Integer;
When this statement is executed, then a runtime error occurs because the start, and end values are converted to the same data type as the Count
control variable. Count
has been declared as an integer variable. The end number 100000000000000
is outside the valid range for integers, and an error occurs.
Example 12
The following example shows how to nest for statements.
Set the Dimensions
property of variable A
to 5;7
.
The following for
statements could be used to initialize every element in a 5x7 array with the value 23
.
for I := 1 to 5 do
for J := 1 to 7 do
A[I, J] := 23;
This example requires the following Integer data type variables.
var
I : Integer;
J : Integer;
Foreach control structure
You can use the foreach
statement to iterate through List, XmlNodeList, XmlAttributeCollection, and JsonArray expressions.
Tip
With Business Central 2023 release wave 1, you can use the foreach
loop on Text
variables as well, which allows you to detect, and remove or replace individual characters in strings.
The foreach
statement has the following syntax.
foreach <Element> in <List> do
<Statement>;
The <List>
variable must be of the List, XmlNodeList, XmlAttributeCollection, or JsonArray type. The <Element>
variable must be a data type that is compatible with elements specified by the <List>
.
The following code example iterates through a list of customer names and returns each customer name in a message.
procedure ShowCustomerNames(CustomerNames : List of [Text])
var
CustomerName : Text;
begin
foreach CustomerName in CustomerNames do
Message(CustomerName);
end;
While-do control structure
The following syntax shows the while-do
statement.
while <Condition> do
<Statement>;
If <Condition>
is true, then <Statement>
is executed repeatedly until <Condition>
becomes false. If <Condition>
is false from the start, then <*Statement>
is never executed.
The while do
statement can be used when some code should be repeated as long as an expression is true.
Programming conventions
When there's only one condition, put while
and do
on the same line. Put the statements on separate lines and indented by two spaces.
When there are multiple conditions, put the conditions on separate lines, and indented by two spaces and put do
on a separate line that's aligned with while
.
Example 13
while <expr> do
<Statement>;
Example 14
while <expr> do begin
<Statement>;
<Statement>;
end;
Example 15
while <expr> and
<expr> and
<expr>
do begin
<Statement>;
<Statement>;
end;
Example 16
The following AL code increases the variable I until it equals 1000 and displays a message when it's finished.
while I < 1000 do
I := I + 1;
Message(Format(I));
This example requires the following integer data type variable.
var
I: Integer
Repeat-until control structure
The following syntax shows the repeat until
statement.
repeat
<Statements>
until <Condition>
<Statements>
is executed repeatedly until <Condition>
is true.
The repeat until
control structure resembles the while control structure. The difference is that because the repeat until
statement is executed from left to right, the <Statements>
is always executed at least one time, regardless of what the <Condition>
is evaluated to. This contrasts with the while
control structure, which performs the evaluation before the <Statement>
is executed. In the while
control structure, if the first evaluation of <Condition>
returns false, then no statements are executed.
Programming conventions
Always put repeat on a separate line.
Example 17
if x < y then begin
repeat
x := x + 1;
a := a - 1;
until x = y;
b := x;
end;
Example 18
This code uses a repeat-until
loop to count the number of entries in the Customer table.
Count := 0;
if Customer.FindSet() then
repeat
Count := Count + 1;
until Customer.Next() = 0;
Message('The Customer table contains %1 records.', Count);
This example requires the following variables.
var
Customer: Record Customer;
Count: Integer;
The FindSet
method finds the first entry in the table. Each time Next
is called, it steps one record forward. When Next
equals 0, there are no more entries in the table. The loop is exited, and a message displays how many entries were found.
Exit statement
The exit
statement is used to control the flow of the execution. The following syntax shows an exit
statement.
exit([<Value>])
An exit
statement is used to interrupt the execution of an AL trigger. The interruption occurs even when the code is executed inside a loop or a similar structure. The exit
statement is also used when a local method should return a value.
Using exit
without a parameter in a local method corresponds to using the parameter value 0. The AL method will return the value 0
or '' (empty string).
A compile-time error occurs if exit
is called by using a return parameter from either:
- System-defined triggers, or
- Local methods that don't return a value.
Example 19
The following example shows the use of the exit
statement in a local method. Assume that the if
statement is used to detect an error. If the error condition is met, then execution is stopped and the local method returns the error code 1
.
for I := 1 to 1000 do begin
if Amount[I] < Total[I] then
exit(1);
A[I] := Amount[I] + Total[I];
end;
Break statement
You use the break
statement to terminate the iterative statement in which it appears.
break;
You typically use the break
statement in the repeating statements such as for
, while
, foreach
, or repeat
to stop an iteration or loop when certain conditions are met.
Note
The break
statement is different than the Break Method (Report, XMLport). Although both stop an iteration or loop, the break method will also terminate the trigger in which it's run.
Example 20
The following AL code increases the variable Count
by one for each iteration, and terminates the iteration when Count
equals 10.
while Count < 1000 do
begin
Count := Count + 1;
message(FORMAT(Count));
if Count = 10 then
break;
end;
end;
This example requires the following the Integer
data type variable.
var
Count : integer
Continue statement
APPLIES TO: Business Central 2025 release wave 1 (v26) and later.
With Business Central runtime 15.0, you can use the the continue
statement to proceed to the next iteration of the iterative statement in which it appears.
continue;
You typically use the continue
statement in the repeating statements such as for
, while
, foreach
, or repeat
to proceed to the next iteration when certain conditions are met.
Note
Due to backwards compatibility, the continue
statement is designed to be backwards compatible with other elements like procedures and variables, which have the same name. For example, if there's a procedure named Continue
in scope, the statement continue
will be interpreted as invoking that procedure. This backwards compatibility will be removed in the future, therefore it's recommended to rename any conflicting elements.
Example 21
The following AL code increases the variable Count
by one for each iteration, but skips all numbers divisible by 42.
while Count < 1000 do
begin
Count := Count + 1;
if (Count MOD 42 = 0) then
continue;
message(FORMAT(Count));
end;
end;
This example requires the following Integer
data type variable.
var
Count: Integer
Related information
Programming in AL
AL simple statements
Directives in AL
AL essential methods