This article will help you to understand how to create a user defined function. It’s also known as stored function or user function.
- User defined functions are similar to procedures. The only difference is that function always returns a value.
- User defined functions can be used as a part of an SQL expression.
Note
Oracle SQL does not support calling of functions with Boolean parameters or returns.
1. Function – Get formatted address
In this example, we will create a function to get formatted address by giving the person’s name.
1.1 Create tables and function.
-- creating table person_info CREATE TABLE person_info ( PERSON_ID number(5) primary key, FIRST_NAME varchar2(20), LAST_NAME varchar2(20) );
--creating table person_address_details CREATE TABLE person_address_details ( PERSON_ADDRESS_ID number(5) primary key, PERSON_ID number(5) references person_info(person_id), CITY varchar2(15), STATE varchar2(15), COUNTRY varchar2(20), ZIP_CODE varchar2(10) );
--creating function get_complete_address
create or replace FUNCTION get_complete_address(in_person_id IN NUMBER)
RETURN VARCHAR2
IS person_details VARCHAR2(130);
BEGIN
SELECT 'Name-'||person.first_name||' '|| person.last_name||',
City-'|| address.city ||', State-'||address.state||',
Country-'||address.country||', ZIP Code-'||address.zip_code
INTO person_details
FROM person_info person, person_address_details address
WHERE person.person_id = in_person_id
AND address.person_id = person.person_id;
RETURN(person_details);
END get_complete_address;
1.2 Insert data for testing.
INSERT INTO person_info VALUES (10,'Luis','Thomas'); INSERT INTO person_info VALUES (20,'Wang','Moris'); INSERT INTO person_address_details VALUES (101,10,'Vegas','Nevada','US','88901'); INSERT INTO person_address_details VALUES (102,20,'Carson','Nevada','US','90220');
1.3 Display the data.
select * from PERSON_INFO;
| PERSON_ID | FIRST_NAME | LAST_NAME |
|---|---|---|
| 10 | Luis | Thomas |
| 20 | Wang | Moris |
select * from PERSON_ADDRESS_DETAILS;
| PERSON_ADDRESS_ID | PERSON_ID | CITY | STATE | COUNTRY | ZIP_CODE |
|---|---|---|---|---|---|
| 101 | 10 | Vegas | Nevada | US | 88901 |
| 102 | 20 | Carson | Nevada | US | 90220 |
1.4 Calling the function. We can call function many ways. Here first we will call it in SELECT statement. And then we will call it from dbms_output.put_line
SELECT get_complete_address(10) AS "Person Address" FROM DUAL; -- output -- Name-Luis Thomas, City-Vegas, State-Nevada, Country-US, ZIP Code-88901
SET SERVEROUTPUT ON; EXECUTE dbms_output.put_line(get_complete_address(20)); -- output -- Name-Wang Moris, City-Carson, State-Nevada, Country-US, ZIP Code-90220
2. Function – Check Palindrome String
In this example, we will create a function to check whether a given string is palindrome or not.
A palindrome is a word, phrase, number, or other sequence of characters which reads the same backward as forward, such as madam or racecar.
https://en.wikipedia.org/wiki/Palindrome
2.1 Creating the function.
CREATE OR REPLACE FUNCTION checkForPalindrome(inputString VARCHAR2)
RETURN VARCHAR2
IS result VARCHAR2(75);
reversedString VARCHAR2(50);
BEGIN
SELECT REVERSE(inputString) INTO reversedString FROM DUAL;
-- Using UPPER to ignore case sensitivity.
IF UPPER(inputString) = UPPER(reversedString)
THEN
RETURN(inputString||' IS a palindrome.');
END IF;
RETURN (inputString||' IS NOT a palindrome.');
END checkForPalindrome;
/
2.2 Calling the function.
SELECT checkForPalindrome('COMPUTER') FROM DUAL;
-- Output
-- COMPUTER IS NOT a palindrome.
SELECT checkForPalindrome('MAdam') FROM DUAL;
-- Output
-- MAdam IS a palindrome.
SELECT checkForPalindrome('KANAK') FROM DUAL;
-- Output
-- KANAK IS a palindrome.
3. Function – Calculate income tax
In this example, we will create a function to calculate income tax, assumed tax rate is 30% of all annual income from salary.
3.1 Create tables and function.
--creating table person CREATE TABLE person ( PERSON_ID number(5) primary key, FULLNAME varchar2(20) );
--creating table person_salary_details CREATE TABLE person_salary_details ( SALARY_ID number(5) primary key, PERSON_ID number(5) references person(person_id), SALARY number(8), MONTH varchar2(9), YEAR number(4) );
--creating function
CREATE OR REPLACE FUNCTION calculate_tax(personId NUMBER)
RETURN NUMBER
IS tax NUMBER(10,2);
BEGIN
tax := 0;
SELECT (sum(salary)*30)/100 INTO tax FROM person_salary_details WHERE person_id = personId;
RETURN tax;
END calculate_tax;
3.2 Insert data for testing.
INSERT INTO person VALUES (101,'Mark Phile'); INSERT INTO person_salary_details VALUES (1,101,15000,'JANUARY',2016); INSERT INTO person_salary_details VALUES (2,101,15000,'FEBRUARY',2016); INSERT INTO person_salary_details VALUES (3,101,15000,'MARCH',2016); INSERT INTO person_salary_details VALUES (4,101,18000,'APRIL',2016); INSERT INTO person_salary_details VALUES (5,101,18000,'MAY',2016); INSERT INTO person_salary_details VALUES (6,101,18000,'JUNE',2016); INSERT INTO person_salary_details VALUES (7,101,18000,'JULY',2016); INSERT INTO person_salary_details VALUES (8,101,18000,'AUGUST',2016); INSERT INTO person_salary_details VALUES (9,101,18000,'SEPTEMBER',2016); INSERT INTO person_salary_details VALUES (10,101,18000,'OCTOBER',2016); INSERT INTO person_salary_details VALUES (11,101,18000,'NOVEMBER',2016); INSERT INTO person_salary_details VALUES (12,101,18000,'DECEMBER',2016);
3.3 Display the data.
select * from PERSON;
| PERSON_ID | FULLNAME |
|---|---|
| 101 | Mark Phile |
select * from PERSON_SALARY_DETAILS;
| SALARY_ID | PERSON_ID | SALARY | MONTH | YEAR |
|---|---|---|---|---|
| 1 | 101 | 15000 | JANUARY | 2016 |
| 2 | 101 | 15000 | FEBRUARY | 2016 |
| 3 | 101 | 15000 | MARCH | 2016 |
| 4 | 101 | 18000 | APRIL | 2016 |
| 5 | 101 | 18000 | MAY | 2016 |
| 6 | 101 | 18000 | JUNE | 2016 |
| 7 | 101 | 18000 | JULY | 2016 |
| 8 | 101 | 18000 | AUGUST | 2016 |
| 9 | 101 | 18000 | SEPTEMBER | 2016 |
| 10 | 101 | 18000 | OCTOBER | 2016 |
| 11 | 101 | 18000 | NOVEMBER | 2016 |
| 12 | 101 | 18000 | DECEMBER | 2016 |
3.4 Calling the function.
SELECT person.fullname, sum(sal.salary) AS AnnualSalary, sal.year,calculate_tax(101) AS tax FROM person,person_salary_details sal WHERE person.person_id = 101 and sal.year = 2016 GROUP BY person.fullname, sal.year;
Output
| FULLNAME | ANNUALSALARY | YEAR | TAX |
|---|---|---|---|
| Mark Phile | 207000 | 2016 | 62100 |