Excerpts a "mastoring oracle SQL" - PLSQL CALLING Stored Functions from queries

xiaoxiao2021-03-05  29

Calling Stored Functions from queries

As mentioned earlier, stored functions may be called from within SQL statements. Since stored functions can in turn make calls to stored procedures, it can also be said that stored procedures may be called, albeit indirectly, from within SQL statements. Since stored functions may BE Used in Expressions, They May BE Included Wherever Expressions Are Allowed In A Query, Including:

The SELECT clauseThe WHERE clauseThe GROUP BY and HAVING clausesThe ORDER BY clauseThe START WITH / CONNECT BY clauses (for hierarchical queries) The FROM clause (indirectly by using inline views or TABLE statements)

One of the most common uses of stored functions is to isolate commonly-used functionality to facilitate code reuse and simplify maintenance. For example, imagine that you are working with a large team to build a custom N-tier application. To simplify integration efforts between THE VARIOUS LAYERS, IT HAS BEEN DECIDED THATED DATES WILL BE Passed Back and Forth As The Number of MilliseConds Since January 1, 1970. You Could Of Your Queries, AS in:

Select Co.Order_nbr, Co., Co.cust_nbr, Co.Sale_price, Round ((Co.RDER_DT - TO_DATE ('01011970') * 86400 * 1000) * 86400 * 1000) from Cust_Order Cowhere Ship_dt = Trunc (sysdate);

However, this would become somewhat tedious and prove problematic should you wish to modify your logic in the future Instead, build a utility package that includes functions for translating between Oracle's internal date format and the desired format.:

