[DB2] User Define Function – SQL Scalar Example

พอดีมีรุ่นน้องให้สอนการเขียน Function บน DB2 ครับ โดย Function ที่ผมเขียนเป็นแบบ SQL Scalar Function โดยเจ้า SQL Scalar Function เป็น Function ที่คืนค่าผลลัพธ์ มีการคืนผลลัพธ์ออกมาค่าเดียวเสมอ โดยมีรูปแบบข้อมูล เช่น datetime, int, varchar เป็นต้นครับ ตัวอย่าง Scalar Function ที่มีในระบบพวก ABS() ที่ส่งตัวเลข เข้าไป คืนผลลัพธ์ที่ตัวเครื่องหมายแล้ว ออกมาค่าเดียวครับ เช่น ABS(-5) = 5

สำหรับ Function นี้ ผมขอเอาตัวอย่างจริงเลยและกัน(ขี้เกียจคิดครับ ฮ่าๆ) โดย Function นี้มีหน้าที่แปลงรอบการจ่าย COUPON FREQUENCY(รอบการจ่ายดอกเบี้ย) ของ BOND ออกมาเป็น TENOR CODE(ช่วงเวลาครับ เช่น 1 เดือน / 2 เดือน เป็นต้นครับ) เพราะในระบบเดิมมันเก็บ COUPON FREQUENCY ใน Table FIXINCOME ดังนี้ครับ

 ID ในระบบ ความหมาย
1Monthly
2Two Month
3Quarterly
4TRI-ANNUALLY
5SEMI-ANNUALLY
6ANNUALLY
7AT CALL

ซึ่งมันไม่สื่อถึงเวลาครับ โดยเจ้าข้อมูลที่สื่อถึงเวลามันอยู่ใน Table TENOR ดังนี้ครับ โดยมี Field ที่จำเป็น ดังนี้

Field Nameความหมาย
TENOR IDID ของ TENOR
TENOR CODEชื่อ TENOR เช่น 1 Week
START_NUMจำนวนของ TENOR ของเวลาเริ่มต้น โดยต้องตีความร่วมกับ START_UNIT
START_UNIT1 = Days, 2 = Weeks. 3 = Months, 4 = Years
END_NUMจำนวนของ TENOR ของเวลาสิ้นสุด โดยต้องตีความร่วมกับ END_UNIT
END_UNIT1 = Days, 2 = Weeks. 3 = Months, 4 = Years

ตัวอย่างแรก ถ้ามี TENOR ช่วง 1ปี-3 ปี มีการจัดเก็บข้อมูล ดังนี้

COLUMN NAMETENOR IDTENOR CODESTART_NUMSTART_UNITEND_NUMEND_UNIT
VALUE11-3 Years1434

ตัวอย่างสอง ถ้ามี TENOR ช่วง 1 ปี มีการจัดเก็บข้อมูล (โดยข้อมูล TENOR ที่ตรงกัน COUPON FREQUENCY จะถูก Config ไว้ในรูปแบบนี้ครับ) ดังนี้

COLUMN NAMETENOR IDTENOR CODESTART_NUMSTART_UNITEND_NUMEND_UNIT
VALUE11 Years1414

โดยแต่ละ TENOR ถ้ามีข้อมูลเรียงลำดับกันเป็นชุด เช่น 1 Month, 2 Month ผมได้จัดลงใน Table TENOR GROUP และ TENOR GROUP DETAIL (อันนี้ไม่ขอลงในรายละเอียดครับ) หลังจากที่ปูความเข้าใจด้าน Business แล้ว เรามาเข้าเรื่องการเขียน Function กันครับ

เจ้า Function ของผมที่เขียนมีลำดับการทำงานประมาณนี้ครับ

  • แปลง COUPON FREQUENCY มาเป็นหน่วยของ TENOR ก่อนครับ คือ TENOR_NUM และ TENOR_UNIT
  • เอา TENOR_NUM และ TENOR_UNIT เข้าไปหา TENOR CODE ตาม TENOR GROUP ที่เราระบุไว้ครับ

