0410 Custom Functions in MonetDB part 2

Table Variables

Table variables are temporary in-memory structures. They exist only during the execution of a function. They are mostly used to build up a result set that will be returned from the function. They can be also used as intermediate result storage for procedural logic.

CREATE FUNCTION funcTableDeclaration()
RETURNS TABLE ( Id INTEGER )
BEGIN
   DECLARE TABLE TableDeclaration ( Id INTEGER );
   INSERT INTO TableDeclaration VALUES ( 1 );        RETURN SELECT * FROM TableDeclaration;
END;
SELECT * FROM funcTableDeclaration();
Tables created with DECLARE TABLE statement are mutable and can be modified by using any of the DML statements like INSERT INTO, UPDATE, DELETE…

It is also possible to create a table variable based on an existing table, but without inheriting the data. I will create a sample table:

CREATE TABLE sys.tabNum ( Number INTEGER );
INSERT INTO tabNum ( Number ) VALUES ( 32 );
SELECT * FROM tabNum;
    
   

CREATE OR REPLACE FUNCTION funcTableDeclaration()
RETURNS TABLE( N INTEGER )
BEGIN
   DECLARE TABLE TableDeclaration AS ( SELECT * FROM tabNum ) WITH DATA;
   RETURN SELECT * FROM TableDeclaration;
END;
We have created table variable based on the sample table.

SELECT * FROM funcTableDeclaration();
No data inherited.

Limitations of the Table Variables

CREATE OR REPLACE FUNCTION funcTableDeclaration()
RETURNS TABLE ( Id INTEGER )
BEGIN
   DECLARE TABLE TableDeclaration ( Id INTEGER PRIMARY KEY );
   INSERT INTO TableDeclaration VALUES ( 1 );
   RETURN SELECT * FROM TableDeclaration;
END;
We can not have constraints on the table variables.  

CREATE OR REPLACE FUNCTION funcTableDeclaration()
RETURNS TABLE( Id INTEGER )
BEGIN    
   DECLARE TABLE TableDeclaration ( Id INTEGER );
   INSERT INTO TableDeclaration VALUES ( 1 );
   ALTER TABLE TableDeclaration RENAME COLUMN Id to Code;
   RETURN SELECT * FROM TableDeclaration;
END;
It is not possible to apply ALTER TABLE on the table variable.  

Declare Scaler Variable

CREATE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN
   DECLARE Var1 INTEGER;
   SET Var1 = 77;
   SET Var1 = 88;
   RETURN SELECT Var1;
END;
It is possible to declare and set scaler variable.
Scaler variables are mutable, and we can change their values several times.

SELECT funcVariableDeclaration();

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS TABLE ( Var1 INTEGER, VAR2 CHAR )
BEGIN
   DECLARE Var1 INTEGER, Var2 CHAR;
   SET Var1 = 77;
   SET Var2 = 'A';
   RETURN SELECT Var1, Var2;
END;
We can declare two variables in one statement.

SELECT * FROM funcVariableDeclaration();

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS TABLE ( Var1 INTEGER, VAR2 CHAR )
   BEGIN
   DECLARE Var1 INTEGER, Var2 CHAR;
   SELECT 1, 'A' INTO Var1, Var2;
   RETURN SELECT Var1, Var2;
END;
Beside declaring two variables at once, we can also set the
values of the both variables by the "SELECT INTO" statement.


Expressions for Values

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN
   DECLARE Var1 INTEGER;
   SET Var1 = 32 + 5;
   RETURN SELECT Var1;
END;
Result of an expression can be assigned as a variable value.

SELECT funcVariableDeclaration();


CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN    
   DECLARE Var1 INTEGER;
   SET Var1 = ( SELECT * FROM tabNum );
   DECLARE Var2 INTEGER;
   SET Var2 = Var1 + 5;
   RETURN SELECT Var2;
END;
Result of a query can be assigned to a variable.  

SELECT funcVariableDeclaration();

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN
   DECLARE Var1 INTEGER;
   SET Var1 = ( SELECT * FROM tabNum ) + 5;
   RETURN SELECT Var1;
END;
We can also use this shorter syntax.

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN    
   DECLARE Var1 INTEGER, Var2 INTEGER;
   SET ( Var1, Var2 ) = ( SELECT Number, Number FROM tabNum );     RETURN Var1 / Var2; END;
