Java如何計(jì)算兩個時(shí)間段內(nèi)的工作日天數(shù)
計(jì)算兩個時(shí)間段內(nèi)的工作日天數(shù)
一般在OA系統(tǒng)中都會遇到計(jì)算員工這段時(shí)間內(nèi)的工作天數(shù)。
這種有兩種方式可以解決
- 一是調(diào)用第三方服務(wù)接口進(jìn)行計(jì)算
- 二是自己在系統(tǒng)中寫代碼計(jì)算
一的好處就是每年的節(jié)假日不用自己去維護(hù)直接用兩個時(shí)間段就可以計(jì)算出來。但缺點(diǎn)是有些公司可以會和法定節(jié)假日期不一樣那么這時(shí)候就計(jì)算不了了,還有就是一般這種OA的系統(tǒng)自己內(nèi)部使用的話可能會部署在內(nèi)網(wǎng)上那么就訪問不了第三方接口了。
二需要自己維護(hù)節(jié)假日和計(jì)算。但有效的解決了方式一的缺陷;
根據(jù)方式二的特點(diǎn)介紹一個工具類,用來計(jì)算兩個時(shí)間段內(nèi)的工作天數(shù):
import java.text.ParseException; import java.util.*; import org.apache.commons.lang3.time.DateFormatUtils; import org.apache.commons.lang3.time.DateUtils; public class CalcWorkDays { /** * 法定節(jié)假日列表 * * @return */ public static List<String> getHolidayList() { List<String> holidays = new ArrayList<>(); // 元旦 holidays.add("2021-01-01"); holidays.add("2021-01-02"); holidays.add("2021-01-03"); // 春節(jié) holidays.add("2021-02-11"); holidays.add("2021-02-12"); holidays.add("2021-02-13"); holidays.add("2021-02-14"); holidays.add("2021-02-15"); holidays.add("2021-02-16"); holidays.add("2021-02-17"); // 清明節(jié) holidays.add("2021-04-03"); holidays.add("2021-04-04"); holidays.add("2021-04-05"); // 勞動節(jié) holidays.add("2021-05-01"); holidays.add("2021-05-02"); holidays.add("2021-05-03"); holidays.add("2021-05-04"); holidays.add("2021-05-05"); // 端午節(jié) holidays.add("2021-06-12"); holidays.add("2021-06-13"); holidays.add("2021-06-14"); // 中秋節(jié) holidays.add("2021-09-19"); holidays.add("2021-09-20"); holidays.add("2021-09-21"); // 國慶節(jié) holidays.add("2021-10-01"); holidays.add("2021-10-02"); holidays.add("2021-10-03"); holidays.add("2021-10-04"); holidays.add("2021-10-05"); holidays.add("2021-10-06"); holidays.add("2021-10-07"); return holidays; } /** * 需要上班的周末列表 * * @return */ public static List<String> getNeedWorkWeekends() { List<String> workWeekends = new ArrayList<>(); workWeekends.add("2021-02-07"); workWeekends.add("2021-02-20"); workWeekends.add("2021-04-25"); workWeekends.add("2021-05-08"); workWeekends.add("2021-09-18"); workWeekends.add("2021-09-26"); workWeekends.add("2021-10-09"); return workWeekends; } /** * 計(jì)算兩個日期之間的工作日天數(shù),包含起始日期,包含終止日期 * * @param start 計(jì)算開始時(shí)間 必須傳yyyy-MM-dd的格式 * @param stop 計(jì)算結(jié)束時(shí)間 必須傳yyyy-MM-dd的格式 * @return * @throws ParseException */ public static int calc(String start, String stop) throws ParseException { //獲取今年的法定節(jié)假日期 List<String> workWeekends = getNeedWorkWeekends(); //獲取今年需要補(bǔ)班的日期 List<String> holidays = getHolidayList(); String pattern = "yyyy-MM-dd"; Date begin = DateUtils.parseDate(start, pattern); Date endTime = DateUtils.parseDate(stop, pattern); Calendar cEnd = new GregorianCalendar(); cEnd.setTime(endTime); // 計(jì)算時(shí)算入stop當(dāng)天 cEnd.add(Calendar.DATE,1); Date end=cEnd.getTime(); // Calendar c = Calendar.getInstance(); c.setTime(begin); int count = 0; String ymd = null; while (c.getTime().before(end)) { ymd = DateFormatUtils.format(c.getTime(), pattern); // 不是法定節(jié)假日 if (!holidays.contains(ymd)) { // 不是休息日 if (workWeekends.contains(ymd)) { count++; } else { // 非周末 if (c.get(Calendar.DAY_OF_WEEK) != 1 && c.get(Calendar.DAY_OF_WEEK) != 7) { count++; } } } c.add(Calendar.DATE, 1); } return count; } }
實(shí)現(xiàn)的大概原理就是
循環(huán)將該時(shí)間段的法定節(jié)假日排除 加上需要補(bǔ)班的日期得出最后該時(shí)間段間總共有多少個需要工作的天數(shù)。
以上兩部分的日期可維護(hù)在數(shù)據(jù)庫中,現(xiàn)測試方法目前放到代碼中了。
可以根據(jù)公司的制度進(jìn)行調(diào)整需要工作的日期,比如在正常的工作日是休息可將該日期加到法定節(jié)假日中,在正常的休息日工作的話可將該日期添加到補(bǔ)班日期中。
計(jì)算兩個日期之間的周末天數(shù)
import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; /** * 計(jì)算兩個日期之間的周末天數(shù) * @author yan * */ public class WeekEnd { public static void main(String[] args) throws ParseException { // 設(shè)置傳入的時(shí)間格式 SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); // 指定一個日期 Date date1 = dateFormat.parse("2016-11-12 13:24:16"); Date date2 = dateFormat.parse("2016-11-20 13:24:16"); //第三個三處代表是否 去除 雙休日 System.out.println(computeHolidays(date2,date1)); } public static int computeHolidays(Date t1,Date t2) throws ParseException{ //初始化第一個日期 Calendar cal1 = Calendar.getInstance(); //初始化第二個日期,這里的天數(shù)可以隨便的設(shè)置 Calendar cal2 = Calendar.getInstance(); // 設(shè)置傳入的時(shí)間格式 SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); // 指定一個日期 Date date1 = dateFormat.parse(dateFormat.format(t1)); Date date2 = dateFormat.parse(dateFormat.format(t2)); // 對 calendar 設(shè)置為 date 所定的日期 cal1.setTime(date1); cal2.setTime(date2); int holidays = 0; //確定一個 大日期 if(cal1.compareTo(cal2) > 0){ Calendar temp = cal1; cal1 = cal2; cal2 = temp; temp = null; } while(cal1.compareTo(cal2)<=0){ if(cal1.get(Calendar.DAY_OF_WEEK)==1||cal1.get(Calendar.DAY_OF_WEEK)==7){ holidays++; System.out.println("周末:"+new SimpleDateFormat("yyyy-MM-dd").format(cal1.getTime())); } cal1.add(Calendar.DAY_OF_YEAR,1); } return holidays; } }
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
idea配置maven環(huán)境時(shí)maven下載速度慢的解決方法
我們在idea配置maven環(huán)境的時(shí)候會發(fā)現(xiàn)maven更新慢的現(xiàn)象,解決辦法就是下載國內(nèi)的鏡像包,完美解決下載速度慢的問題,文中有詳細(xì)的具體操作方法,并通過圖文介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02妙用Java8中的Function接口消滅if...else
在開發(fā)過程中經(jīng)常會使用if...else...進(jìn)行判斷拋出異常、分支處理等操作。這些if...else...充斥在代碼中嚴(yán)重影響了代碼代碼的美觀,本文就妙用Java8中的Function接口消滅if...else,感興趣的可以了解一下2022-01-01Spring Boot 實(shí)現(xiàn)配置文件加解密原理
這篇文章主要介紹了Spring Boot 實(shí)現(xiàn)配置文件加解密原理,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06Java設(shè)計(jì)模式之代理模式(Proxy模式)介紹
這篇文章主要介紹了Java設(shè)計(jì)模式之代理模式(Proxy模式)介紹,本文講解了為什么要使用代理模式、如何使用代理模式等內(nèi)容,需要的朋友可以參考下2015-03-03一文解決springboot打包成jar文件無法正常運(yùn)行的問題
這篇文章主要介紹了一文解決springboot打包成jar文件無法正常運(yùn)行的問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07SpringBoot使用token簡單鑒權(quán)的具體實(shí)現(xiàn)方法
這篇文章主要介紹了SpringBoot使用token簡單鑒權(quán)的具體實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11Java StackTraceElement實(shí)例代碼
這篇文章主要介紹了Java StackTraceElement實(shí)例代碼,分享了相關(guān)代碼示例,小編覺得還是挺不錯的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-02-02