• 售前

  • 售后

热门帖子
入门百科

postgresql 实现得到时间对应周的周一案例

[复制链接]
此人正在学习 显示全部楼层 发表于 2021-10-26 13:56:14 |阅读模式 打印 上一主题 下一主题
两种方法:

第一种:
  1. DO $$
  2. declare d int;
  3. declare d1 varchar(100);
  4. declare d2 varchar(100);
  5. declare d3 date;
  6. declare d4 date;
  7. begin
  8. d3:= CURRENT_DATE;
  9. d1:='select date'''|| d3 ||'''';
  10. d:=(SELECT EXTRACT(DOW FROM d3))-1;
  11. d2:=d1 || '-INTERVAL ''' || d || ' day '' ';
  12. EXECUTE d2 into d4;
  13. RAISE NOTICE 'ok %',d4;
  14. end$$
复制代码
结果:
  1. [SQL]DO $$
  2. declare d int;
  3. declare d1 varchar(100);
  4. declare d2 varchar(100);
  5. declare d3 date;
  6. declare d4 date;
  7. begin
  8. d3:= CURRENT_DATE;
  9. d1:='select date'''|| d3 ||'''';
  10. d:=(SELECT EXTRACT(DOW FROM d3))-1;
  11. d2:=d1 || '-INTERVAL ''' || d || ' day '' ';
  12. EXECUTE d2 into d4;
  13. RAISE NOTICE 'ok %',d4;
  14. end$$NOTICE: ok 2016-06-13时间: 0.004s受影响的行: 0
复制代码
解析:
declare :声明变量
CURRENT_DATE : 得到当前日期
SELECT CURRENT_DATE;
结果:
  1. date
  2. 2016-06-12
复制代码
extract :从时间中抽出相应的字段
DOW 一周里的第几天 (sunday =0 saturday=6)
格式:
  1. EXTRACT(field FROM source)
复制代码
当前日期是一周内里的第几天
  1. SELECT EXTRACT(DOW FROM CURRENT_DATE);
复制代码
结果:
  1. date_part
  2.     0
复制代码
INTERVAL :时间隔断类型
EXECUTE :实行一个准备好的查询
RAISE NOTICE :把结果显示出来
第二种:
  1. SELECT CURRENT_DATE +cast(-1*(TO_NUMBER(to_char(CURRENT_DATE,'D'),'99')-2) ||' days' as interval);
复制代码
结果:
  1. ?column?
  2. 2016-06-13 00:00:00
复制代码
解析:
TO_NUMBER 将一个字符串转换成数字
格式:
  1. TO_NUMBER(string,format)
