近期做项目时 客户提出一个新需求,计划的工期要去掉双休日节假日,因为节假日的特殊性(由国家规定,非固定的不好计算,无法预测)不做考虑。
下面我们探讨下如何将两个时间段之间的周末去掉
首先一种比较笨的方法是 一天一天去判断是否是周六或周天,该方法简单粗暴 主要用 getDay()去判断获得的星期数是否等于 0或6 是的话则在周末的天数累加 最后去开始与结束日期的差值 扣除 周末总和 就等于工作日啦(PS: 最后日期得加1),以下为实现的方法
function stringToDate(dateString){ dateString = dateString.split('-'); return new Date(dateString[0], dateString[1] - 1, dateString[2]); } function countWorkDay(date1, date2){ date1 = stringToDate(date1); date2 = stringToDate(date2); delta = (date2 - date1) / (1000 * 60 * 60 * 24) + 1; // 计算出总时间 weeks = 0; for(i = 0; i < delta; i++){ if(date1.getDay() == 0 || date1.getDay() == 6) weeks ++; // 若为周六或周天则加1 date1 = date1.valueOf(); date1 += 1000 * 60 * 60 * 24; date1 = new Date(date1); } return delta - weeks; } console.log(countWorkDay('2017-08-01','2017-08-06')); // 4
以上的方法简单粗暴,但是有个大缺点就是 如果时间跨度比较大,这边计算的工作量就比较大了相差一年就得循环365次了。所以改进了下方法,我们都知道一周7天,其中两天为周末5天为工作日,那么 一个时间段就可以看成是0-6的循环即2017-08-01为周二 2017-08-12为周六那么就可以看成2、3、4、5、6、0、1、2、3、4、5、6这样就跟我们以前做数学题一个道理了先算出这个至少有几组完整的,然后在算剩下的有几天
实现如下:
function countWorkDay(sDay,eDay){ var s = stringToDate(sDay), e = stringToDate(eDay); var s_t_w = s.getDay(), e_t_w = e.getDay(); //相差天数 var diffDay = (e - s) / (1000 * 60 * 60 * 24) + 1; var diffWeekDay = diffDay - (s_t_w ==0?0:7-s_t_w) + e_t_w; //计算有几个完整的周 var weeks = Math.floor(diffWeekDay/7); if(weeks<=0){ //在同一周内 即开始结束时间不可能同时为周天与周六(周天为一周第一天) return e_t_w-s_t_w+(s_t_w?1:0)+(e_t_w==6?-1:0); }else{ return weeks*5 + (e_t_w==6?5:e_t_w) + ( s_t_w >= 1 && s_t_w <= 5 ? (6-s_t_w):0); } } console.log(countWorkDay('2017-08-01','2017-08-06'));
以上就是如何计算工作日的两种方法。