CREATE OR REPLACE PACKAGE BODY pkg_util AS FUNCTION translate_date (dt IN DATE) RETURN NUMBER IS BEGIN RETURN ROUND ((dt - TO_DATE ( '01011970', 'MMDDYYYY')) * 86400 * 1000); END translate_date; FUNCTION translate_date (dt IN NUMBER RETURN DATE IS Begin Return TO_DATE ('01011970', 'MMDDYYYY') (DT / (86400 * 1000)); End Translate_Date; End Pkg_UTIL; if you think you're seeing Double, Don't Worry; The Packs two identically-named functions, one that requires a DATE parameter and returns a NUMBER, and another that requires a NUMBER parameter and returns a DATE. This strategy, called overloading, is only possible when your functions are contained in a package.

Your Development Team Can Now Use these functions WHENE NEY NEED TO Convert Date Formats, AS in:

Select Co.Order_nbr, co.cust_nbr, co.sale_price, pkg_util.translate_date (Co.RDER_DT) UTC_ORDER_DTFROM CUST_ORDER COWHERE CO.SHIP_DT = Trunc (sysdate);

Another common use of stored functions is to simplify and hide complex IF-THEN-ELSE logic from your SQL statements. Suppose you have to generate a report detailing all customer orders for the past month. You want to sort the orders using the ship_dt column if an order has been shipped, the expected_ship_dt column if a ship date has been assigned and is not in the past, the current day if the expected_ship_dt is in the past, or the order_dt column if the order has not been assigned a ship date. You Could Utilize a Case Statement in The Order by Clause:

SELECT co.order_nbr, co.cust_nbr, co.sale_priceFROM cust_order coWHERE co.order_dt> TRUNC (SYSDATE, 'MONTH') AND co.cancelled_dt IS NULLORDER BY CASE WHEN co.ship_dt IS NOT NULL THEN co.ship_dt WHEN co.expected_ship_dt IS NOT NULL AND co.expected_ship_dt> SYSDATE THEN co.expected_ship_dt WHEN co.expected_ship_dt IS NOT NULL THEN GREATEST (SYSDATE, co.expected_ship_dt) ELSE co.order_dt END; However, there are two problems with this approach:

THE RESULTINGORY BY CLAUSE IS FAIRLY COMPLEX.YOU MAY TO USE This Logic Elsewhere, And Duplicating It Will Create Maintenance Problems.

INSTEAD, Add A Stored Function To The Pkg_UTIL Package That Returns The Appropriate Date for a Given Order:

FUNCTION get_best_order_date (ord_dt IN DATE, exp_ship_dt IN DATE, ship_dt IN DATE) RETURN DATE ISBEGIN IF ship_dt IS NOT NULL THEN RETURN ship_dt; ELSIF exp_ship_dt IS NOT NULL AND exp_ship_dt> SYSDATE THEN RETURN exp_ship_dt; ELSIF exp_ship_dt IS NOT NULL THEN RETURN SYSDATE; ELSE Return ORD_DT; END IF; END GET_BEST_ORDER_DATE;

You May THEN CALL THIS FUNCTION from Both the Select and ORDER BY CLAUSES:

SELECT co.order_nbr, co.cust_nbr, co.sale_price, pkg_util.get_best_order_date (co.order_dt, co.expected_ship_dt, co.ship_dt) best_dateFROM cust_order coWHERE co.order_dt> TRUNC (SYSDATE, 'MONTH') AND co.cancelled_dt IS NULLORDER BY PKG_UTIL.GET_BEST_ORDER_DATE (Co.RDER_DT, Co.Expected_ship_dt, co.ship_dt);

If you are bothered by the fact this the store with the same Parameters, you can always retrieve the data with, an inline view and sort the results offward, AS in:

SELECT orders.order_nbr, orders.cust_nbr, orders.sale_price, orders.best_dateFROM (SELECT co.order_nbr order_nbr, co.cust_nbr cust_nbr, co.sale_price sale_price, pkg_util.get_best_order_date (co.order_dt, co.expected_ship_dt, co.ship_dt) best_date From cofust_dt> trunc (sysdate, 'month') and co.cancelled_dt is null) Ordersorder by Orders.best_date; 1 Stored Functions And Views

Since a view is nothing more than a stored query and stored functions can be called from the SELECT clause of a query, columns of a view can map to stored function calls. This is an excellent way to shield your user community from complexity, and it Has Another Interesting Benefit As Well. Consider The Following View Definition, Which Includes Calls to Several Different Stored Functions:

Create Or Replace View VW_EXAMPLE (COL1, COL2, COL3, COL4, COL5, COL6, COL7, COL8) AS SELECT T1.COL1, T1.COL2, T2.COL3, T2.COL4, PKG_EXAMPLE.FUNC1 (T1.COL1, T2. Col3), pkg_example.func2 (t1.col2, t2.col4), pkg_example.func3 (t1.col1, t2.col3), pkg_example.func4 (t1.col2, t2.col4) from Tab1 T1 Inner Join Tab2 T2 on T1 .col1 = t2.col3;

Whereas the first four columns of the view map to columns of the tab1 and tab2 tables, values ​​for the remaining columns are generated by calling various functions in the pkg_example package If one of your users executes the following query.:

Select col2, col4, col7from vw_examplewhere col1 = 1001;

Only one stored function (pkg_example.func3) is actually executed even though the view contains four columns that map to stored function calls. This is because when a query is executed against a view, the Oracle server constructs a new query by combining the original query And the view definition. in this case = t2.col3where t1.col1 = 1001;

Therefore, your View Could Contain Dozens of Stored Function Calls, But Only Those That Explicitly Reference by Queries Will Be Executed. [3]

[3] This is one entry you will.com. Always Explicitly Name The Columns That You NEVER GENERATING DATA That You Never Reference.

2 Avoiding Table Join

Imagine that you have deployed a set of views for your users to generate reports and ad-hoc queries against, and one of your users asks that a new column be added to one of the views. The column is from a table not yet included in the FROM clause, and the column is only needed for a single report issued once a month. You could add the table to the FROM clause, add the column to the SELECT clause, and add the join conditions to the ON clause. However, every .

An alternative strategy is to write a stored function that queries the new table and returns the desired column. The stored function can then be added to the SELECT clause without the need to add the new table to the FROM clause. To illustrate, let's expand on the previous simple example If the desired column is col6 in the tab3 table, you could add a new function to the pkg_example package such as:. fUNCTION func5 (param1 iN NUMBER) RETURN VARCHAR2 IS ret_val VARCHAR2 (20); BEGIN SELECT col6 INTO ret_val From Tab3 WHERE COL5 = param1; return RET_VAL; EXCEPTION WHEN NO_DATA_FOUND THEN RETURN NULL; END FUNC5;

You can now add a column to the view That Maps to the New function, as in:

Create Or Replace View VW_EXAMPLE (Col1, Col2, Col3, Col4, Col5, Col6, Col7, Col8, Col9) AS SELECT T1.COL1, T1.COL2, T2.COL3, T2.COL4, PKG_EXAMPLE.FUNC1 (T1.COL1, T2.col3), pkg_example.func2 (t1.col2, t2.col4), pkg_example.func3 (t1.col1, t2.col3), pkg_example.func4 (t1.col2, t2.col4), pkg_example.func5 (T2. COL3) from Tab1 T1 Inner Join Tab2 T2 on T1.col1 = t2.col3;

Thus, you have provided your users access to column col6 of the tab3 table without adding the tab3 table to the view's FROM clause. Users who do not reference the new col9 column of the view will experience no changes to the performance of their queries against Vw_example.

Even though the column was originally targeted for a single report, do not be surprised if other users decide to include the new column in their queries. As the column utilization increases, it may be advantageous to abandon the stored function strategy and include the tab3 Table in the from clause. Since a View Was Employed, HoWever, You WOULD Be Able To Make This Change Without The need to any of your users to modify their queries.3 deterministic functions

Earlier in this section, we created a package containing two functions to be used for translating between a date and the number of milliseconds since January 1, 1970. Because these functions do not depend on data stored in the database or in package variables, they will Always Return The Same Result for Any Given Input Parameter. ANY Function Having this Property Can Be Marked As Deterministic in The Function Declaration, AS in:

Create Or Replace Package Body PKG_UTIL AS Function Translate_date (DT IN DATE) RETURN NUMBER DETERMINISTIC; FUNCTION TRANSLATE_DATE (DT IN Number) Return Date Deterministic; End Pkg_Util;

Marking your functions as DETERMINISTIC allows the Oracle server to perform certain optimizations, such as storing a function's parameters and results in memory so that subsequent calls to the same function can be handled without the need to call the function again.

转载请注明原文地址:https://www.9cbs.com/read-35951.html

New Post(0)