各位好:
遇到一个比较棘手的取间隔时间问题.要写一个存储过程,传入三个参数,开始时间和结束时间(YYYY-MM-DD格式的字符串),还有一个是选择间隔(周/月/季度/半年/年).若选择为周,那么要得到起止时间内的每周一和周日的具体日期写入一张表中.举例如下:
开始时间为2007-08-01,结束时间为2007-08-31,间隔为周.那么我要得到如下数据.
2007-08-06 2007-08-12
2007-08-13 2007-08-19
2007-08-20 2007-08-26若选择间隔为月,那么要得到起止时间内的每月1日至每月末.
如开始时间为2007-06-12,结束时间为2007-10-20,那么得到如下数据
2007-07-01 2007-07-31
2007-08-01 2007-08-31
2007-09-01 2007-09-30若选择间隔为季度,那么就要得到起止时间内的每季度的第一天和最后一天.
举上面的例子来说.那么要得到的就是:
2007-07-01 2007-09-30以此类推,若间隔为半年就是取起止时间内的半年头和半年尾,若间隔为年,就是起止时间内的年头和年尾啦.问题比较难,若解决,分不是问题.望赐教谢谢.
遇到一个比较棘手的取间隔时间问题.要写一个存储过程,传入三个参数,开始时间和结束时间(YYYY-MM-DD格式的字符串),还有一个是选择间隔(周/月/季度/半年/年).若选择为周,那么要得到起止时间内的每周一和周日的具体日期写入一张表中.举例如下:
开始时间为2007-08-01,结束时间为2007-08-31,间隔为周.那么我要得到如下数据.
2007-08-06 2007-08-12
2007-08-13 2007-08-19
2007-08-20 2007-08-26若选择间隔为月,那么要得到起止时间内的每月1日至每月末.
如开始时间为2007-06-12,结束时间为2007-10-20,那么得到如下数据
2007-07-01 2007-07-31
2007-08-01 2007-08-31
2007-09-01 2007-09-30若选择间隔为季度,那么就要得到起止时间内的每季度的第一天和最后一天.
举上面的例子来说.那么要得到的就是:
2007-07-01 2007-09-30以此类推,若间隔为半年就是取起止时间内的半年头和半年尾,若间隔为年,就是起止时间内的年头和年尾啦.问题比较难,若解决,分不是问题.望赐教谢谢.
解决方案 »
- WINDOWS7 能不能安装 oracle BI publisher
- ora 12154 无法处理服务名 问题
- 《ORACLE9i_优化设计与系统调整》DOC电子书
- 除了用TRIGGER监控DML,还有没有其它方法监控表的改动??牛人大哥些,救救小弟,感激不尽~~~~
- oracle 字符集问题 急!
- 问:如何将文件存在数据库中,怎么存储,读取?
- 表自身的连接!要求两行合一行。
- 一个问题 ,全部的分相送!!!
- oracle 11g 在不区分大小写的情况下,求出邮件地址中包含姓的员工信息。其中邮件以首字母大写的形式返回 怎么弄?
- cheak语句哪儿错了?
- 为什么安装不上oracle provider ole db?
- 多 表 更 新
结合Oracle的函数,简单的说一下思路。
周:
next_day(开始日期,'星期一') 返回开始时间之后的第一个星期一。后面的就循环加7吧。
计算出每个周的开始日和结束日。插入数据库。
知道一个周的开始日大于结束时间。
月:
使用last_day(开始时间)+1获得下月的1号,然后再使用last_day(add_months(开始时间,1))获得下个月的结束时间。然后再辅助以一些循环变量控制循环即可。
半年:
其实和月很相近只不过add_months的时候从加1变成加7
年:
不用借鉴半年了,直接分析出是哪一年开始到哪一年结束。
to_number(trim(to_char(开始时间,'YYYY')))+1 即可得知下一个年头是啥。
后面就循环插入数据吧。
(
beg_date varchar2(10),
end_date varchar2(10)
)
;create or replace procedure test
(
date_beg in varchar2,
date_end in varchar2,
duration_type in varchar2
)
as
beg_date date;
end_date date;
beg_month date;
end_month date;
begin
if date_beg>date_end then
dbms_output.put_line('beginning date must be earlier than end date');
else
beg_date := to_date(date_beg,'yyyy-mm-dd');
end_date := to_date(date_end,'yyyy-mm-dd');
if duration_type='week' then
while beg_date+6<=end_date loop
if to_char(beg_date,'D')='2' then
while beg_date+6<=end_date loop
insert into test2 values(to_char(beg_date,'yyyy-mm-dd'),to_char(beg_date+6,'yyyy-mm-dd'));
beg_date := beg_date+7;
end loop;
end if;
beg_date := beg_date+1;
end loop;
else
if to_char(beg_date,'dd')='01' then
beg_month := beg_date;
else
beg_month := to_date(to_char(add_months(beg_date,1),'yyyy-mm')||'-01','yyyy-mm-dd');
end if;
if to_char(end_date,'yyyy-mm-dd')=to_char(last_day(end_date),'yyyy-mm-dd') then
end_month := end_date;
else
end_month := to_date(to_char(last_day(add_months(end_date,-1)),'yyyy-mm-dd'),'yyyy-mm-dd');
end if;
if duration_type='month' then
while to_char(beg_month,'yyyy-mm')<=to_char(end_month,'yyyy-mm') loop
insert into test2 values (to_char(beg_month,'yyyy-mm-dd'),to_char(last_day(beg_month),'yyyy-mm-dd'));
beg_month := add_months(beg_month,1);
end loop;
elsif duration_type='quarter' then
while to_char(add_months(beg_month,2),'yyyy-mm')<=to_char(end_month,'yyyy-mm') loop
if to_char(beg_month,'mmdd') in ('0101','0401','0701','1001') then
while to_char(add_months(beg_month,2),'yyyy-mm')<=to_char(end_month,'yyyy-mm') loop
insert into test2 values (to_char(beg_month,'yyyy-mm-dd'),to_char(last_day(add_months(beg_month,2)),'yyyy-mm-dd'));
beg_month := add_months(beg_month,3);
end loop;
end if;
beg_month := add_months(beg_month,1);
end loop;
elsif duration_type='semiyear' then
while to_char(add_months(beg_month,5),'yyyy-mm')<=to_char(end_month,'yyyy-mm') loop
if to_char(beg_month,'mmdd') in ('0101','0701') then
while to_char(add_months(beg_month,5),'yyyy-mm')<=to_char(end_month,'yyyy-mm') loop
insert into test2 values (to_char(beg_month,'yyyy-mm-dd'),to_char(last_day(add_months(beg_month,5)),'yyyy-mm-dd'));
beg_month := add_months(beg_month,6);
end loop;
end if;
beg_month := add_months(beg_month,1);
end loop;
end if;
end if;
end if;
commit;
end;
/输入参数必须是如下格式
exec test('2007-08-01','2007-08-31','week');
exec test('2007-08-01','2007-10-01','month');
exec test('2007-06-01','2007-12-31','quarter');
exec test('2007-06-01','2009-11-30','semiyear');SQL> select * from test2;no rows selectedSQL> exec test('2007-08-01','2007-08-31','week');PL/SQL procedure successfully completed.SQL> select * from test2;BEG_DATE END_DATE
---------- ----------
2007-08-06 2007-08-12
2007-08-13 2007-08-19
2007-08-20 2007-08-26SQL>
next_day(开始日期,'星期一') 返回开始时间之后的第一个星期一。后面的就循环加7吧。
计算出每个周的开始日和结束日。插入数据库。
知道一个周的开始日大于结束时间。
月:
使用last_day(开始时间)+1获得下月的1号,然后再使用last_day(add_months(开始时间,1))获得下个月的结束时间。然后再辅助以一些循环变量控制循环即可。
半年:
其实和月很相近只不过add_months的时候从加1变成加7
年:
不用借鉴半年了,直接分析出是哪一年开始到哪一年结束。
to_number(trim(to_char(开始时间,'YYYY')))+1 即可得知下一个年头是什么。