SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

Lazy Loading, Singleton and Bridge design pattern in JavaScript and in ABAP

In this blog, I will use an example of SCN log on to illustrate its usage in JavaScript and how to simulate the implementation in ABAP.

When we click log on link in SCN:

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

The whole background of SCN home page turns dark which gives user a hint that something occurs under the hood.

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

Let’s assume that SCN creates a dummy div element with proper CSS style to achieve this mask effect. Of course the productive implementation might be completely different and more complex.

Pseudo Implementation version 1

Suppose we have a log on button with a click event handler. Every time it is clicked, a new mask div element is created.

var createMask = function(){
return document.body.appendChild( document. createElement(‘div’) );
}

$(‘#logon_button’).click(function(){
var mask = createMask();
mask.show();
})

Drawback of this version

Unnecessary mask div element creation. It makes sense to make it a singleton so no matter how many times log on button is clicked, only one mask div element exists.

Pseudo Implementation version 2

In this version the mask div element is created in advance.
var mask = document.body.appendChild(document.createElement(‘div’ ) );
$( ‘#logon_button’).click(function(){
mask.show();
})

Drawback of this version

In fact it is one variant of singleton design pattern: eager singleton, which means even the log on button is never clicked ( the user just would like to surf the SCN anonymously ), this mask div element is still created in vain.

Pseudo Implementation version 3

var mask;
var createMask = function(){
if(mask)
return mask;
else{
mask = document,body.appendChild( document.createElement(‘div’) );
return mask;
}
}
Drawback of this version

This version tries to make mask div element lazy loaded: the creation is delayed until it is really needed. And once it is created, its reference is buffered with a global variable mask. It is not a good practice to use global variable to do such task.

Pseudo Implementation version 4

var createMask = function() {
var mask;
return function() {
return mask || ( mask = document.body.appendChild(document.createElement(‘div’)));
}
}();

In this version, we use closure in JavaScript to achieve the lazy load behavior. The free variable, declared within createMask will not be exposed to external consumer, so it acts actually as a private member attribute in OO world and is the best candidate to buffer the created div element instance. Till now we are very near to the final implementation but this version still have space to improve.

Drawback of this version

From single responsibility principle point of view, the function createMask mixes the mask div element creation with element buffering. Is there any approach to decouple these two different tasks?

Pseudo Implementation version 5 – the final one

var singleton = function(fn){
var result;
return function() {
return result || ( result = fn.apply(this,arguments));
}
}
var createMask = singleton(function(){
return document.body.appendChild(document.createElement(‘div’));
});

In this version we use a Bridge design pattern to decouple the concrete staff ( mask div element creation ) with the abstract staff ( created element must be singleton ), so that each staff can change independently without influence on the other.

Lazy Loading in ABAP

In ABAP actually it is also very easy to implement Lazy Loading: we just declare some private member attribute in class or static variable in function module for instance buffer. Nevertheless, let’s try to simulate the implementation done in JavaScript with Bridge pattern, in order to deepen our understanding on Bridge pattern as an ABAPer.
I have a dummy class to simulate DOM node:
class ZCL_DOM_NODE definition
public
final
create public .

public section.
methods CONSTRUCTOR
importing
!IV_NODE_NAME type STRING .
protected section.
private section.
data MV_NODE_NAME type STRING .
ENDCLASS.

CLASS ZCL_DOM_NODE IMPLEMENTATION.
method CONSTRUCTOR.
mv_node_name = iv_node_name.
endmethod.
ENDCLASS.

And I have a function module which is dedicatedly responsible for DOM node creation:

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

The singleton handling is separately covered another Singleton factory API, which enables any function module with singleton feature by wrapping the original function module, delegating the call to original function module, buffering its call result and returning the buffered result if necessary.

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

Let’s first see what is achieved in ABAP.
This is a test report:

REPORT zsingleton_test.

DATA(lv_new_fm) = zcl_singleton_factory=>get( iv_func = ‘ZCREATE_MASK’ ).
DATA: lo_node1 TYPE REF TO zcl_dom_node,
lo_node2 TYPE REF TO zcl_dom_node,
lo_node3 TYPE REF TO zcl_dom_node,
lo_node4 TYPE REF TO zcl_dom_node.
CALL FUNCTION lv_new_fm
EXPORTING
iv_node_name = ‘Jerry’
IMPORTING
eo_node = lo_node1.

CALL FUNCTION lv_new_fm
EXPORTING
iv_node_name = ‘Jerry’
IMPORTING
eo_node = lo_node2.

CALL FUNCTION lv_new_fm
EXPORTING
iv_node_name = ‘Java’
IMPORTING
eo_node = lo_node3.

CALL FUNCTION lv_new_fm
EXPORTING
iv_node_name = ‘Java’
IMPORTING
eo_node = lo_node4.

zcl_singleton_factory=>cleanup( ).

A new function module is dynamically generated by ZCL_SINGLETON_FACTORY based on original function module ZCREATE_MASK. The new function module name is stored in variable lv_new_fm.
The new function module has exactly the same signature as ZCREATE_MASK. When it is called, we can observed that the singleton is fulfilled.

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

The automatically created wrapped function module will be cleared by the following call in the end of test report.

zcl_singleton_factory=>cleanup( ).

Comment out this statement then we can have a look at the content of generated function module. Here below is one example for ZCREATE_MASK.

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

The whole source code of this automatically generated function module are prepared in method ADAPT_SOURCE_CODE:

SAP HANA, SAP HR, SAP Tutorials, SAP Guide, SAP All Guide Tips

Leave a Reply

Your email address will not be published. Required fields are marked *