ABAP does not support enumerations as natively and completely as other programming languages.
ABAPers therefore were forced to think up their own solutions and came up with a set of patterns that can be found in the majority of today's object-oriented ABAP code.
When deciding for an enumeration pattern, or wanting to design one of your own, consider the guidelines.
We recommend using either the constant pattern or the object pattern because they combine most advantages and can be generally considered clean.
The widely used interface pattern is also acceptable, but has some slight drawbacks.
Think twice before resorting to the collection pattern. Although it has become widely spread through BOPF, and can be quite convenient in some scenarios, it harbors the danger of degrading into a mess.
CLASS /clean/message_severity DEFINITION PUBLIC ABSTRACT FINAL.
PUBLIC SECTION.
CONSTANTS:
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E'.
ENDCLASS.
CLASS /clean/message_severity IMPLEMENTATION.
ENDCLASS.
used as
IF log_contains( /clean/message_severity=>warning ).
CLASS /clean/message_severity DEFINITION PUBLIC CREATE PRIVATE FINAL.
PUBLIC SECTION.
CLASS-DATA warning TYPE REF TO /clean/message_severity READ-ONLY,
CLASS-DATA error TYPE REF TO /clean/message_severity READ-ONLY.
DATA value TYPE symsgty READ-ONLY.
CLASS-METHODS class_constructor.
METHODS constructor IMPORTING value TYPE /clean/severity.
ENDCLASS.
CLASS /clean/message_severity IMPLEMENTATION.
METHOD class_constructor.
warning = NEW /clean/message_severity( 'W' ).
error = NEW /clean/message_severity( 'E' ).
ENDMETHOD.
METHOD constructor.
me->value = value.
ENDMETHOD.
ENDCLASS.
used as
IF log_contains( /clean/message_severity=>warning->value ).
" inferior pattern
INTERFACE /dirty/message_severity.
CONSTANTS:
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E'.
ENDINTERFACE.
used as
IF log_contains( /dirty/message_severity=>warning ).
" inferior pattern
INTERFACE /dirty/message_constants.
CONSTANTS:
BEGIN OF message_severity,
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E',
END OF message_severity,
BEGIN OF message_lifecycle,
transitional TYPE i VALUE 1,
persisted TYPE i VALUE 2,
END OF message_lifecycle.
ENDINTERFACE.
used as
IF log_contains( /dirty/message_constants=>message_severity-warning ).
CLASS /clean/message_severity DEFINITION PUBLIC ABSTRACT FINAL.
PUBLIC SECTION.
CONSTANTS:
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E'.
ENDCLASS.
CLASS /clean/document_type DEFINITION PUBLIC ABSTRACT FINAL.
PUBLIC SECTION.
CONSTANTS:
sales_order TYPE char02 VALUE '01',
purchase_order TYPE char02 VALUE '02'.
ENDCLASS.
This simplifies searching for enumerations because you can search for the name of the development object instead of hassling with where-used lists and fulltext code searches.
Effective search is important as observations suggest that being unable to find the required enumeration causes people to quickly create constants a second and third time in different places, violating the don't-repeat-yourself principle.
Separate development objects also improve cohesion of your classes because consumers depend only on exactly what they need, not some other enumerations that only accidentally happen to reside in the same development object.
" anti-pattern
CLASS /dirty/common_constants DEFINITION PUBLIC ABSTRACT FINAL.
PUBLIC SECTION.
CONSTANTS:
BEGIN OF message_severity,
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E',
END OF message_severity,
BEGIN OF document_type,
sales_order TYPE char02 VALUE '01',
purchase_order TYPE char02 VALUE '02',
END OF document_type.
ENDCLASS.
CLASS /clean/message_severity DEFINITION PUBLIC ABSTRACT FINAL.
PUBLIC SECTION.
CONSTANTS:
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E'.
ENDCLASS.
Classes allow adding supportive methods,
such as the often-encountered
is_valid
, equals
, contains
, and to_string
methods,
or enumeration-specific ones such as is_more_severe_than
.
They also provide a natural place for unit tests,
especially if you added supportive methods,
but also for common cases such as
in_sync_with_domain_fixed_vals
.
The local test class for the constant pattern can look like this
CLASS ltcl_constant_pattern DEFINITION CREATE PRIVATE
FOR TESTING
RISK LEVEL HARMLESS
DURATION SHORT.
PUBLIC SECTION.
PROTECTED SECTION.
PRIVATE SECTION.
CONSTANTS class_name TYPE classname VALUE '/CLEAN/MESSAGE_SEVERITY'.
CONSTANTS domain_name TYPE domname VALUE 'DOM_MSG_SEVERITY'.
DATA domain_values TYPE STANDARD TABLE OF dd07v.
DATA constants_of_enum_class TYPE abap_attrdescr_tab.
METHODS setup.
METHODS all_values_as_constant FOR TESTING.
METHODS all_constants_in_domain FOR TESTING.
ENDCLASS.
CLASS ltcl_constant_pattern IMPLEMENTATION.
METHOD setup.
CALL FUNCTION 'DD_DOMVALUES_GET'
EXPORTING
domname = domain_name
TABLES
dd07v_tab = domain_values
EXCEPTIONS
OTHERS = 1.
ASSERT sy-subrc = 0.
DATA class_descr TYPE REF TO cl_abap_classdescr.
class_descr ?= cl_abap_typedescr=>describe_by_name( class_name ).
constants_of_enum_class = class_descr->attributes.
ENDMETHOD.
METHOD all_constants_in_domain.
LOOP AT constants_of_enum_class INTO DATA(component).
ASSIGN (class_name)=>(component-name) TO FIELD-SYMBOL(<value>).
ASSERT sy-subrc = 0.
IF NOT line_exists( domain_values[ domvalue_l = <value> ] ).
cl_abap_unit_assert=>fail( |Component { component-name } not found in domain fix values| ).
ENDIF.
ENDLOOP.
ENDMETHOD.
METHOD all_values_as_constant.
DATA value_found TYPE abap_bool.
LOOP AT domain_values INTO DATA(domain_value).
CLEAR value_found.
LOOP AT constants_of_enum_class INTO DATA(component).
ASSIGN (class_name)=>(component-name) TO FIELD-SYMBOL(<value>).
ASSERT sy-subrc = 0.
IF domain_value-domvalue_l = <value>.
value_found = abap_true.
EXIT.
ENDIF.
ENDLOOP.
IF value_found = abap_false.
cl_abap_unit_assert=>fail( |Domainvalue { domain_value-domvalue_l } not available as constant| ).
ENDIF.
ENDLOOP.
ENDMETHOD.
ENDCLASS.
Moreover, classes enforce clean object orientation
through the additions ABSTRACT
and FINAL
.
Interfaces tempt people to "implement" them.
While this shortens their syntax by using the constants
without a leading /dirty/message_severity=>
,
this kind of "inheritance out of convenience"
makes no sense in object orientation
and should be avoided.
" inferior pattern
INTERFACE /dirty/message_severity.
CONSTANTS:
warning TYPE symsgty VALUE 'W',
error TYPE symsgty VALUE 'E'.
ENDINTERFACE.
METHODS log_contains
IMPORTING
minimum_severity TYPE REF TO /clean/message_severity.
The real advantage of enumerations in other programming languages is not that they provide constants, but that they provide all constants, meaning they enforce type safety by making the compiler reject invalid values.
Without type safety, you still get helpful constants
but will find yourself repeating is_valid( )
validations
all over the place.
" inferior pattern
METHODS log_contains
IMPORTING
minimum_severity TYPE symsgty.
So far we have not seen an efficient wide-spread pattern
that exploits the ABAP keyword ENUM
.
We assume there are such patterns
but that people haven't explored them in depth so far.
One of the problems with ENUM
is that it
does not only create constants,
but also new data types alongside.
This makes it harder to apply it to common cases
where the data types already exist,
especially communication with APIs and the database.
If you know a good enum-based pattern, or designed one on your own, let us know about it.