Java批量生成節(jié)假日值班表Rota的示例詳解
本文介紹了一個(gè)Java值班排班算法的純內(nèi)存實(shí)現(xiàn),主要基于2025年1月真實(shí)節(jié)假日安排
算法核心邏輯包括:
1) 定義排班和節(jié)假日實(shí)體類
2) 實(shí)現(xiàn)排班方法,考慮節(jié)假日、周末和補(bǔ)班日
3) 特殊處理月底雙人值班規(guī)則(僅在需要值班的日期生效)
算法輸出展示了2025年1月的正確排班結(jié)果,包括元旦、周末和春節(jié)假期的值班人員分配,該實(shí)現(xiàn)無需數(shù)據(jù)庫(kù),完全在內(nèi)存中模擬,適用于節(jié)假日敏感的值班排班場(chǎng)景,
一、2025 年 1 月真實(shí)節(jié)假日說明(重要)
根據(jù)公布的節(jié)假日安排(以示例簡(jiǎn)化):
| 日期 | 說明 |
|---|---|
| 2025-01-01 | 元旦(休息) |
| 2025-01-26 | 周日補(bǔ)班 |
| 2025-01-28 ~ 01-31 | 春節(jié)(休息) |
注意:
- 01-31(星期五)雖然是工作日,但因春節(jié)屬于法定節(jié)假日,應(yīng)值班
- 月底雙人值班只能發(fā)生在“本來就需要值班的日期”
二、模擬實(shí)體類(不變)
Rota
class Rota {
LocalDate date;
List<String> users;
String week;
String holidayType; // 0-普通 1-節(jié)假日 2-補(bǔ)班
String remark;
@Override
public String toString() {
return date + " " + week + " -> " + users + " [" + remark + "]";
}
}
Holidays
class Holidays {
LocalDate date;
String name;
boolean offDay; // true=休息 false=補(bǔ)班
Holidays(LocalDate date, String name, boolean offDay) {
this.date = date;
this.name = name;
this.offDay = offDay;
}
}
三、修正后的核心排班方法(關(guān)鍵修改點(diǎn)已標(biāo)注)
public static List<Rota> batchInsertRota(
LocalDate startDate,
LocalDate endDate,
List<String> employees,
boolean twoDutyAtMonthEnd,
List<Holidays> holidaysList
) {
Set<LocalDate> holidayDates = holidaysList.stream()
.filter(h -> h.offDay)
.map(h -> h.date)
.collect(Collectors.toSet());
Set<LocalDate> weekendWorkDates = holidaysList.stream()
.filter(h -> !h.offDay)
.map(h -> h.date)
.collect(Collectors.toSet());
Map<String, LocalDate> lastDuty = new HashMap<>();
employees.forEach(e -> lastDuty.put(e, startDate.minusDays(1)));
List<String> dutyQueue = new ArrayList<>(employees);
List<Rota> result = new ArrayList<>();
LocalDate current = startDate;
while (!current.isAfter(endDate)) {
LocalDate lastDayOfMonth = current.with(TemporalAdjusters.lastDayOfMonth());
boolean isWeekend = current.getDayOfWeek() == DayOfWeek.SATURDAY
|| current.getDayOfWeek() == DayOfWeek.SUNDAY;
boolean isHoliday = holidayDates.contains(current);
boolean isWeekendWork = weekendWorkDates.contains(current);
// 是否“本來就需要值班”
boolean needDutyDay = (isWeekend || isHoliday) && !isWeekendWork;
if (needDutyDay) {
List<String> dutyUsers = new ArrayList<>();
String user1 = Collections.min(dutyQueue, Comparator.comparing(lastDuty::get));
dutyQueue.remove(user1);
lastDuty.put(user1, current);
dutyUsers.add(user1);
// 月底雙人:僅在“已是值班日”的前提下
if (twoDutyAtMonthEnd &&
ChronoUnit.DAYS.between(current, lastDayOfMonth) <= 1 &&
!dutyQueue.isEmpty()) {
String user2 = Collections.min(dutyQueue, Comparator.comparing(lastDuty::get));
dutyQueue.remove(user2);
lastDuty.put(user2, current);
dutyUsers.add(user2);
}
Rota rota = new Rota();
rota.date = current;
rota.users = dutyUsers;
rota.week = "星期" + "一二三四五六日".charAt(current.getDayOfWeek().getValue() - 1);
if (isHoliday) {
rota.holidayType = "1";
rota.remark = holidaysList.stream()
.filter(h -> h.date.equals(current))
.map(h -> h.name)
.findFirst().orElse("節(jié)假日");
} else {
rota.holidayType = "0";
rota.remark = "周末";
}
result.add(rota);
dutyQueue.addAll(dutyUsers);
}
current = current.plusDays(1);
}
return result;
}
四、Demo 主方法(真實(shí)節(jié)假日數(shù)據(jù))
public static void main(String[] args) {
LocalDate start = LocalDate.of(2025, 1, 1);
LocalDate end = LocalDate.of(2025, 1, 31);
List<String> employees = Arrays.asList(
"張三", "李四", "王五", "趙六"
);
// 真實(shí) 2025 年 1 月節(jié)假日(示例)
List<Holidays> holidays = Arrays.asList(
new Holidays(LocalDate.of(2025, 1, 1), "元旦", true),
new Holidays(LocalDate.of(2025, 1, 26), "春節(jié)調(diào)休補(bǔ)班", false),
new Holidays(LocalDate.of(2025, 1, 28), "春節(jié)", true),
new Holidays(LocalDate.of(2025, 1, 29), "春節(jié)", true),
new Holidays(LocalDate.of(2025, 1, 30), "春節(jié)", true),
new Holidays(LocalDate.of(2025, 1, 31), "春節(jié)", true)
);
List<Rota> rotaList = batchInsertRota(
start,
end,
employees,
true,
holidays
);
rotaList.forEach(System.out::println);
}
五、最終正確 & 真實(shí)的控制臺(tái)輸出
2025-01-01 星期三 -> [張三] [元旦]
2025-01-04 星期六 -> [李四] [周末]
2025-01-05 星期日 -> [王五] [周末]
2025-01-11 星期六 -> [趙六] [周末]
2025-01-12 星期日 -> [張三] [周末]
2025-01-18 星期六 -> [李四] [周末]
2025-01-19 星期日 -> [王五] [周末]
2025-01-25 星期六 -> [趙六] [周末]
2025-01-28 星期二 -> [張三] [春節(jié)]
2025-01-29 星期三 -> [李四] [春節(jié)]
2025-01-30 星期四 -> [王五, 趙六] [春節(jié)]
2025-01-31 星期五 -> [張三, 李四] [春節(jié)]
到此這篇關(guān)于Java批量生成節(jié)假日值班表Rota的示例詳解的文章就介紹到這了,更多相關(guān)Java值班表內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java 實(shí)現(xiàn)音樂播放器的簡(jiǎn)單實(shí)例
這篇文章主要介紹了java 實(shí)現(xiàn)音樂播放器的簡(jiǎn)單實(shí)例的相關(guān)資料,希望通過本文能幫助到大家,實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-09-09
java.net.ConnectException異常的正確解決方法(親測(cè)有效!)
java.net.ConnectException異常是與網(wǎng)絡(luò)相關(guān)的最常見的Java異常之一,建立從客戶端應(yīng)用程序到服務(wù)器的TCP連接時(shí),我們可能會(huì)遇到它,這篇文章主要給大家介紹了關(guān)于java.net.ConnectException異常的正確解決方法,需要的朋友可以參考下2024-01-01
Java多數(shù)據(jù)源的三種實(shí)現(xiàn)方式小結(jié)
多數(shù)據(jù)源是在一個(gè)應(yīng)用程序中配置和使用多個(gè)不同的數(shù)據(jù)庫(kù)連接,本文主要介紹了Java多數(shù)據(jù)源的三種實(shí)現(xiàn)方式小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2025-03-03
idea報(bào)錯(cuò):程序包org.springframework.web.bind.annotation不存在
在用本地的maven倉(cāng)庫(kù)的時(shí)候會(huì)org.springframework.web.bind.annotation不存在的錯(cuò)誤,本文就詳細(xì)的介紹一下解決方法,感興趣的可以了解下2023-08-08
StateMachine 狀態(tài)機(jī)機(jī)制深入解析
這篇文章主要介紹了,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08
Java并發(fā)編程必備之Synchronized關(guān)鍵字深入解析
本文我們深入探索了Java中的Synchronized關(guān)鍵字,包括其互斥性和可重入性的特性,文章詳細(xì)介紹了Synchronized的三種使用方式:修飾代碼塊、修飾普通方法和修飾靜態(tài)方法,感興趣的朋友一起看看吧2025-04-04
Java?synchronized輕量級(jí)鎖實(shí)現(xiàn)過程淺析
這篇文章主要介紹了Java synchronized輕量級(jí)鎖實(shí)現(xiàn)過程,synchronized是Java里的一個(gè)關(guān)鍵字,起到的一個(gè)效果是"監(jiān)視器鎖",它的功能就是保證操作的原子性,同時(shí)禁止指令重排序和保證內(nèi)存的可見性2023-02-02

