1 Star 0 Fork 0

ward.peng/abap-cheat-sheets

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
zcl_demo_abap_prog_flow_logic.clas.abap 48.37 KB
一键复制 编辑 原始数据 按行查看 历史
Daniel Reger 提交于 2年前 . Update content
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383
***********************************************************************
*
* ABAP cheat sheet: Program Flow Logic
*
* -------------------------- PURPOSE ----------------------------------
* - Demonstration example for the topic program flow logic covering
* the following:
* - Expressions and functions for conditions
* - Control structures with IF and CASE, including the COND and SWITCH
* operators
* - Unconditional loops with DO
* - Conditional loops with WHILE
* - Handling exceptions
*
* ----------------------- GETTING STARTED -----------------------------
* - Open the class with the ABAP Development Tools (ADT).
* - Choose F9 to run the class.
* - Check the console output.
* - To understand the context and the ABAP syntax used, check the notes
* included in the class as comments or refer to the respective topic
* in the ABAP Keyword Documentation.
* - Due to the amount of output in the console, the examples include
* numbers (e. g. 1) ..., 2) ..., 3) ...) for the individual example
* sections. Plus, the variable name is displayed in most cases. Hence,
* to easier and faster find the relevant output in the console, just
* search in the console for the number/variable name (CTRL+F in the
* console) or use the debugger.
*
* ----------------------------- NOTE -----------------------------------
* The code presented in this class is only meant for supporting the ABAP
* cheat sheets. It is not intended for direct use in a
* production system environment. The code examples in the ABAP cheat
* sheets are primarily intended to provide a better explanation and
* visualization of the syntax and semantics of ABAP statements and not to
* solve concrete programming tasks. For production application programs,
* a dedicated solution should therefore always be worked out for each
* individual case. There is no guarantee for either the correctness or
* the completeness of the code. In addition, there is no legal
* responsibility or liability for possible errors or their consequences
* which occur through the use of the example code.
*
***********************************************************************
"! <p class="shorttext synchronized">ABAP cheat sheet: Program Flow Logic</p>
"! Demonstration example for the topic program flow logic.<br>Choose F9 in ADT to run the class.
CLASS zcl_demo_abap_prog_flow_logic DEFINITION
PUBLIC
FINAL
CREATE PUBLIC .
PUBLIC SECTION.
INTERFACES: if_oo_adt_classrun.
PRIVATE SECTION.
"Structured type for calculation example
TYPES: BEGIN OF calc_results_struc,
sy_tabix LIKE sy-tabix,
calculation TYPE string,
res_if TYPE string,
res_case TYPE string,
res_cond TYPE string,
res_switch TYPE string,
END OF calc_results_struc.
CLASS-METHODS:
check_is_supplied IMPORTING num1 TYPE i DEFAULT 0
num2 TYPE i DEFAULT 0
RETURNING VALUE(res) TYPE string_table,
addition IMPORTING num1 TYPE i DEFAULT 0
num2 TYPE i DEFAULT 0
RETURNING VALUE(res) TYPE i,
calc IMPORTING num1 TYPE i DEFAULT 0
operator TYPE c
num2 TYPE i DEFAULT 0
RETURNING VALUE(res) TYPE calc_results_struc,
validate_email IMPORTING email TYPE string
RETURNING VALUE(is_valid_email) TYPE abap_bool
RAISING lcx_invalid_email,
meth_with_return IMPORTING num TYPE i
RETURNING VALUE(res) TYPE string,
whats_my_user IMPORTING get_name TYPE abap_bool
RETURNING VALUE(name) TYPE string
RAISING lcx_static_exc_class,
prep_calc_result CHANGING VALUE(res) TYPE string,
power2_and_sqrt IMPORTING num TYPE i
RETURNING VALUE(result) TYPE string
RAISING cx_sy_arithmetic_overflow.
CLASS-DATA: exception_text TYPE string,
exception TYPE REF TO cx_root.
ENDCLASS.
CLASS zcl_demo_abap_prog_flow_logic IMPLEMENTATION.
METHOD if_oo_adt_classrun~main.
DATA(output) = NEW zcl_demo_abap_display( out ).
output->display( `ABAP Cheat Sheet Demonstration Example: Program Flow Logic` ).
**********************************************************************
output->display( `1) Control Structure with IF` ).
"Simple control structure realized by an IF ... ELSEIF ... ELSE ... ENDIF.
"statement. Multiple statement blocks can be included, of which only 1 is
"executed at most and depending on conditions.
DATA(operator) = `+`.
DATA(num1) = 5.
DATA(num2) = 7.
IF operator = `+`.
output->display( |The result of { num1 } { operator } { num2 } is { num1 + num2 }. | ).
ELSEIF operator = `-`.
output->display( |The result of { num1 } { operator } { num2 } is { num1 - num2 }. | ).
ELSE.
output->display( |The operator { operator } is not possible.| ).
ENDIF.
**********************************************************************
output->next_section( `2) IF: Checking sy-subrc` ).
"A prominent use case for IF statements: Checking sy-subrc.
"In the case below, a FIND statement is used. If there is a finding,
"sy-subrc has the value 0.
DATA(to_be_found) = `AB`.
DATA(string_to_search) = `ABAP is great!`.
FIND to_be_found IN string_to_search.
IF sy-subrc = 0.
output->display( |'{ to_be_found }' was found in the string '{ string_to_search }'.| ).
ELSE.
output->display( |'{ to_be_found }' was not found in the string '{ string_to_search }'.| ).
ENDIF.
**********************************************************************
output->next_section( `3) Excursion: COND Operator` ).
"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 example provides an output based on the time stamp.
DATA(syst_time) = cl_abap_context_info=>get_system_time( ).
DATA(greetings) = COND #( WHEN syst_time BETWEEN '050001' AND '120000'
THEN |It's { syst_time TIME = ISO }. Good morning, { sy-uname }.|
WHEN syst_time BETWEEN '170001' AND '210000'
THEN |It's { syst_time TIME = ISO }. Good evening, { sy-uname }.|
WHEN syst_time BETWEEN '210001' AND '050000'
THEN |It's { syst_time TIME = ISO }. Good night, { sy-uname }.|
ELSE |It's { syst_time TIME = ISO }. Hallo, { sy-uname }.| ).
output->display( input = greetings name = `greetings`).
**********************************************************************
output->next_section( `4) Expressions and Functions for Conditions` ).
"Control structures are generally controlled by logical expressions that
"define conditions for operands. The result of such an expression is either true or false.
"The example demonstrates a selection of possible expressions and operands of such expressions
"using a big IF statement. It includes multiple expressions combined by AND to demonstrate
"different options. Here, it is just meant to touch on many syntax options in one go
"for demonstration purposes.
"Data 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` ) ).
"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 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
"CO, CN ,CA, NA, CS, NS, CP, NP "Comparison operators for character-like data types;
"see the cheat sheet on string processing
"Predicate Expressions
AND empty_string IS INITIAL "Checks whether the operand 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;
"IS NOT BOUND is possible which is valid for the following examples, too
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
"The predicate expression IS SUPPLIED is dealt with further down.
"Predicate Functions
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 converts from c to string and ignores 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( ) ) = abap_true
"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
"Method with exactly 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 CONV i( '2.03' ) = num
AND VALUE string_table( ( `a` ) ( `b` ) ( `c` ) ) = str_table
"Table expression
AND str_table[ 2 ] = `b`.
output->display( `All of the logical expressions are true.` ).
ELSE.
output->display( `At least one of the logical expressions is false.` ).
ENDIF.
**********************************************************************
output->next_section( `5) Predicate Expression with IS SUPPLIED` ).
"The predicate expression IS SUPPLIED is available in method implementations
"and checks whether a formal parameter of a procedure is filled or requested.
"In this example, a method includes two importing parameters that are both
"declared as non-mandatory. The method implementation includes a check with
"IS SUPPLIED and stores the result. The result is returned and displayed.
DATA(is_supplied) = check_is_supplied( num1 = 123 ).
output->display( input = is_supplied name = `is_supplied`).
is_supplied = check_is_supplied( num2 = 456 ).
output->display( input = is_supplied name = `is_supplied`).
**********************************************************************
output->next_section( `6) Control Structure with CASE` ).
"CASE statements are used for case distinctions. If the content of an operand
"specified after WHEN matches the content specified after CASE, the statement
"block is executed. Constant values should be specified as operands.
"The example is basically the same as above.
DATA(op) = '-'.
DATA(n1) = 8.
DATA(n2) = 3.
"Simple calculation
CASE op.
WHEN '+'.
output->display( |The result of { n1 } { op } { n2 } is { n1 + n2 }. | ).
WHEN '-'.
output->display( |The result of { n1 } { op } { n2 } is { n1 - n2 }. | ).
WHEN OTHERS.
output->display( |The operator { op } is not possible.| ).
ENDCASE.
**********************************************************************
output->next_section( `7) CASE TYPE OF` ).
"CASE TYPE OF: Checks the type of object reference variables
"Creating an object reference variable and an object
DATA oref_check TYPE REF TO object.
oref_check = NEW zcl_demo_abap_prog_flow_logic( ).
CASE TYPE OF oref_check.
WHEN TYPE zcl_demo_abap_prog_flow_logic.
output->display( |Type zcl_demo_abap_prog_flow_logic? True!| ).
WHEN TYPE if_oo_adt_classrun.
output->display( |Type if_oo_adt_classrun? True!| ).
WHEN TYPE zcl_demo_abap_sql.
output->display( |Type zcl_demo_abap_sql? True!| ).
WHEN OTHERS.
output->display( |Other type.| ).
ENDCASE.
"The same logic as above is realized in the following IF statements
"using IS INSTANCE OF.
"In the example, the type check for if_oo_adt_classrun 'comes first'.
"This type is also true for the object reference variable. This class
"implements the interface.
IF oref_check IS INSTANCE OF if_oo_adt_classrun.
output->display( |Type if_oo_adt_classrun? True!| ).
ELSEIF oref_check IS INSTANCE OF zcl_demo_abap_prog_flow_logic.
output->display( |Type zcl_demo_abap_prog_flow_logic? True!| ).
ELSEIF oref_check IS INSTANCE OF zcl_demo_abap_sql.
output->display( |Type zcl_demo_abap_sql? True!| ).
ELSE.
output->display( |Other type.| ).
ENDIF.
**********************************************************************
output->next_section( `8) Excursion: SWITCH Operator` ).
"The conditional operator SWITCH can also be used to make case
"distinctions in operand positions. Such conditional expressions have
"a result that is dependent on the logical expressions.
"The example provides an output based on the date.
DATA(syst_date) = cl_abap_context_info=>get_system_date( ).
DATA(switch_res) = SWITCH #( syst_date+4(2) "Extracting the month
WHEN '01' THEN `January`
WHEN '02' THEN `February`
WHEN '03' THEN `March`
WHEN '04' THEN `April`
WHEN '05' THEN `May`
WHEN '06' THEN `June`
WHEN '07' THEN `July`
WHEN '08' THEN `August`
WHEN '09' THEN `September`
WHEN '10' THEN `October`
WHEN '11' THEN `November`
WHEN '12' THEN `December`
ELSE `Oops ...` ).
output->display( input = switch_res name = `switch_res`).
**********************************************************************
output->next_section( `Loops (Iterations)` ).
output->display( `9) Unconditional Loops with DO` ).
"The example demonstrate the restriction of loop passes by specifying
"a number (of maximum loop passes) and the TIMES addition in a DO loop.
"The value of sy-index containing the number of loop passes is output, too.
DATA do_counter TYPE i.
DATA do_sy_index TYPE string.
DO 10 TIMES.
do_counter += 1.
do_sy_index = do_sy_index && sy-index && ` `.
ENDDO.
output->display( input = do_counter name = `do_counter`).
output->display( input = do_sy_index name = `do_sy_index`).
**********************************************************************
output->next_section( `10) Terminating Loops Completely Using EXIT` ).
"Using the EXIT statement, you can terminate a loop completely.
"The program flow resumes after the closing statement of the loop.
CLEAR: do_counter, do_sy_index.
DO.
do_counter += 1.
do_sy_index = do_sy_index && sy-index && ` `.
IF sy-index = 5.
EXIT.
ENDIF.
ENDDO.
output->display( input = do_counter name = `do_counter`).
output->display( input = do_sy_index name = `do_sy_index`).
**********************************************************************
output->next_section( `11) Terminating Loop Passes` ).
"CONTINUE: The current loop pass is terminated immediately and the
" program flow is continued with the next loop pass.
"CHECK: Conditional termination. If the specified logical expression
" is false, the current loop pass is terminated immediately and
" the program flow is continued with the next loop pass.
CLEAR: do_counter, do_sy_index.
DO.
IF sy-index = 2. "skipped
CONTINUE.
ENDIF.
CHECK sy-index <> 5. "skipped
IF sy-index = 8. "terminates the loop completely
EXIT.
ENDIF.
do_counter += 1.
do_sy_index = do_sy_index && sy-index && ` `.
ENDDO.
output->display( input = do_counter name = `do_counter`).
output->display( input = do_sy_index name = `do_sy_index`).
**********************************************************************
output->next_section( `12) Excursion: Terminating Procedures Using RETURN` ).
"RETURN statements immediately terminate the current processing block.
"However, according to the guidelines, RETURN should only be used to exit
"procedures like methods.
"The method implementation includes a check if the passed number is positive.
"If so, the square root is calculated. Otherwise, the method call is terminated.
"That means, the returned value for the second method call is initial.
DATA(return1) = meth_with_return( 81 ).
output->display( input = return1 name = `return1`).
DATA(return2) = meth_with_return( -9 ).
output->display( input = return2 name = `return2`).
**********************************************************************
output->next_section( `Conditional Loops with WHILE` ).
output->display( `13) WHILE Example 1` ).
"The following example highlights the setting of sy-index within loop passes.
"The value is added to an internal table. The loop iteration stops when
"the internal table has 10 lines.
DATA int_itab TYPE TABLE OF i.
WHILE lines( int_itab ) < 10.
int_itab = VALUE #( BASE int_itab ( sy-index ) ).
ENDWHILE.
output->display( input = int_itab name = `int_itab`).
**********************************************************************
output->display( `14) WHILE Example 2` ).
"In the following example, all occurrences of a certain substring
"should be replaced by another string. Instead of, for example, using
"a REPLACE ALL OCCURRENCES statement, a WHILE loop is used which
"contains a FIND statement that stores offset and length of the found
"substring. Then, a REPLACE statement is used and that only replaces
"the substring according to the values. The condition for WHILE is
"tailored in a way that the loop is terminated if there are no more
"findings. The number of findings is also displayed - a number that
"corresponds to the number of loop passes.
DATA(while_string) = `##abap####abap#abap######abap###abapabap##abap`.
DATA count_occ TYPE i.
DATA count_occ_via_sy_index TYPE i.
DATA(subrc) = 0. "separate dobj to store the sy-subrc value
WHILE subrc = 0.
FIND FIRST OCCURRENCE OF `abap` IN while_string
MATCH LENGTH DATA(len)
MATCH OFFSET DATA(off)
MATCH COUNT DATA(cnt).
"Note: cnt is always 1 since it is always the first occurrence.
" len is always 4 -> abap
" Only offset has different values
"A separate data object subrc gets assigned the value of sy-subrc.
"As long as there are findings in the string, the while loop continues.
subrc = sy-subrc.
IF subrc = 0.
count_occ_via_sy_index = sy-index. "to hold the
"To hold the total number of findings from left to right of the string
count_occ = count_occ + cnt.
REPLACE SECTION OFFSET off LENGTH len OF while_string WITH `ABAP`.
ENDIF.
ENDWHILE.
output->display( input = count_occ_via_sy_index name = `count_occ_via_sy_index`).
output->display( input = count_occ name = `count_occ`).
output->display( input = while_string name = `while_string`).
**********************************************************************
output->display( `15) WHILE Example 3` ).
"The example demonstrates the 3 options for loop terminations
"in the context of a WHILE ... ENDWHILE statement.
DATA while_cnt TYPE i.
DATA while_sy_index TYPE string.
DATA(index) = 0.
WHILE index <> 10.
index = sy-index.
IF sy-index = 2. "Skips loop pass
CONTINUE.
ENDIF.
CHECK sy-index <> 5. "Skips loop pass
IF sy-index = 8. "Terminates loop
EXIT.
ENDIF.
while_cnt += 1.
while_sy_index = while_sy_index && sy-index && ` `.
ENDWHILE.
output->display( input = while_cnt name = `while_cnt`).
output->display( input = while_sy_index name = `while_sy_index`).
**********************************************************************
output->next_section( `Loops across Tables` ).
output->display( `16) Loop across Internal Table Using LOOP ... ENDLOOP` ).
"LOOP ... ENDLOOP statements are meant for loops across internal tables.
"For more examples, see the cheat sheet example on internal tables.
"Note: In contrast to the loops above, the system field sy-index is not set.
"Instead, the system field sy-tabix is set and which contains the table index
"of the current table line in the loop pass.
"The example demonstrates a loop across an internal table. It combines loops
"and control structures mentioned before:
"- An internal table includes two numbers and an operator based on which a
" a calculation is carried out.
"- The calculation is done in a method. It is implemented in a way that
" IF and CASE statements as well as COND and SWITCH operators are used
" for the calculation.
"- The calculation result is stored in an internal table which is output.
"- All four results - which are all the same, of course, just demonstrating
" the variety of options to achieve the same regarding control structures -
" are available in the table. Plus, the value in sy-tabix is also added to
" the internal table representing the table index.
TYPES: BEGIN OF calc_struc,
num1 TYPE i,
operator TYPE c LENGTH 1,
num2 TYPE i,
END OF calc_struc.
DATA calc_itab TYPE TABLE OF calc_struc WITH EMPTY KEY.
"Internal table containing numbers and operators on whose basis
"calculations are to be carried out.
calc_itab = VALUE #( ( num1 = 123 operator = '+' num2 = 456 )
( num1 = -10 operator = '+' num2 = 9 )
( num1 = 12 operator = '-' num2 = 89 )
( num1 = -5 operator = '-' num2 = -12 )
( num1 = 11 operator = '*' num2 = 10 )
( num1 = -3 operator = '*' num2 = 3 )
( num1 = 1 operator = '/' num2 = 5 )
( num1 = -40 operator = '/' num2 = 2 )
( num1 = 5 operator = '/' num2 = 0 )
( num1 = 7 operator = '#' num2 = 4 ) ).
DATA calc_results TYPE TABLE OF calc_results_struc WITH EMPTY KEY.
LOOP AT calc_itab ASSIGNING FIELD-SYMBOL(<calc>).
"Method call to calculate and return the result
DATA(res) = calc( num1 = <calc>-num1
operator = <calc>-operator
num2 = <calc>-num2 ).
"Adding the sy-tabix value to the table, too.
res-sy_tabix = sy-tabix.
APPEND res TO calc_results.
ENDLOOP.
output->display( input = calc_results name = `calc_results`).
**********************************************************************
output->display( `17) SELECT Loop` ).
"SELECT ... ENDSELECT statements loop across the result set of a database access.
"For more examples, see the cheat sheet example on ABAP SQL.
"The example is meant to give an idea. The SELECT loop is purposely done across
"an internal table which is also possible - just in the interest of not dealing
"with a database table and not ensuring that a demo database table is filled.
"As above, the example includes simple calculations.
TYPES: BEGIN OF struc4loop,
num TYPE i,
calc_result TYPE i,
END OF struc4loop.
TYPES ty_itab_select_loop TYPE TABLE OF struc4loop WITH EMPTY KEY.
DATA(loop_pass) = 0.
DATA(itab_select_loop) = VALUE ty_itab_select_loop( ( num = 1 )
( num = 2 )
( num = 3 )
( num = 4 )
( num = 5 )
( num = 6 ) ).
SELECT *
FROM @itab_select_loop AS itab
INTO @DATA(wa).
IF sy-subrc = 0.
"Loop pass stored and representing the table index value for the
"ABAP SQL statement further down.
loop_pass += 1.
IF loop_pass <= 3.
wa-calc_result = wa-num * 5.
ELSEIF loop_pass = 6.
"No calculation for this loop pass. Loop is terminated.
EXIT.
ELSE.
wa-calc_result = wa-num * 100.
ENDIF.
"Inserting calculation result in table
MODIFY itab_select_loop FROM wa INDEX loop_pass TRANSPORTING calc_result.
ENDIF.
ENDSELECT.
output->display( input = itab_select_loop name = `itab_select_loop`).
**********************************************************************
output->next_section( `Exception Handling` ).
output->display( `18) TRY Control Structures` ).
"TRY control structures are meant for handling catchable exceptions locally
"The example shows divisions. The predefined exception class cx_sy_zerodivide
"as suitable exception class is used.
"If the exception is not handled, the program is terminated and the runtime
"error COMPUTE_INT_ZERODIVIDE occurs.
"The third calculation is not carried out because the statement block is
"left due to the previous erroneous 0 division.
TRY.
DATA(div1) = 4 / 2.
output->display( input = div1 name = `div1`).
DATA(div2) = 4 / 0.
output->display( input = div2 name = `div2`).
DATA(div3) = 9 / 3.
output->display( input = div3 name = `div3`).
CATCH cx_sy_zerodivide.
output->display( `0 division. The exception was caught.`).
ENDTRY.
"The following example shows a catchable exception that is
"raised if a line is not found when using table expressions.
TRY.
DATA(line) = str_table[ 12345 ].
"The predefined exception class cx_sy_itab_line_not_found
"as suitable exception class is used here.
"If the exception is not handled, the program is terminated
"and the runtime error ITAB_LINE_NOT_FOUND occurs.
CATCH cx_sy_itab_line_not_found.
output->display( `The line was not found. The exception was caught.`).
ENDTRY.
**********************************************************************
output->next_section( `19) Multiple CATCH Blocks` ).
"It is possible to specify multiple exception classes in a list and
"multiple CATCH blocks.
"Note: If there are multiple CATCH blocks for exceptions that are in an inheritance
"relationship, you must pay attention that the more special exceptions are specified
"before the more general ones.
"The calculation example shows multiple CATCH blocks that themselves have more than
"one exception class specified. Here, local exception classes are specified just for
"demonstration purposes. They are not relevant in this TRY control structure.
"Filling internal table of type i as basis for calculations
int_itab = VALUE #( ( 5 ) ( 0 ) ( 987654321 ) ).
LOOP AT int_itab ASSIGNING FIELD-SYMBOL(<fs_int>).
TRY.
output->display( |--- Calculations with { <fs_int> } ---| ).
DATA(calc1) = CONV decfloat34( 1 / <fs_int> ).
output->display( input = calc1 name = `calc1`).
DATA(calc2) = ipow( base = <fs_int> exp = 2 ).
output->display( input = calc2 name = `calc2`).
CATCH cx_sy_arithmetic_overflow lcx_calc_error.
output->display( `Arithmetic overflow. The exception was caught.`).
CATCH cx_sy_zerodivide lcx_some_error.
output->display( `0 division. The exception was caught.`).
ENDTRY.
ENDLOOP.
**********************************************************************
output->next_section( `20) Using an Exception Class Higher Up in the Inheritance Tree` ).
"In the following CATCH block, the predefined exception class cx_sy_arithmetic_error
"is specified. Both cx_sy_zerodivide and cx_sy_arithmetic_overflow are derived from
"cx_sy_arithmetic_error which is an exception class higher up in the inheritance
"tree. Hence, cx_sy_arithmetic_error can be specified and handle both exceptions, too.
"The following example is basically the same as above. However, only one exception
"class is specified.
LOOP AT int_itab ASSIGNING FIELD-SYMBOL(<fs_int_inh>).
TRY.
output->display( |--- Calculations with { <fs_int_inh> } ---| ).
calc1 = 1 / <fs_int_inh>.
output->display( input = calc1 name = `calc1`).
calc2 = ipow( base = <fs_int_inh> exp = 2 ).
output->display( input = calc2 name = `calc2`).
CATCH cx_sy_arithmetic_error.
output->display( `Arithmetic error. The exception was caught.`).
ENDTRY.
ENDLOOP.
**********************************************************************
output->next_section( `21) Storing a Reference to the Exception Object` ).
"You can use the addition INTO plus an object reference variable to store
"a reference to an exception object. It is, for example, relevant to
"determine the exact exception.
"The following example is the same as above using the more general exception
"class cx_sy_arithmetic_error. You can carry out certain tasks, for
"example, retrieving and displaying the exception text. To retrieve exception
"texts, you can call, for example, the method get_text.
LOOP AT int_itab ASSIGNING FIELD-SYMBOL(<fs_int_into>).
TRY.
output->display( |--- Calculations with { <fs_int_into> } ---| ).
calc1 = 1 / <fs_int_into>.
output->display( input = calc1 name = `calc1`).
calc2 = ipow( base = <fs_int_into> exp = 2 ).
output->display( input = calc2 name = `calc2`).
CATCH cx_sy_arithmetic_error INTO exception.
"Note:
"- The object reference variable is of type cx_root.
"- You could also create the variable inline, e. g. ... INTO DATA(exc).
"Retrieving and displaying exception text
exception_text = exception->get_text( ).
output->display( input = exception_text name = `exception_text`).
ENDTRY.
ENDLOOP.
**********************************************************************
output->next_section( `22) Raising Exceptions Programmatically` ).
"The following examples demonstrate the ABAP statement RAISE EXCEPTION
"using the addition TYPE. Note there are more additions available that
"are not dealt with here.
"ABAP 'allows' zero division if the first operand is also 0 as shown
"here.
DATA(division) = 0 / 0.
output->display( input = division name = `division`).
"In this example, the appropriate exception - the predefined exception
"class cx_sy_zerodivide - is raised.
TRY.
division = 0 / 0.
"raise predefined exception for 0 division
RAISE EXCEPTION TYPE cx_sy_zerodivide.
CATCH cx_sy_zerodivide INTO DATA(error_oref).
exception_text = error_oref->get_text( ).
output->display( input = exception_text name = `exception_text`).
ENDTRY.
"The following example just demonstrates a locally defined
"exception class that must be raised programmatically.
"In ADT, see the definition of the class in the include, i. e.
"the tab 'Class-relevant Local Types'.
TRY.
RAISE EXCEPTION TYPE lcx_some_error.
CATCH lcx_some_error INTO exception.
exception_text = exception->get_text( ).
output->display( `Default exception text for a local exception class:`).
output->display( input = exception_text name = `exception_text`).
ENDTRY.
**********************************************************************
output->next_section( `23) Nested TRY Control Structure` ).
TRY.
TRY.
RAISE EXCEPTION TYPE lcx_some_error.
CATCH lcx_some_error INTO DATA(error_inner_catch).
output->display( `Inner CATCH`).
RAISE EXCEPTION error_inner_catch.
ENDTRY.
CATCH lcx_some_error.
output->display( `Outer CATCH`).
ENDTRY.
**********************************************************************
output->next_section( `24) Raising an Exception in the Context of Conditional Expressions` ).
"In this example, the optional addition THROW is used in a conditional
"expression. A self-defined local exception class is used.
DATA(number) = 3.
TRY.
DATA(cond_raise) = COND #( WHEN number = 1 THEN `one`
WHEN number = 2 THEN `two`
ELSE THROW lcx_some_error( ) ).
output->display( input = cond_raise name = `cond_raise`).
CATCH lcx_some_error INTO exception.
exception_text = exception->get_text( ).
output->display( input = exception_text name = `exception_text`).
ENDTRY.
**********************************************************************
output->next_section( `25) RAISING Parameter in Method Delcarations` ).
"In the following example, a string table is looped across. The table
"includes valid and invalid email addresses. The validity is checked
"in a method. In the method, a local exception class is raised if the
"email address is invalid.
str_table = VALUE string_table( ( `john.doe@email.com` ) "valid
( `john.doe@#email.com` ) "invalid
( `jane.doe@email.com` ) "valid
( `jane#doe@email.com` ) "invalid
( `max.mustermann@email12.com` ) "valid
( `max.mustermann@email12.c#m` ) "invalid
( `some_name@email.com` ) "valid
( `some_name@email.c` ) "invalid
).
TYPES: BEGIN OF struc_email_check,
email TYPE string,
is_email_valid TYPE abap_bool,
exception_raised TYPE abap_bool,
END OF struc_email_check.
DATA itab_email_check TYPE TABLE OF struc_email_check WITH EMPTY KEY.
LOOP AT str_table ASSIGNING FIELD-SYMBOL(<email>).
TRY.
DATA(email_valid) = validate_email( email = <email> ).
DATA(exc_raised) = abap_false.
CATCH lcx_invalid_email.
email_valid = abap_false.
exc_raised = abap_true.
ENDTRY.
APPEND VALUE #( email = <email>
is_email_valid = email_valid
exception_raised = exc_raised ) TO itab_email_check.
ENDLOOP.
output->display( input = itab_email_check name = `itab_email_check`).
**********************************************************************
output->next_section( `Exception Classes Derived from CX_STATIC_CHECK` ).
"Exception Classes of type cx_static_check force users to handle exceptions.
"Exceptions that are declared in the method signature make users aware of which
"potential exception might occur. And the exceptions should be respected when using
"such a method.
"In this simplified example, a method has a self-defined local exception class
"specified for the RAISING addition. If an actual parameter has a certain
"value, the method raises an exception.
"You can comment in the following line of code to see the enforcement.
"A syntax warning is displayed since there is no proper exception handling.
"DATA(my_user_a) = whats_my_user( get_name = abap_false ).
"Method call with a proper exception handling
TRY.
DATA(my_user_b) = whats_my_user( get_name = abap_true ).
output->display( input = my_user_b name = `my_user_b`).
DATA(my_user_c) = whats_my_user( get_name = abap_false ).
output->display( input = my_user_c name = `my_user_c`).
CATCH lcx_static_exc_class INTO exception.
exception_text = exception->get_text( ).
output->display( input = exception_text name = `exception_texts`).
ENDTRY.
**********************************************************************
output->next_section( `Exception Classes Derived from CX_DYNAMIC_CHECK` ).
"Exception Classes derived from cx_dynamic_check are for exceptions that
"can be checked and avoided by preconditions.
"The checking if a local handling or an explicit declaration in procedure
"interfaces is available is carried out at runtime only ("dynamic check")
"and only in case the exception is indeed raised.
"The following example includes two TRY control structures in which a
"method is called that carries out calculations. In this case, the
"method uses the imported value and carries out two calculations (the
"value powered by 2 and the square root).
"The method signature purposely only includes the exception class
"cx_sy_arithmetic_overflow, i. e. calculation errors regarding the
"square root calculation are 'ignored' here.
"The two exception classes cx_sy_arithmetic_overflow and cx_sy_arg_out_of_domain
"are both derived from cx_dynamic_check.
"This TRY control structure only catches exception for
"cx_sy_arithmetic_overflow. The example does purposely not
"include an error for the square root calculation. If you
"implemented the code outside of the TRY control structure,
"there would not be a syntax warning. Actually, you could
"prevent a calculation failure by setting appropriate values.
TRY.
DATA(res1) = power2_and_sqrt( num = 4 ).
output->display( input = res1 name = `res1`).
DATA(res2) = power2_and_sqrt( num = 123456789 ).
output->display( input = res2 name = `res2`).
CATCH cx_sy_arithmetic_overflow INTO exception.
exception_text = exception->get_text( ).
output->display( input = exception_text name = `exception_text`).
ENDTRY.
"This TRY control structure demonstrates the following:
"If it is determined at runtime that an exception derived from cx_dynamic_check
"is neither locally handled nor an interface is declared appropriately - and the
"exception is raised - a new exception of type cx_sy_no_handler is raised. In this
"case, the attribute 'previous' contains a reference to the original exception.
TRY.
DATA(res3) = power2_and_sqrt( num = 16 ).
output->display( input = res3 name = `res3`).
DATA(res4) = power2_and_sqrt( num = -1 ).
output->display( input = res4 name = `res4`).
"The specification of the suitable exception class does not help here.
"The error is raised while processing the method.
CATCH cx_sy_arg_out_of_domain INTO exception.
exception_text = exception->get_text( ).
output->display( input = exception_text name = `exception_text`).
CATCH cx_sy_no_handler INTO exception.
exception_text = exception->get_text( ).
output->display( input = exception_text name = `exception_text`).
"Attribute 'previous'
"In this case, the information of the actual runtime error is provided:
"Here, it is COMPUTE_SQRT_DOMAIN.
output->display( input = exception->previous name = `exception->previous`).
"For demo purposes, RTTI is used here to retrieve the absolute name of the type,
"i. e. the exception class that was raised. The replace function is only used
"to clear the section 'sub' (which includes '\CLASS=') in 'val'.
DATA(absolute_name) = replace(
val = cl_abap_typedescr=>describe_by_object_ref( exception->previous )->absolute_name
sub = `\CLASS=`
with = `` ).
output->display( input = absolute_name name = `absolute_name`).
ENDTRY.
**********************************************************************
"Excursion: Runtime Errors and Terminating Programs
output->next_section( `Excursion: Runtime Errors and Terminating Programs` ).
"ASSERT statements are followed by a logical expression. If the expression is false,
"the program is terminated and an uncatchable exception is raised resulting in the
"runtime error ASSERTION_FAILED.
"You can comment out the code below to check out the effect.
"In ADT, you will see a message popping up and informing you about the runtime error.
"You can check the details by choosing the "Show" button in the pop-up. Furthermore,
"you can check the content of the "Feed Reader" tab in ADT. There, just expand your
"project and find the runtime errors caused by you.
"Terminating a program using ASSERT
ASSERT 1 = 1.
"ASSERT 1 + 2 = 5 - 3.
DATA(some_flag) = abap_false.
"ASSERT some_flag = abap_true.
"Not handling predefined exception classes
"Caused runtime errors COMPUTE_INT_ZERODIVIDE and ITAB_LINE_NOT_FOUND
"DATA(zero_division) = 1 / 0.
"DATA(nope) = str_table[ 12345 ].
"Not handling self-defined exception classes
"Causes runtime error UNCAUGHT_EXCEPTION
"DATA(is_email_valid) = validate_email( email = `john.doe@email.c##` ).
output->display( `This text is displayed if you left all statements causing a runtime error commented out :)` ).
ENDMETHOD.
METHOD check_is_supplied.
IF num1 IS SUPPLIED.
APPEND `num1 is supplied` TO res.
ELSE.
APPEND `num1 is not supplied` TO res.
ENDIF.
IF num2 IS NOT SUPPLIED.
APPEND `num2 is not supplied` TO res.
ELSE.
APPEND `num2 is supplied` TO res.
ENDIF.
ENDMETHOD.
METHOD addition.
res = num1 + num2.
ENDMETHOD.
METHOD calc.
DATA calc_if TYPE string.
DATA calc_case TYPE string.
DATA calc_cond TYPE string.
DATA calc_switch TYPE string.
"IF statements
IF operator = `+`.
calc_if = num1 + num2.
ELSEIF operator = `-`.
calc_if = num1 - num2.
ELSEIF operator = `*`.
calc_if = num1 * num2.
ELSEIF operator = `/`.
IF num2 = 0.
calc_if = `Division by 0`.
ELSE.
calc_if = num1 / num2.
ENDIF.
ELSE.
calc_if = |Check the operator { operator }.|.
ENDIF.
prep_calc_result( CHANGING res = calc_if ).
"CASE
CASE operator.
WHEN '+'.
calc_case = num1 + num2.
WHEN '-'.
calc_case = num1 - num2.
WHEN '*'.
calc_case = num1 * num2.
WHEN '/'.
CASE num2.
WHEN 0.
calc_case = `Division by 0`.
WHEN OTHERS.
calc_case = num1 / num2.
ENDCASE.
WHEN OTHERS.
calc_case = |Check the operator { operator }.|.
ENDCASE.
prep_calc_result( CHANGING res = calc_case ).
"COND
calc_cond = COND #( WHEN operator = '+'
THEN num1 + num2
WHEN operator = '-'
THEN num1 - num2
WHEN operator = '*'
THEN num1 * num2
WHEN operator = '/' AND num2 = 0 THEN `Division by 0`
WHEN operator = '/' AND num2 <> 0 THEN num1 / num2
ELSE |Check the operator { operator }.|
).
prep_calc_result( CHANGING res = calc_cond ).
"SWITCH
calc_switch = SWITCH #( operator
WHEN '+' THEN num1 + num2
WHEN '-' THEN num1 - num2
WHEN '*' THEN num1 * num2
WHEN '/' THEN SWITCH #( num2 WHEN 0 THEN `Division by 0` ELSE num1 / num2 )
ELSE |Check the operator { operator }.| ).
prep_calc_result( CHANGING res = calc_switch ).
res = VALUE #( calculation = |{ num1 } { operator } { num2 }|
res_if = calc_if
res_case = calc_case
res_cond = calc_cond
res_switch = calc_switch
).
ENDMETHOD.
METHOD meth_with_return.
IF num >= 0.
DATA(sqr_res) = sqrt( num ).
ELSE.
RETURN.
ENDIF.
res = `The method call was not terminated. The square root of ` && num && ` is ` && sqr_res.
ENDMETHOD.
METHOD validate_email.
IF matches( val = email
pcre = `\w+(\.\w+)*@(\w+\.)+(\w{2,4})` ).
is_valid_email = abap_true.
ELSE.
RAISE EXCEPTION TYPE lcx_invalid_email.
ENDIF.
ENDMETHOD.
METHOD whats_my_user.
IF get_name = abap_true.
name = sy-uname.
ELSE.
RAISE EXCEPTION TYPE lcx_static_exc_class.
ENDIF.
ENDMETHOD.
METHOD power2_and_sqrt.
result = |Calculation result:\n{ num } powered by 2 = { ipow( base = num exp = 2 ) }\nSquare root of { num } = { sqrt( num ) }|.
ENDMETHOD.
METHOD prep_calc_result.
FIND PCRE `-$` IN res. "trailing minus
IF sy-subrc = 0.
SHIFT res BY 1 PLACES RIGHT CIRCULAR.
ENDIF.
"trailing .0
IF res CP `*.0*`.
SHIFT res RIGHT DELETING TRAILING ` `.
SHIFT res LEFT DELETING LEADING ` `.
FIND PCRE `\.0$` IN res.
IF sy-subrc = 0.
REPLACE `.0` IN res WITH ``.
ENDIF.
ENDIF.
ENDMETHOD.
ENDCLASS.
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/wardpeng/abap-cheat-sheets.git
git@gitee.com:wardpeng/abap-cheat-sheets.git
wardpeng
abap-cheat-sheets
abap-cheat-sheets
main

搜索帮助