If the query is returning one row with several columns, we can assign all of those values to several variables in one statement.

Usage of the Scaler Variables

Scaler variables are, also, temporary in-memory structures. We can use them anywhere where we can use constants:

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN
   DECLARE Var1 INTEGER;
   SET Var1 = 32;
   RETURN SELECT * FROM tabNum WHERE Number = Var1;
END;
SELECT funcVariableDeclaration();

But we can not use scaler variable to give a name to a column of a table variable. We can not use variables instead of the identifiers:

CREATE OR REPLACE FUNCTION funcTableDeclaration()
RETURNS TABLE ( Id INTEGER )
BEGIN
   DECLARE Var1 CHAR;
   SET Var1 = 'N';
   DECLARE TABLE TableDeclaration ( Var1 INTEGER );    
   RETURN SELECT N FROM TableDeclaration;
END;
SELECT * FROM funcTableDeclaration();  

CREATE OR REPLACE FUNCTION funcVariableDeclaration()
RETURNS INTEGER
BEGIN
   DECLARE Var1 INTEGER;
   RETURN SELECT Var1;
END;
Variable without assigned value will be NULL.

SELECT funcVariableDeclaration();

Control Flow With CASE

In a police lineup, a witness goes from left to right, scanning each person. When the witness recognizes the culprit, he stops scanning.

This is how the CASE control flow structure works. We examine the values ​​one by one until we find a value that matches the description.

Search CASE

CREATE OR REPLACE FUNCTION funcCASE( argNumber INTEGER )
RETURNS CHAR
BEGIN    
   DECLARE Res CHAR;    
   CASE          
      WHEN argNumber = 1 THEN SET Res = 'A';
      WHEN argNumber = 2 THEN SET Res = 'B';
      ELSE SET Res = 'C';    
   END CASE;
   RETURN Res;
END;
CASE statement is made of conditions checks ( WHEN subclause ) and
the results ( THEN subclause ) when the condition is met.

The results must be complete statements. In this case, it is a SET statement.
If none of the conditions is met, we will return the statement after the "ELSE" keyword.


SELECT funcCASE( 1 );

SELECT funcCASE( 2 );

SELECT funcCASE( 99 );

CREATE OR REPLACE FUNCTION funcCASE( argNumber INTEGER )
RETURNS CHAR

BEGIN
   DECLARE Res CHAR;
   CASE
      WHEN argNumber < 3 THEN SET Res = 'A';
      WHEN argNumber < 5 THEN SET Res = 'B';
      ELSE SET Res = 'C';
   END CASE;
   RETURN Res;
END;
If the conditions overlap, the condition closer to the top of the list will win.
For example, if argNumber is "2", then both the conditions "2 < 3" and "2 < 5" will be true.
The final result will be the letter "A", because it is related to the top condition in the list.

SELECT funcCASE( 2 );

 

Simple CASE

CREATE OR REPLACE FUNCTION funcCASE( argNumber INTEGER )
RETURNS CHAR
BEGIN     
   DECLARE Res CHAR;     
   CASE argNumber
      WHEN 1 THEN SET Res = 'A';
      WHEN 2 THEN SET Res = 'B';
      ELSE SET Res = 'D';     
   END CASE;     
   RETURN Res;
END;
This here is a shorthand for writing the conditions "ArgNumber = 1" and "ArgNumber = 2".
This syntax is only useful when we are sure that we will only use equality-based conditions.  

SELECT funcCASE( 2 );
 

On the other hand, when we use "Search CASE" then we can use whatever condition we want.

Nested CASE and Default Result of the CASE statement

CREATE OR REPLACE FUNCTION funcCASE( argNumber INTEGER )
RETURNS CHAR
BEGIN
   DECLARE Res CHAR;
   CASE
      WHEN argNumber = 1 THEN SET Res = 'A';

      CASE
   
      WHEN argNumber = 2 THEN SET Res = 'B';
   
   END CASE;
   END CASE;
   RETURN Res;
END;
We can nest CASE structures, one inside of another.

Notice that this time we don't have ELSE substatement. In that case
every value of the argument, except 1 and 2, will cause the function to return NULL.

SELECT funcCASE( 99 );

Leave a Comment

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