zoukankan      html  css  js  c++  java
  • Date小技巧:set相关操作及应用_获取当前月(季度/年)的最后一天

    set操作还是有不少的,具体见 http://www.w3school.com.cn/jsref/jsref_obj_date.asp

    今天我就只说 setFullYear, setMonth, setDate,因为今天的应用只涉及到这三个。

    这3个方法顾名思义分别设置年、月、日,之前我所了解到的应用,比如“倒计时”中设置目标时间点的时候会用到,其他的暂时记不起来,回头再补。

    今天来说说“获取当前月(季度/年)的最后一天”的用法

    你可能想到了,每个月的天数是不应定的,有28、29(闰年2月)、30、31,总不能去算一下现在是几月份,然后去arr或者map(json)里边去找对应的日期吧,当然这也是个方法,能达到效果(还要考虑闰年)。

    那么,不这么繁琐的话,怎么搞呢?

    其实,Date本来就会自动处理每月的天数,包括闰年什么的特殊情况,所以,这个我们根本不用去关心。

    这里用到一个用法,估计大家有可能也用过,date.setDate(0)。没错,你没看错,就是0。(喂,这不是month啊,不用加1,你确定不是要用setDate(1) ?)

    0是不存在的一天,date.setDate(0)之后,这一天不存在,或者说设置的是1号的前一天。那么,1号的前一天,自然就是前一个月的最后一天

    1 var date = new Date();
    2 console.log('今天是 ', date.getMonth() + 1, date.getDate());
    3 date.setDate(0);
    4 console.log('上个月最后一天是 ', date.getMonth() + 1, date.getDate());

     控制台跑一下上边代码,就会得到上个月最后一天了。

    继续,我们要得到的是这个月最后一天,怎么破

    先把月份设置到下个月,然后获取这个月最后一天: 

    1 date.setMonth(date.getMonth() + 1);
    2 date.setDate(0);

    date就是这个月最后一天对应的date对象。

    到此,简单地获取当月最后一天已经完了,改变setMonth可以让你获取任何一个月的最后一天了。

    ~~~~~~~~~~~~~~~~~~~~~~~~~   分割线   ~~~~~~~~~~~~~~~~~~~~~~

    下面,来讲获取这一年的最后一天的date。

    同理,我们只要设置到明年就可以了,然后设置月份到0月(就是1月),然后设置到0日,这样就成了2017-01-00,也就是元旦的前一天,当然也就是2016-12-31这天了。

    1 date.setFullYear(date.getFullYear() + 1); // 设置到明年
    2 date.setMonth(0); // 明年的0月,也就是对应到1月,是存在的哦,不是不存在的0
    3 date.setDate(0); // 明年的0日

    ~~~~~~~~~~~~~~~~~~~~~~~~~  又来分割线   ~~~~~~~~~~~~~~~~~~~~~

    然后,又到了获取当前季度的最后一天。

    Date中是没有季度的概念的(如果有,请及时告诉我,让我多个知识点),所以,首先我们要判断当前月在哪个季度

    1 var m = date.getMonth();
    2 var quarter = Math.floor(m / 3) + 1; // 从1开始,到4
    3 var qLastMonth = quarter * 3;

    其中,quarter 是第几季度,qLastMonth 是这个季度的最后一个月是几月 // 从1开始到12,当年可以减去1,从0到11。

    既然拿到了这个季度的最后一个月是几月,那再用上边的获取任一月份的最后天的方法来处理就好了

    1 date.setMonth(qLastMonth);
    2 date.setDate(0);

    这时候,date就是当前季度的最后一天的date对象了。当然,做一下扩展就可以拿到任意季度的最后一天,只要在第一步设置(喂喂,干嘛呢,绕远了)

    要拿到任意季度的最后一天,比如第2季度,只需要。。。  

    1 date.setMonth(6); // 4 || 5 || 6 都行
    2 date.setDate(0);

    最后,来一个项目中遇到的实际应用里子

    是酱紫的:

        1、获取本季度的最后一天; 当今天是当前季度最后一天切大于15日的时候(比如6月20日),返回下个季度的最后一天。

        2、获取本年最后一天;当今天是大于12月15日的时候,返回明年的最后一天。

    开始的实现代码如下:

     1 /**
     2  * 获取 当前月/当前年/当前季度 的最后一天的date对象
     3  * @param  {String} type 类型选择: month/m, year/y, quarter/q
     4  * @return {object}      最后一天的date对象,目前只可用到天
     5  */
     6 
     7 var SERVER_TIME = Date.now() / 1000; // 服务器时间,这里用本地时间代替
     8 
     9 function getMaxDate(type) {
    10     var date = new Date(SERVER_TIME * 1000);
    11     var m = date.getMonth();
    12     var y = date.getFullYear();
    13     var d = date.getDay();
    14     var today = date.getDate();
    15     console.log(['SERVERTIME: ',y,'-',m+1,'-',today,' 星期',d].join(''));
    16     switch (type) {
    17         case 'm':
    18         case 'month':
    19             date.setMonth(m + 1);
    20             date.setDate(0);
    21             break;
    22         case 'y':
    23         case 'year':
    24             if (m == 11 && today >= 15) {
    25                 date.setFullYear(y + 2);
    26             } else {
    27                 date.setFullYear(y + 1);
    28             }
    29             date.setMonth(0);
    30             date.setDate(0);
    31             break;
    32         case 'q':
    33         case 'quarter':
    34             var qLastMonth = (Math.floor(m / 3) + 1) * 3;
    35             date.setMonth(qLastMonth);
    36             date.setDate(0);
    37 
    38             if (m === date.getMonth() && today >= 15) {
    39                 date.setMonth(qLastMonth + 3);
    40                 date.setDate(0);
    41             }
    42             break;
    43         default:
    44             date = null;
    45     }
    46 
    47     return date;
    48 }

    然后,这段代码有个大坑,平时还不一定能遇到,这不到年底了么,它就出来了(还好不是千年虫之类的千年一遇)

    季度选择的时候,36行或导致后边的38行的代码块出问题。

    比如第2季度,是没有问题的。36行设置date的日到30号,即6月30日;假如满足38行代码逻辑的话,39-40行设置月份到9月30日,没毛病,完美~

    但是,假如现在是第1季度,36行设置日期到3月31日;假如满足38行代码逻辑的话,39-40行设置月份到6月31日,阿西吧,6月没有31日,那么这时候得到的date,就会是~~~~没错,7月1日,自挖坑啊。

    所以,改代码

     1 function getMaxDate(type) {
     2     var date = new Date(SERVER_TIME * 1000);
     3     var m = date.getMonth();
     4     var y = date.getFullYear();
     5     var d = date.getDay();
     6     var today = date.getDate();
     7     console.log(['SERVERTIME: ', y, '-', m + 1, '-', today, ' 星期', d].join(''));
     8     switch (type) {
     9         case 'm':
    10         case 'month':
    11             date.setMonth(m + 1);
    12             date.setDate(0);
    13             break;
    14         case 'y':
    15         case 'year':
    16             if (m == 11 && today >= 15) {
    17                 date.setFullYear(y + 2);
    18             } else {
    19                 date.setFullYear(y + 1);
    20             }
    21             date.setMonth(0);
    22             date.setDate(0);
    23             break;
    24         case 'q':
    25         case 'quarter':
    26             var qLastMonth = (Math.floor(m / 3) + 1) * 3;
    27             if (m === date.getMonth() && today >= 15) {
    28                 date.setMonth(qLastMonth + 3);
    29             } else {
    30                 date.setMonth(qLastMonth);
    31             }
    32             date.setDate(0); // 得到当前季度最后一天的date
    33             break;
    34         default:
    35             date = null;
    36     }
    37 
    38     return date;
    39 }

     这里得到一个注意项,月份的相关判断,一定要在日的设置之前做好,不要再日期设置后在操作月份;或者你把日期设置到1-28之间任意数字,也就是保证不会跑到下一个月去。

    bla~bla~说了一堆,有些废话,反正是有感而写,就当一次记录吧~~


    see U

  • 相关阅读:
    spring boot 扫描不到自定义Controller
    SpringBoot+Maven多模块项目(创建、依赖、打包可执行jar包部署测试)完整流程
    spring boot 中使用 jpa以及jpa介绍
    java8 快速实现List转map 、分组、过滤等操作
    Mysql 创建函数出现This function has none of DETERMINISTIC, NO SQL, or READS SQL DATA
    Spring mvc @initBinder 类型转化器的使用
    @RequestMapping 和@ResponseBody 和 @RequestBody和@PathVariable 注解 注解用法
    ssm的自动类型转换器
    如果将get请求转换成post请求
    如何将post请求转换成put和delete请求
  • 原文地址:https://www.cnblogs.com/ufex/p/get_last_day.html
Copyright © 2011-2022 走看看