不使用日期函式實現ADD_MONTHS函式功能
看到開發區的nyfor版主出了一道題,用PL/SQL實現Oracle的ADD_MONTHS的功能,覺得比較有意思,忍不住嘗試了一下。
原文如下:http://www.itpub.net/thread-977079-1-1.html
要求不能使用任何日期函式,不能使用SQL,不能使用Oracle提供的標準包以外的包,只透過標準函式來實現這個功能,且要求原始碼儘量短。
經過多次嘗試和努力,將字元控制到了420之內,下面貼出這個函式的實現:
SQL> CREATE OR REPLACE FUNCTION MY_ADD_MONTHS(P_DATE_STRING VARCHAR2, P_MONTHS NUMBER)
2 RETURN VARCHAR2 AS
3 SUBTYPE I IS INT;
4 N I:=P_DATE_STRING;
5 C I:=100;
6 D I:=MOD(N,C);
7 Y I:=N/C/C;
8 M I:=MOD(N/C,C);
9 Z I:=M+P_MONTHS;
10 FUNCTION F(Y I,M I) RETURN I
11 AS
12 BEGIN
13 RETURN 31-CASE WHEN M IN(4,6,9,11) THEN 1
14 WHEN M!=2 THEN 0
15 WHEN MOD(Y,4*C)=0 OR MOD(Y,4)=0 AND MOD(Y,C)>0 THEN 2
16 ELSE 3
17 END;
18 END;
19 BEGIN
20 N:=MOD(MOD(Z,12)+11,12)+1;
21 Z:=Y+(Z-N)/12;
22 IF D=F(Y,M) OR D>F(Z,N) THEN
23 D:=F(Z,N);
24 END IF;
25 RETURN Z*C*C+N*C+D;
26 END;
27 /
函式已建立。
雖然這個實現不是最短的,但是這個函式中從頭到尾只用了一個函式MOD。
下面利用nyfor版主提供的測試程式碼進行測試:
SQL> SET SERVEROUT ON
SQL> declare
2 ln number;
3 ld1 date;
4 ld2 date;
5 ls varchar2(8);
6 lt number := dbms_utility.get_time;
7 ex exception;
8 i number;
9 j number;
10 begin
11 ld1 := to_date('20080101', 'yyyymmdd');
12 i := 0;
13 loop
14 exit when i > 800;
15 j := -2400;
16 loop
17 exit when j > 2400;
18 if to_char(ld1 + i,'dd')= '02' then
19 i := i + 25;
20 end if;
21 if to_char(add_months(ld1 + i, j), 'yyyymmdd') <>
22 nvl(my_add_months(to_char(ld1 + i, 'yyyymmdd'), j), '*') then
23 dbms_output.put_line('Sorry: stop at p_date_string=' ||
24 to_char(ld1 + i, 'yyyymmdd') || ',p_months=' || j);
25 dbms_output.put_line('my_add_months returned: '||my_add_months(to_char(ld1 + i, 'yyyymmdd'), j) ||
26 ', add_months returned: ' || to_char(add_months(ld1 + i, j), 'yyyymmdd'));
27 raise ex;
28 end if;
29 j := j + 1;
30 end loop;
31 i := i + 1;
32 end loop;
33 ln := 0;
34 for c in (select text
35 from user_source
36 where name = 'MY_ADD_MONTHS'
37 and type = 'FUNCTION') loop
38 ln := ln + nvl(lengthb(translate(c.text,
39 '*' || chr(9) || chr(10) || chr(13) ||
40 chr(32),
41 '*')),
42 0);
43 end loop;
44
45 lt := (dbms_utility.get_time - lt) / 100;
46 dbms_output.put_line('Congratulation ... Code Length: ' || ln ||
47 ' Bytes. Times: ' ||
48 to_char(to_date(to_char(lt, 'fm00000'), 'sssss'),
49 'hh24:mi:ss'));
50 exception
51 when ex then
52 null;
53 end;
54 /
Congratulation ... Code Length: 412 Bytes. Times: 00:00:10
PL/SQL 過程已成功完成。
程式碼本身比較簡單,只不過為了縮短程式碼的數量,使用了一些小技巧。當然這個實現肯定還有很多可以最佳化的地方,這裡希望能起到一個拋磚引玉的作用。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/4227/viewspace-255480/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 不使用日期函式實現ADD_MONTHS函式功能(二)函式
- SQL函式之日期函式SQL函式
- 字元函式、數字函式和日期函式字元函式
- MySQL(四)日期函式 NULL函式 字串函式MySql函式Null字串
- 日期函式函式
- 【函式】Oracle函式系列(2)--數學函式及日期函式函式Oracle
- MYSQL事件使用 日期函式MySql事件函式
- 【Mysql 學習】日期函式函式MySql函式
- (函式)實現strstr函式函式
- MySQL函式大全(字串函式,數學函式,日期函式,系統級函式,聚合函式)MySql函式字串
- Oracle 函式大全(字串函式,數學函式,日期函式,邏輯運算函式,其他函式)Oracle函式字串
- ABAP日期函式函式
- Sybase日期函式函式
- sql 日期函式SQL函式
- SqlServer中使用CONVERT函式實現日期格式化SQLServer函式
- Oracle OCP(03):字元函式、數字函式和日期函式Oracle字元函式
- ORACLE單行函式與多行函式之四:日期函式示例Oracle函式
- [PY3]——函式——函式註解 | 實現型別檢查功能函式型別
- 利用指標實現strncmp函式功能指標函式
- 有關日期的SAP函式使用函式
- SQLSERVER日期函式(zt)SQLServer函式
- 日期時間函式函式
- 日期提取函式EXTRACT函式
- oracle日期函式(1)Oracle函式
- oracle日期函式(2)Oracle函式
- oracle日期函式(3)Oracle函式
- oracle日期函式(4)Oracle函式
- oracle日期函式(5)Oracle函式
- Oracle的日期函式Oracle函式
- Sql Server 日期函式SQLServer函式
- oracle 10g函式大全–日期型函式Oracle 10g函式
- oracle 10g函式大全--日期型函式Oracle 10g函式
- mysql 自定義分析函式 least 及 日期函式MySql函式AST
- 使用函式式實現觀察者模式模式函式模式
- 探索MySQL高階語句(數學函式、聚合函式、字串函式、日期時間函式)MySql函式字串
- C#函式實現的小功能集合C#函式
- SQL 10 函式 3 日期時間函式 - 5 計算日期差額SQL函式
- 使用函式式方式實現責任鏈模式函式模式