- Program Flow Logic
This cheat sheet gathers information on program flow logic. Find more details here in the ABAP Keyword Documentation.
In ABAP, the flow of a program is controlled by control structures, procedure calls and the raising or handling of exceptions.
Using control structures as an example, you can determine the conditions for further processing of code, for example, if at all or how often a statement block should be executed. Control structures - as, for example, realized by an IF ... ELSEIF ... ELSE ... ENDIF.
statement - can include multiple statement blocks that are executed depending on conditions.
In a very simple form, such an IF
statement might look as follows:
DATA(num) = 1 + 1.
"A simple condition: Checking if the value of num is 2
IF num = 2.
... "Statement block
"Here goes some code that should be executed if the condition is true.
ELSE.
... "Statement block
"Here goes some code that should be executed if the condition is false.
"For example, if num is 1, 8, 235, 0 etc., then do something else.
ENDIF.
- So, such control structures are executed depending on conditions as specified above:
... num = 2 ...
- a logical expression. - Control structures are generally controlled by logical expressions that define conditions for operands.
- The result of such an expression is either true or false.
- Logical expressions are either single relational expressions or expressions combined from one or more logical expressions with Boolean operators like
NOT
,AND
andOR
.
"Single relational expression
IF num = 1.
...
ENDIF.
"Multiple expressions
IF num = 1 AND flag = 'X'.
...
ENDIF.
IF num = 1 OR flag = 'X'.
...
ENDIF.
"Multiple expressions can be parenthesized explicitly
IF ( num = 1 AND flag = 'X' ) OR ( num = 2 AND flag = 'X' ).
...
ENDIF.
- The components of such relational expressions can be comparisons or predicates. Note that for comparison expressions, the comparisons are carried out according to comparison rules.
The following code snippet shows a selection of possible expressions and operands of such expressions using a big IF
statement. Certainly, such a huge statement is far from ideal. Here, the intention is to just cover many syntax options in one go for demonstration purposes. For more information on built-in functions, you can refer to the Built-In Functions cheat sheet.
"Some declarations to be used in the IF statement below
DATA(num) = 2. "integer
DATA(empty_string) = ``. "empty string
DATA(flag) = 'x'.
DATA(dref) = NEW string( `ref` ). "data reference variable
"Object reference variable
DATA oref TYPE REF TO object.
"Creating an object and assigning it to the reference variable
oref = NEW zcl_demo_abap_prog_flow_logic( ).
"Declaration of and assignment to a field symbol
FIELD-SYMBOLS <fs> TYPE string.
ASSIGN `hallo` TO <fs>.
"Creating an internal table of type string inline
DATA(str_table) = VALUE string_table( ( `a` ) ( `b` ) ( `c` ) ).
"The following IF statement includes multiple expressions combined by AND to demonstrate different options
"Comparisons
IF 2 = num "equal, alternative EQ
AND 1 <> num "not equal, alternative NE
AND 1 < num "less than, alternative LT
AND 3 > num "greater than, alternative GT
AND 2 >= num "greater equal, alternative GE
AND 2 <= num "less equal, alternative LE
"Checks whether the content of an operand is within a closed interval
AND num BETWEEN 1 AND 3
AND NOT num BETWEEN 5 AND 7 "NOT negates a logical expression
AND ( num >= 1 AND num <= 3 ) "Equivalent to 'num BETWEEN 1 AND 3';
"here, demonstrating the use of parentheses
"Comparison operators CO, CN ,CA, NA, CS, NS, CP, NP for character-like data types;
"see the cheat sheet on string processing
"Predicate Expressions
AND empty_string IS INITIAL "Checks whether the operand is initial. The expression
"is true, if the operand contains its type-dependent initial value
AND num IS NOT INITIAL "NOT negates
AND dref IS BOUND "Checks whether a data reference variable contains a valid reference and
"can be dereferenced;
"Negation (IS NOT BOUND) is possible which is also valid for the following examples
AND oref IS BOUND "Checks whether an object reference variable contains a valid reference
"IS INSTANCE OF checks whether for a
"a) non-initial object reference variable the dynamic type
"b) for an initial object reference variable the static type
"is more specific or equal to a comparison type.
AND oref IS INSTANCE OF zcl_demo_abap_prog_flow_logic
AND oref IS INSTANCE OF if_oo_adt_classrun
AND <fs> IS ASSIGNED "Checks whether a memory area is assigned to a field symbol
"See the predicate expression IS SUPPLIED in the executable example.
"It is available in method implementations and checks whether a formal parameter
"of a procedure is filled or requested.
"Predicate function: Some examples
AND contains( val = <fs> pcre = `\D` ) "Checks whether a certain value is contained;
"the example uses the pcre parameter for regular expressions;
"it checks whether there is any non-digit character contained
AND matches( val = <fs> pcre = `ha.+` ) "Compares a search range of the argument for the val parameter;
"the example uses the pcre parameter for regular expressions;
"it checks whether the value matches the pattern 'ha'
"and a sequence of any characters
"Predicate functions for table-like arguments
"Checks whether a line of an internal table specified in the table expression
"exists and returns the corresponding truth value.
AND line_exists( str_table[ 2 ] )
"Predicative method call
"The result of the relational expression is true if the result of the functional method call
"is not initial and false if it is initial. The data type of the result of the functional method call,
"i. e. the return value of the called function method, is arbitrary.
"A check is made for the type-dependent initial value.
AND check_is_supplied( )
"It is basically the short form of such a predicate expression:
AND check_is_supplied( ) IS NOT INITIAL
"Boolean Functions
"Determine the truth value of a logical expression specified as an argument;
"the return value has a data type dependent on the function and expresses
"the truth value of the logical expression with a value of this type.
"Function boolc: Returns a single-character character string of the type string.
"If the logical expression is true, X is returned. False: A blank is returned.
"Not to be compared with the constants abap_true and abap_false in relational expressions,
"since the latter convert from c to string and ignore any blanks. Note: If the logical
"expression is false, the result of boolc does not meet the condition IS INITIAL since
"a blank and no empty string is returned. If this is desired, the function xsdbool
"can be used instead of boolc.
AND boolc( check_is_supplied( ) ) = 'X'
"Result has the same ABAP type as abap_bool.
AND xsdbool( check_is_supplied( ) ) = abap_true
"Examples for possible operands
"Data objects as shown in the examples above
AND 2 = 2
AND num = 2
"Built-in functions
AND to_upper( flag ) = 'X'
AND NOT to_lower( flag ) = 'X'
"Numeric functions
AND ipow( base = num exp = 2 ) = 4
"Functional methods
"Assume such a method exists having one return value
AND addition( num1 = 1 num2 = 1 ) = 2
"Calculation expressions
AND 4 - 3 + 1 = num
"String expressions
AND `ha` && `llo` = <fs>
"Constructor expression
AND VALUE i( ) = 0
AND VALUE string_table( ( `a` ) ( `b` ) ( `c` ) ) = str_table
"Table expression
AND str_table[ 2 ] = `b`.
... "All of the logical expressions are true.
ELSE.
... "At least one of the logical expressions is false.
ENDIF.
💡 Note
Logical expressions and functions can also be used in other ABAP statements.
- As already shown above,
IF
statements define statement blocks that can be included in branches. - The statement blocks are executed depending on conditions.
- A maximum of one statement block is executed.
- The check is carried out from top to bottom. The statement block after the first logical expression that is true is executed.
- If none of the logical expressions are true, the statement block after the
ELSE
statement is executed. ELSE
andELSEIF
statements are optional. However, it is recommended that you specify anELSE
so that at least one statement block is executed.- If the end of the executed statement block is reached or if no statement block has been executed, the processing is continued after
ENDIF.
.
DATA(abap) = `ABAP`.
FIND `AB` IN abap.
IF sy-subrc = 0.
"found
...
ELSE.
"not found
...
ENDIF.
"IF statement with multiple included ELSEIF statements
DATA(current_utc_time) = cl_abap_context_info=>get_system_time( ).
DATA greetings TYPE string.
IF current_utc_time BETWEEN '050000' AND '115959'.
greetings = |Good morning, it's { current_utc_time TIME = ISO }.|.
ELSEIF current_utc_time BETWEEN '120000' AND '175959'.
greetings = |Good afternoon, it's { current_utc_time TIME = ISO }.|.
ELSEIF current_utc_time BETWEEN '180000' AND '215959'.
greetings = |Good evening, it's { current_utc_time TIME = ISO }.|.
ELSE.
greetings = |Good night, it's { current_utc_time TIME = ISO }.|.
ENDIF.
Control structures can be nested.
DATA(num) = 1.
DATA(flag) = 'X'.
IF num = 1.
IF flag = 'X'.
...
ELSE.
...
ENDIF.
ELSE.
... "statement block, e. g.
"ASSERT 1 = 0.
"Not to be executed in this example.
ENDIF.
💡 Note
- Control structures can be nested. It is recommended that you do not include more than 5 nested control structures since the code will get really hard to understand. Better go for outsourcing functionality into methods to reduce nested control structures.
- Keep the number of consecutive control structures low.
- If you are convinced that a specified logical expression must always be true, you might include a statement like
ASSERT 1 = 0.
to go sure - as implied in the example'sELSE
statement above. However, anELSE
statement that is never executed might be a hint that logical expressions might partly be redundant.
- The conditional operator
COND
can also be used to implement branches in operand positions that are based on logical expressions. - Such conditional expressions have a result that is dependent on the logical expressions.
- The result's data type is specified after
COND
right before the first parenthesis. It can also be the#
character as a symbol for the operand type if the type can be derived from the context. - All operands specified after
THEN
must be convertible to the result's data type. - See also the Constructor Expressions cheat sheet for more information and examples.
DATA greetings_cond TYPE string.
DATA(current_utc_time_cond) = cl_abap_context_info=>get_system_time( ).
greetings_cond = COND #( WHEN current_utc_time_cond BETWEEN '050000' AND '115959' THEN |Good morning, it's { current_utc_time_cond TIME = ISO }.|
WHEN current_utc_time_cond BETWEEN '120000' AND '175959' THEN |Good afternoon, it's { current_utc_time_cond TIME = ISO }.|
WHEN current_utc_time_cond BETWEEN '180000' AND '215959' THEN |Good evening, it's { current_utc_time_cond TIME = ISO }.|
ELSE |Good night, it's { current_utc_time_cond TIME = ISO }.| ).
CASE
statements are used for case distinctions.- Such statements can also contain multiple statement blocks of which a maximum of one is executed depending on the value of the operand specified after
CASE
. - The check is carried out from top to bottom. If the content of an operand specified after
WHEN
matches the content specified afterCASE
, the statement block is executed. Constant values should be specified as operands. - The
WHEN
statement can include more than one operand using the syntaxWHEN op1 OR op2 OR op3 ...
. - If no matches are found, the statement block is executed after the statement
WHEN OTHERS.
which is optional. - If the end of the executed statement block is reached or no statement block is executed, the processing continues after
ENDCASE.
.
"Getting a random number
DATA(random_num) = cl_abap_random_int=>create( seed = cl_abap_random=>seed( )
min = 1
max = 5 )->get_next( ).
DATA num TYPE string.
CASE random_num.
WHEN 1.
num = `The number is 1.`.
WHEN 2.
num = `The number is 2.`.
WHEN 3 OR 4.
num = `The number is either 3 or 4.`.
WHEN OTHERS.
num = `The number is not between 1 and 4.`.
ENDCASE.
Special control structure introduced by CASE TYPE OF
: Checks the type of object reference variables. An object reference variable with the static type of a class or an interface must be specified after CASE TYPE OF
.
"The example shows the retrieval of type information at runtime (RTTI).
"For more information on RTTI, refer to the Dynamic Programming cheat
"sheet. The result of the method call is a type description object
"that points to one of the classes specified.
DATA stringtab TYPE TABLE OF string WITH EMPTY KEY.
DATA(type_description) = cl_abap_typedescr=>describe_by_data( stringtab ).
CASE TYPE OF type_description.
WHEN TYPE cl_abap_elemdescr.
...
WHEN TYPE cl_abap_refdescr.
...
WHEN TYPE cl_abap_structdescr.
...
WHEN TYPE cl_abap_tabledescr.
"This is the class the type description object points to for
"the example data object.
...
WHEN OTHERS.
...
ENDCASE.
The conditional operator SWITCH
can also be used to make case distinctions in operand positions. As mentioned above for COND
, a result is constructed. The same criteria apply for SWITCH
as for COND
regarding the type. See also the ABAP Keyword Documentation and the Constructor Expressions cheat sheet for more information and examples.
DATA(random_num_switch) = cl_abap_random_int=>create( seed = cl_abap_random=>seed( )
min = 1
max = 5 )->get_next( ).
DATA(num_switch) = SWITCH #( random_num_switch
WHEN 1 THEN `The number is 1.`
WHEN 2 THEN `The number is 2.`
WHEN 3 OR 4 THEN `The number is either 3 or 4.`
ELSE `The number is not between 1 and 4.` ).
-
A statement block specified between
DO
andENDDO
is carried out multiple times. -
The loop is exited when a statement to terminate the loop is reached (
EXIT
, see further down). Otherwise, it is executed endlessly.DATA str_a TYPE string. DO. str_a &&= sy-index. IF sy-index = 5. EXIT. ENDIF. ENDDO. "str_a: 12345
-
To restrict the loop passes, you can use the
TIMES
addition and specify the maximum number of loop passes.DATA str_b TYPE string. DO 9 TIMES. str_b &&= sy-index. ENDDO. "str_b: 123456789
-
The value of the system field
sy-index
within the statement block contains the number of previous loop passes including the current pass.
The following ABAP keywords are available for interrupting and exiting loops:
Keyword | Syntax | Details |
---|---|---|
CONTINUE |
CONTINUE. |
The current loop pass is terminated immediately and the program flow is continued with the next loop pass. |
CHECK |
CHECK log_exp. |
Conditional termination. If the logical expression log_exp is false, the current loop pass is terminated immediately and the program flow is continued with the next loop pass. |
EXIT |
EXIT. |
The loop is terminated completely. The program flow resumes after the closing statement of the loop. |
"------------ CONTINUE ------------
DATA str_c TYPE string.
DO 15 TIMES.
"Continue with the next loop pass if the number is even
"Terminating the loop pass and continuing with the next loop pass if the condition specified
"with the IF statement is true (if the number is even)
IF sy-index MOD 2 = 0.
CONTINUE.
ELSE.
str_c = |{ str_c }{ COND #( WHEN str_c IS NOT INITIAL THEN `, ` ) }{ sy-index }|.
ENDIF.
ENDDO.
"str_c: 1, 3, 5, 7, 9, 11, 13, 15
"------------ CHECK ------------
DATA str_d TYPE string.
DO 15 TIMES.
"Terminating the loop pass and continuing with the next loop pass if the condition is
"true (if the number is odd)
CHECK sy-index MOD 2 = 0.
str_d = |{ str_d }{ COND #( WHEN str_d IS NOT INITIAL THEN `, ` ) }{ sy-index }|.
ENDDO.
"str_d: 2, 4, 6, 8, 10, 12, 14
"CHECK NOT
DATA str_e TYPE string.
DO 15 TIMES.
"Here, it is checked whether the sy-index value is not an even number
CHECK NOT sy-index MOD 2 = 0.
str_e = |{ str_e }{ COND #( WHEN str_e IS NOT INITIAL THEN `, ` ) }{ sy-index }|.
ENDDO.
"str_e: 1, 3, 5, 7, 9, 11, 13, 15
"------------ EXIT ------------
DATA str_f TYPE string.
DO 15 TIMES.
"Terminating the entire loop
IF sy-index = 7.
EXIT.
ELSE.
str_f = |{ str_f }{ COND #( WHEN str_f IS NOT INITIAL THEN `, ` ) }{ sy-index }|.
ENDIF.
ENDDO.
"str_f: 1, 2, 3, 4, 5, 6
💡 Note
RETURN
statements immediately terminate the current processing block. However, according to the guidelines (F1 docu for standard ABAP),RETURN
should only be used to exit procedures like methods.EXIT
andCHECK
might also be used for exiting procedures. However, their use inside loops is recommended.
- Conditional loops introduced by
WHILE
and ended byENDWHILE
are repeated as long as a logical expression is true. - These loops can also be exited using the statements mentioned above.
- Like in
DO
loops, the system fieldsy-index
contains the number of previous loop passes including the current pass. - Also check the conditional loops options using
FOR ... WHILE
with a constructor expression in the Constructor Expressions cheat sheet.
DATA int_itab TYPE TABLE OF i WITH EMPTY KEY.
WHILE lines( int_itab ) = 5.
int_itab = VALUE #( BASE int_itab ( sy-index ) ).
ENDWHILE.
"Content of int_itab:
"1
"2
"3
"4
"5
"The following string replacement example uses a WHILE
"statement for demo purposes - instead of using a REPLACE ALL
"OCCURRENCES statement or the replace function to replace all
"occurrences. The WHILE loop exits when there are no more '#'
"characters to be replaced in the string. The value of the data
"object, which is checked in the logical expression, is then set
"to make the logical expression false.
DATA(str_to_replace) = `Lorem#ipsum#dolor#sit#amet`.
DATA(subrc) = 0.
DATA(counter) = 0.
WHILE subrc = 0.
REPLACE `#` IN str_to_replace WITH ` `.
IF sy-subrc <> 0.
subrc = 1.
ELSE.
counter += 1.
ENDIF.
ENDWHILE.
"str_to_replace: Lorem ipsum dolor sit amet
"counter: 4
Further keywords for defining loops are as follows. They are not dealt with here since they are covered in other ABAP cheat sheets.
LOOP ... ENDLOOP
statements are meant for loops across internal tables. See also the cheat sheet on internal tables.- In contrast to the loops above, the system field
sy-index
is not set. Instead, the system fieldsy-tabix
is set and which contains the table index of the current table line in the loop pass.
- In contrast to the loops above, the system field
FOR
loops: You can also realize loops using iteration expressions withVALUE
andREDUCE
. For more information, refer to the Constructor Expressions cheat sheet.SELECT ... ENDSELECT
statements loop across the result set of a database access. See also the cheat sheet on ABAP SQL.
Procedures can be explicitly called within an ABAP program, thereby influencing the program flow logic.
In modern ABAP programs, classes and methods are the way to go for modularization purposes (instead of function modules and subroutines in most cases; the latter is only available in Standard ABAP). Note that methods and calling methods are described in the context of the ABAP Object Orientation cheat sheet. Find more information and examples there.
💡 Note
In ABAP for Cloud Development, function modules can technically be used, but they are not recommended for new implementations. Many features available in standard ABAP, such as various includes, are not compatible with ABAP for Cloud Development (for example, dynpro-related functionality).
Function modules ...
- are reusable cross-program procedures (i.e. processing blocks callable via an ABAP statement).
- are organized and implemented within function pools.
- are the precursor technology of public methods in global classes.
- are implemented (i.e. their functionality is implemented) between the following statements:
FUNCTION ... . ... ENDFUNCTION.
- have a parameter interface that's similar to ABAP classes. Note: In ADT, the parameter interface of a function module is defined in ABAP pseudo syntax. These statements are not compiled like genuine ABAP statements and are not subject to the regular ABAP syntax checks. Find more information on the parameter interface in the ABAP Keyword Documentation.
- are called using
CALL FUNCTION
statements.
Function pools ...
- serve as a framework for function modules and are organized in include programs.
- The main program is implicitly created, while subordinate include programs are automatically generated, each with specific prefixes and suffixes.
- can contain multiple function modules. They can hold up to 99 function modules.
- are loaded when one of its function modules is called.
- are introduced by the
FUNCTION-POOL
statement.
Syntax to call function modules:
CALL FUNCTION func params.
func
: Character-like data object (for example, a literal) that contains the name of a function module in uppercase letters- Since all function modules have unique names, there is no need to specify the function pool.
params
: Parameter list or table- Incorrectly provided function module names or parameters are not checked until runtime
- Unlike method calls, you cannot specify inline declarations as actual parameters.
- Regarding dynamic function module calls: Static and dynamic function module calls are syntactically identical. In a static call, the function module is specified as a character literal or a constant, with parameters passed statically. Conversely, in a dynamic call, the function module's name is specified in a variable, with parameters passed dynamically. For dynamic calls, you can utilize the
CL_ABAP_DYN_PRG
as shown in the Released ABAP Classes cheat sheet. - When a function module call is made, the system field
sy-subrc
is set to 0. If a non-class-based exception is raised and a value is assigned to handle it, this value updatessy-subrc
.
Example function module calls with parameter passing and exception handling:
"Handling non-class-based exception
DATA it TYPE some_table_type.
CALL FUNCTION 'SOME_FUNCTION_MODULE_A'
EXPORTING
param_a = 'somevalue'
IMPORTING
param_b = it
EXCEPTIONS
not_found = 4.
IF sy-subrc <> 0.
...
ENDIF.
"Handling class-based exception
TRY.
CALL FUNCTION 'SOME_FUNCTION_MODULE_B'
EXPORTING
param_c = 'somevalue'
IMPORTING
param_d = it.
CATCH cx_some_exception INTO DATA(exc).
...
ENDTRY.
"---------- Dynamic function method calls ----------
"Function module name contained in a variable
DATA(func_name) = 'SOME_FUNCTION_MODULE_C'.
CALL FUNCTION func_name ...
"For parameters in a parameter table, use the addition ... PARAMETER-TABLE ptab ...
"ptab: Sorted table of type abap_func_parmbind_tab (line type abap_func_parmbind)
"For exceptions, use the addition ... EXCEPTION-TABLE ...
DATA(ptab) = VALUE abap_func_parmbind_tab( ... ).
CALL FUNCTION func_name PARAMETER-TABLE ptab.
Expand the following section to get a simple executable example:
🟢 Click to expand for more information and example code
The following example demonstrates a simple function module:
- The implementation in the function module represents a calculator. In the function module call, you specify two numbers and an operator.
- The example includes static function module calls, and a dynamic function module call using the
PARAMETER-TABLE
addition.
To get started quickly with copiable code snippets, you can proceed as follows.
- Create a function pool.
- In ADT, you can, for example, right-click the package in which you want to create the function pool and module.
- Choose New -> Other ABAP Repository Object.
- In the input field, enter function, select ABAP Function Group (which is the function pool), and choose Next.
- Make entries in the Name (e.g.
Z_DEMO_ABAP_TEST_FUNC_P
) and Description Field (e.g. Demo function group) fields, and choose Next. - If prompted, select a transport request and choose Finish.
- Create a function module.
- In ADT, you can, for example, right-click the package in which you have created the function pool.
- Choose New -> Other ABAP Repository Object.
- In the input field, enter function, select ABAP Function Module, and choose Next.
- Make entries in the Name (e.g.
Z_DEMO_ABAP_TEST_FUNC_M
), Description Field (e.g. Demo function module), and Function Group (e.g. the previously createdZ_DEMO_ABAP_TEST_FUNC_P
) fields and choose Next. - If prompted, select a transport request and choose Finish.
- The function module is created and can be filled with an implementation.
- You can copy and paste the code below to have a sample implementation.
- To demonstrate the function module, you can create a class, for example, with the name
ZCL_DEMO_ABAP_FUNC_TEST
, and copy and paste the code below.- In ADT, you can run the class by choosing F9. Some output is displayed in the ADT console, demonstrating the result of function module calls.
Code for the function module Z_DEMO_ABAP_TEST_FUNC_M
:
FUNCTION z_demo_abap_test_func_m
IMPORTING
num1 TYPE i
operator TYPE string
num2 TYPE i
EXPORTING
result TYPE string
RAISING
cx_sy_arithmetic_error.
"ABAP 'allows' zero division if both operands are 0.
IF num1 = 0 AND num2 = 0.
RAISE EXCEPTION TYPE cx_sy_zerodivide.
ENDIF.
DATA op TYPE c LENGTH 1.
op = condense( val = operator to = `` ).
result = SWITCH #( op
WHEN '+' THEN |{ num1 } + { num2 } = { num1 + num2 STYLE = SIMPLE }|
WHEN '-' THEN |{ num1 } - { num2 } = { num1 - num2 STYLE = SIMPLE }|
WHEN '*' THEN |{ num1 } * { num2 } = { num1 * num2 STYLE = SIMPLE }|
WHEN '/' THEN |{ num1 } / { num2 } = { CONV decfloat34( num1 / num2 ) STYLE = SIMPLE }|
ELSE `Use one of the operators + - * /` ).
ENDFUNCTION.
Code for the class ZCL_DEMO_ABAP_FUNC_TEST
that can be run in ADT choosing F9:
CLASS zcl_demo_abap_func_test DEFINITION
PUBLIC
FINAL
CREATE PUBLIC .
PUBLIC SECTION.
INTERFACES if_oo_adt_classrun.
PROTECTED SECTION.
PRIVATE SECTION.
ENDCLASS.
CLASS zcl_demo_abap_func_test IMPLEMENTATION.
METHOD if_oo_adt_classrun~main.
"Calling a function module
"For handling a possible wrong operator in the example, you may want to implement
"a separate exception. The simple example just catches calculation errors
"and uses a data object of type string for storing the calculation result.
DATA calculation_result TYPE string.
TRY.
CALL FUNCTION 'Z_DEMO_ABAP_TEST_FUNC_M'
EXPORTING
num1 = 1
operator = `+`
num2 = 2
IMPORTING
result = calculation_result.
CATCH cx_sy_arithmetic_error INTO DATA(exc).
calculation_result = exc->get_text( ).
ENDTRY.
out->write( calculation_result && |\n\n| ).
"More calculation examples; calculations are stored in a table
DATA calculation_result_table TYPE string_table.
TYPES: BEGIN OF s,
num1 TYPE i,
operator TYPE string,
num2 TYPE i,
END OF s,
it_type TYPE TABLE OF s WITH EMPTY KEY.
DATA(itab) = VALUE it_type( ( num1 = 10 operator = `-` num2 = 12 )
( num1 = 15 operator = `*` num2 = 4 )
( num1 = 7 operator = `/` num2 = 2 )
( num1 = 1 operator = `/` num2 = 0 )
( num1 = 0 operator = `/` num2 = 0 )
( num1 = 9999999 operator = `*` num2 = 9999999 ) ).
LOOP AT itab INTO DATA(wa).
TRY.
CALL FUNCTION 'Z_DEMO_ABAP_TEST_FUNC_M'
EXPORTING
num1 = wa-num1
operator = wa-operator
num2 = wa-num2
IMPORTING
result = calculation_result.
CATCH cx_sy_arithmetic_error INTO exc.
calculation_result = |{ wa-num1 } { wa-operator } { wa-num2 } -> Error: { exc->get_text( ) }|.
ENDTRY.
APPEND calculation_result TO calculation_result_table.
ENDLOOP.
out->write( calculation_result_table ).
out->write( |\n\n\n| ).
"----- Dynamic function module call ----
DATA(func_name) = 'Z_DEMO_ABAP_TEST_FUNC_M'.
DATA(ptab) = VALUE abap_func_parmbind_tab( ( name = 'NUM1'
kind = abap_func_exporting
value = NEW i( 3 ) )
( name = 'OPERATOR'
kind = abap_func_exporting
value = NEW string( `+` ) )
( name = 'NUM2'
kind = abap_func_exporting
value = NEW i( 5 ) )
( name = 'RESULT'
kind = abap_func_importing
value = NEW string( ) ) ).
CALL FUNCTION func_name PARAMETER-TABLE ptab.
out->write( data = ptab name = `ptab` ).
ENDMETHOD.
ENDCLASS.
Special function modules exist in Standard ABAP (and not in ABAP for Cloud Development), and for which special properties are specified in the Function Builder:
- Update function modules:
- Typically contain modifying database accesses and can be used to register for later execution
- Are called with
CALL FUNCTION ... IN UPDATE TASK
- Find more information here (note that the links in this section refer to the ABAP Keyword Documentation for Standard ABAP) and in the SAP LUW cheat sheet
- Remote function calls (RFC):
- Remote-enabled function modules are called using the RFC interface
- You can make these calls within the same system or a different one, determined by an RFC destination. Find more information here
- The calls can be ...
- synchronous (sRFC): The calling program waits for the remote function to finish processing; called using
CALL FUNCTION ... DESTINATION
- asynchronous (aRFC): A remote function call that proceeds without waiting for the remotely called function to finish processing; called using
CALL FUNCTION ... STARTING NEW TASK
- transactional
- transactional calls (tRFC) are related to the concept of the SAP LUW. tRFC is considered obsolete.
- Successor technology: Background RFC (bgRFC), executed with the statement
CALL FUNCTION ... IN BACKGROUND UNIT
. Find more information here. - The newer background Processing Framework (bgPF) encapsulates bgRFC to execute time-consuming methods asynchronously. Find more information here.
- synchronous (sRFC): The calling program waits for the remote function to finish processing; called using
- Obsolete procedures you may find in older ABAP programs. Before ABAP Objects was introduced, subroutines were mainly used for local modularization.
- They are implemented between the statements
FORM
andENDFORM
. - Called using
PERFORM
statements - Find more information here. The SAP LUW cheat sheet example also uses subroutines in the context of an SAP LUW (these subroutines are called using
PERFORM ... ON COMMIT
and... ROLLBACK
).
Regarding the exiting of procedures, note the hint mentioned above. The use of RETURN
is recommended.
RETURN
terminates the current processing block. Usually, the statement is intended for leaving processing blocks early.
In case of functional methods, i.e. methods that have one returning parameter, the RETURN
statement can also be specified with an expression. In doing so, the
following statement
res = some_expr.
RETURN.
can also be specified as follows:
RETURN some_expr.
Here, the expression result is passed to the returning parameter without naming it explicitly. As an expression, you can specify a constructor expression (and using type inference with #
means using the type of the returning parameter).
Example:
CLASS zcl_demo_abap DEFINITION
PUBLIC
FINAL
CREATE PUBLIC.
PUBLIC SECTION.
INTERFACES: if_oo_adt_classrun.
PROTECTED SECTION.
PRIVATE SECTION.
CLASS-METHODS multiply
IMPORTING num1 TYPE i
num2 TYPE i
RETURNING VALUE(result) TYPE i.
ENDCLASS.
CLASS zcl_demo_abap IMPLEMENTATION.
METHOD if_oo_adt_classrun~main.
DATA(res1) = multiply( num1 = 2 num2 = 3 ).
DATA(res2) = multiply( num1 = 10 num2 = 10 ).
DATA(res3) = multiply( num1 = 99999999 num2 = 99999999 ).
out->write( res1 ). "6
out->write( res2 ). "100
out->write( res3 ). "0
ENDMETHOD.
METHOD multiply.
TRY.
"result = num1 * num2.
RETURN num1 * num2.
CATCH cx_sy_arithmetic_error.
RETURN VALUE #( ).
ENDTRY.
ENDMETHOD.
ENDCLASS.
Using WAIT UP TO
statements, you can interrupt the program execution by a specified number of seconds.
"First retrieval of the current time stamp
DATA(ts1) = utclong_current( ).
...
WAIT UP TO 1 SECONDS.
...
WAIT UP TO 3 SECONDS.
...
"Second retrieval of the current time stamp after the WAIT statements
DATA(ts2) = utclong_current( ).
"Calculating the difference of the two time stamps
cl_abap_utclong=>diff( EXPORTING high = ts2
low = ts1
IMPORTING seconds = DATA(seconds) ).
"The value of the 'seconds' data object holding the delta of the time stamps
"should be greater than 4.
ASSERT seconds > 4.
Exceptions and runtime errors affect the program flow. Find an overview in the Exceptions and Runtime Errors cheat sheet.
💡 Note
- The executable example ...
- covers the following topics, among others:
- Control structures with
IF
,CASE
, andTRY
- Excursions:
COND
andSWITCH
operators- Expressions and functions for conditions
- Predicate expression with
IS SUPPLIED
- Loops with
DO
,WHILE
, andLOOP
- Terminating loop passes
- Handling exceptions
- The steps to import and run the code are outlined here.
- Disclaimer