รูปแบบโครงสร้างของ Function ใน DB2 ครับ (ขอสรุปสั้น ตามนี้ครับ เพราะ มันยาวมาก ถ้าอยากศึกษาเพิ่มเติมกดดูได้เลยครับ

CREATE OR REPLACE FUNCTION name-of-udf --------------------(1)
	(List of the input parameters of the UDF) ---------(2)
	Returns -------------------------------------------(3)
	Function properties -------------------------------(4)
	Generation options --------------------------------(5)

การ Config เพิ่มเติมสำหรับ Function ครับ

  • NOT SECURED: บอกให้ตัว DBMS เข้าใจว่า Parameter ของ Function นี้ ไม่ได้อ้างอิงถึง Column ที่ถูกกำหนดสิทธิในส่วน Column Access Control (ถ้า Column ที่ถูกกำหนดสิทธิถูกส่งเข้ามาเป็น Parameter ของ Function ตัว DBMS จะ Raise Exception เกี่ยวกับสิทธิออกมาครับ)
  • LANGUAGE SQL: บอกภาษาที่เราใช้เขียนไส้ในของ Function ครับ โดยถ้าเป็นภาษา C หรือ Java เราสามารถกำหนดให้ไปชี้ Path ที่เก็บ Library เพื่อ Run คำสั่งได้ครับ
  • INHERIT SPECIAL REGISTERS: ให้ Function ของเราเนี่ย ใช้งานตัว SPECIAL REGISTERS ของ DB2 ได้เลยครับ เช่น SESSION TIME ZONE หรือ SQLRULES เป็นต้นครับ

สำหรับพวก BEGIN ATOMIC อ่านได้จากบทความนี้ครับ

อ๋อ และก็ถ้าดูใน QUERY ที่มันต้องมี FETCH FIRST 1 ROWS ONLY เพราะ เนื่องจาก Scalar Function มันมีข้อกำหนดว่าเราต้อ Return ได้ค่าเดียวเท่านั้นครับ

ตัวอย่างของ Function ครับ

[sql]CREATE OR REPLACE FUNCTION INVEST.FINDTENOROFCOUPONFREQUENCY (
COUPONFREQUENCY INTEGER,
TENORGROUPCODE VARCHAR(30) )
RETURNS VARCHAR(30)
NOT SECURED
LANGUAGE SQL
INHERIT SPECIAL REGISTERS
BEGIN ATOMIC
DECLARE TENOR_NUM INTEGER;
DECLARE TENOR_UNIT INTEGER;
IF COUPONFREQUENCY = 1 THEN --1 = Monthly
SET TENOR_NUM = 1;
SET TENOR_UNIT = 3;
ELSEIF COUPONFREQUENCY = 2 THEN --2 = Two Month
SET TENOR_NUM = 2;
SET TENOR_UNIT = 3;
ELSEIF COUPONFREQUENCY = 3 THEN --3 = Quarterly (3 เดือน)
SET TENOR_NUM = 3;
SET TENOR_UNIT = 3;
ELSEIF COUPONFREQUENCY = 4 THEN --4 = TRI-ANNUALLY (4 เดือน)
SET TENOR_NUM = 4;
SET TENOR_UNIT = 3;
ELSEIF COUPONFREQUENCY = 5 THEN --5 = SEMI-ANNUALLY (6 เดือน)
SET TENOR_NUM = 6;
SET TENOR_UNIT = 3;
ELSEIF COUPONFREQUENCY = 6 THEN --6 = ANNUALLY (1 ปี)
SET TENOR_NUM = 12;
SET TENOR_UNIT = 3;
ELSEIF COUPONFREQUENCY = 7 THEN --7 = AT CALL (ครบกำหนด)
SET TENOR_NUM = 999;
SET TENOR_UNIT = 4;
END IF;
RETURN (SELECT TENORCODE
FROM TENORGROUP TG
INNER JOIN TENORGROUPDETAIL TGD
ON TG.TENORGROUPID = TGD.TENORGROUPID
AND TG.CREATETIME = TGD.GROUPCREATETIME
INNER JOIN TENOR T
ON TGD.TENORTYPEID = T.TENORID
AND TGD.TYPECREATETIME = T.CREATETIME
AND T.ACTIVEFLAG = 'A'
WHERE TG.TENORGROUPCODE = TENORGROUPCODE
AND TG.ACTIVEFLAG = 'A'
AND T.STARTNUM = TENOR_NUM
AND T.STARTUNIT = TENOR_UNIT
AND T.ENDNUM = TENOR_NUM
AND T.ENDUNIT = TENOR_UNIT
FETCH FIRST 1 ROWS ONLY);
END;
[/sql]

ตัวอย่างการเรียกใช้งานครับ โดย SYSIBM.SYSDUMMY1 มัน คือ Temp ของ DB2 มองว่าเป็นกระดาษทดครับ เพราะ เวลาเรียกใช้ Function จริง เราต้องเรียกใช้คำสั่ง SQL ที่มีพวก SELECT...FROM แต่ที่นี้ ผมเอาคำสั่ง SQL จริงมาแสดงให้ดูไม่ได้ เพราะมันยาวมากกก 555 อ่านไปเดี๋ยวงงครับ เอาแบบนี้สั้นกระชับดีครับ (สรุปคือให้มองง่ายๆ ว่ามันเรียกใช้เหมือนพวกคำสั่ง SUM(), AVG(), MIN(), MAX() แหละครับ)

[sql]SELECT "INVEST"."FINDTENOROFCOUPONFREQUENCY"(1, 'BIBOR_BOND') FROM SYSIBM.SYSDUMMY1;[/sql]

2015-05-24_150708
Reference


Discover more from naiwaen@DebuggingSoft

Subscribe to get the latest posts to your email.