复制代码
  1. -- 一周里的日子(1-7;周日是1)
  2. select to_char(CURRENT_DATE ,'D')
  3. DDD 一年里的日子(001-366)
  4. DD 一个月里的日子(01-31)
  5. D  一周里的日子(1-7;周日是1)
  6. select to_char (to_date('2016-06-12','yyyy-mm-dd'),'D')
  7. select to_number(‘1.1','9.99') from dual;
  8. 1.1
  9. select to_number(‘1.121','9.99') from dual;
  10. 1.12
  11. -- 将得到的字符串转换成数字
  12. select TO_NUMBER(to_char(CURRENT_DATE,'D'),'99')
  13. -- 因为得到的星期一为2,所以要减去2
  14. select TO_NUMBER(to_char(CURRENT_DATE,'D'),'99')-2
  15. -- 将得到的数字乘以 -1 比如例子中:-1*3 就是 -3 ,也就是减去 3天
  16. select cast(-1*3 || 'days' as interval)
  17. -- 就是将当天减去0天 得到了星期一的日期
  18. select cast(-1*0 || 'days' as interval) + CURRENT_DATE
  19. SELECT  to_char(CURRENT_DATE +cast(-1*(TO_NUMBER(to_char(CURRENT_DATE,'D'),'99')-2) ||' days' as interval),'yyyy-mm-dd');
复制代码
补充:Postgresql数据数据库中按日、月、周、年、时、分,30分钟的统计解决方案
对要统计的时间字段进行字符转换处置处罚,再按照其分组即可实现对数据进行日,周,月,年,时,分,秒的统计
1、按日统计
  1. to_char( h.row_date, 'yyyy-MM-dd' ) AS row_date2
  2. GROUP BY to_char( h.row_date, 'yyyy-MM-dd' )
复制代码
2、按月统计
  1. to_char(h.row_date, 'yyyy-MM' ) AS row_date2
  2. GROUP BY to_char(h.row_date, 'yyyy-MM' )
复制代码
3、按年统计
  1. to_char( h.row_date,'yyyy' ) AS row_date2
  2. GROUP BY to_char( h.row_date,'yyyy' )
复制代码
4、按小时统计
  1. to_char( h.row_date, 'yyyy-MM-dd HH' ) AS row_date2
  2. GROUP BY to_char( h.row_date, 'yyyy-MM-dd HH' )
复制代码
5、按分钟统计
  1. to_char( h.row_date, 'yyyy-MM-dd HH:mm' ) AS row_date2
  2. GROUP BY to_char( h.row_date, 'yyyy-MM-dd HH:mm' )
复制代码
6、按周统计
按周统计最简单法
对时间row_date字段做处置处罚,变成对应日期周一时间,然后按这个周一的时间去统计。减1的操作表示为对应日期的星期一,减1,2,3,4,5,6,7分别是对应日期的周一,周二,周三,周四,周五、周六、周日。
  1. to_char( h.row_date-(extract (dow from h.row_date) - 1 ||'day')::interval,'yyyy-MM-dd') row_date
复制代码
然后按上面的语句分组统计即可实现按周统计,下面临应分组函数
  1. GROUP BY to_char(h.row_date-(extract (dow from h.row_date) - 1 ||'day')::interval,'yyyy-MM-dd')
复制代码
按周统计之方法二(较复杂,不建议利用)
  1. to_char(h.row_date, 'yyyy' ) || EXTRACT ( week FROM h.row_date ) :: INTEGER ASrow_date2
复制代码
获取到数据库输出的字段中的年份和周数。
  1. String row_date=rs.getString("row_date2");
  2. //获取数据库输出日期的年份
  3. int year=Integer.parseInt(row_date.substring(0, 4));
  4. //获取数据库输出日期的周数
  5.   if(row_date.length()>=6){
  6.          week=Integer.parseInt(row_date.substring(4,6));}
  7.   else{
  8.         week=Integer.parseInt(row_date.substring(4,5));
  9.       }
  10.       String row_date2=getFirstDayOfWeek(year, week);
  11.       trafficMap.put("row_date", row_date2);
复制代码
将查询出的内容日期转换成当周周一的时间
  1. //将周统计中获取的如201636,表示2016年36周,获取其周一的时间
  2.   public String getFirstDayOfWeek(int year, int week) {
  3.     // 先滚动到该年
  4.      nows.set(Calendar.YEAR, year);
  5.     // 滚动到周
  6.      nows.set(Calendar.WEEK_OF_YEAR, week);
  7.     // 得到该周第一天
  8.      nows.set(Calendar.DAY_OF_WEEK, 2);
  9.     String firstDay = df.format(nows.getTime());
  10.     return firstDay;
  11.     }
复制代码
7、按30分钟进行统计
  1. case when substr( to_char(h.row_date, 'yyyy-mm-dd hh24:mi'),15, 16) :: integer <=30 then to_char(h.row_date, 'yyyy-mm-dd hh24')||':30' else to_char( h.row_date, 'yyyy-mm-ddhh24' )||':60' end as row_date2
  2. GROUP BY  case when substr( to_char(h.row_date, 'yyyy-mm-dd hh24:mi'),15, 16) :: integer <=30 then to_char(h.row_date, 'yyyy-mm-dd hh24')||':30' else to_char( h.row_date, 'yyyy-mm-ddhh24' )||':60' end
复制代码
以上为个人履历,希望能给各人一个参考,也希望各人多多支持草根技能分享。如有错误或未思量完全的地方,望不吝赐教。

帖子地址: 

回复

使用道具 举报

分享
推广
火星云矿 | 预约S19Pro,享500抵1000!
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

草根技术分享(草根吧)是全球知名中文IT技术交流平台,创建于2021年,包含原创博客、精品问答、职业培训、技术社区、资源下载等产品服务,提供原创、优质、完整内容的专业IT技术开发社区。
  • 官方手机版

  • 微信公众号

  • 商务合作