支持节假日及调休的工作日历实现

支持节假日及调休的工作日历实现

最近产品及项目中要用到工作日历, 找了一下发现jBPM的工作日历BusinessCalendar可用, 但缺少对节假日调休的支持, 通过简单改造实现了节假日调休和中文配置

主要修改如下:
1\ 在假日维护中增加一个加班或调休的定义维护
public class Holiday{
public static List parseOvertimes(Properties calendarProperties,
			BusinessCalendar businessCalendar) {
		List overtimes = new ArrayList();

		DateFormat dateFormat = new SimpleDateFormat(
				calendarProperties.getProperty("day.format"));
		Iterator iter = calendarProperties.keySet().iterator();
		while (iter.hasNext()) {
			String key = (String) iter.next();
			if (key.startsWith("overtime")) {
				Holiday holiday = new Holiday(
						calendarProperties.getProperty(key), dateFormat,
						businessCalendar);
				overtimes.add(holiday);
			}
		}
		return overtimes;
	}
}

2\ 查找工作日时如发现为非工作日则检查是否为加班或调休
public class BusinessCalendar{

       public Day findDay(Date date) {
		Calendar calendar = getCalendar();
		calendar.setTime(date);
		Day day = weekDays[calendar.get(Calendar.DAY_OF_WEEK)];
		
		//当某天无工作时间时检查是否有加班或调休
		if(day==null || day.dayParts==null || day.dayParts.length==0){
			if(isOvertime(date)){
				day = weekDayCommon;
			}
		}
		
		return day;
	}

}

3\ 扩展BusinessCalendar的add方法支持Calendar的add方法
public class BusinessCalendar{

       /**
	 * 工作日历计算
	 * @param date 初始时间
	 * @param field 时间单位,同Calendar
	 * @param amount  时间数量
	 * @return 按工作日历计算后时间
	 */
	public Date add(Date date, int field, int amount){
		return add(date, field, amount, true);
	}
	
	public Date add(Date date, int field, int amount, boolean isBusinessTime){
		Date end = null;
		if (isBusinessTime) {
			DayPart dayPart = findDayPart(date);
			boolean isInbusinessHours = (dayPart != null);
			if (!isInbusinessHours) {
				Object[] result = new Object[2];
				findDay(date).findNextDayPartStart(0, date, result);
				date = (Date) result[0];
				dayPart = (DayPart) result[1];
			}
			end = dayPart.add(date, field, amount);
		} else {
			Calendar calendar = getCalendar();
			calendar.setTime(date);
			calendar.add(field, amount);
			end = calendar.getTime();
		}
		return end;
	}
}

3\ 增加Duration支持中文定义
public class DayPart{

        public Duration(String duration) {
		if (duration == null)
			throw new NullPointerException("duration is null");
		
		
		int separatorIndex = -1;
		String quantityText = null;
		String unitText = null;
		
		//检查空格 for english formate
		separatorIndex = duration.indexOf(' ');
		if (separatorIndex>1){
			quantityText = duration.substring(0, separatorIndex).trim().toLowerCase();
			unitText = duration.substring(separatorIndex + 1).trim().toLowerCase();	
			if (unitText.startsWith("business")) {
				isBusinessTime = true;
			}
		}
		//检查中文"工作"
		if(separatorIndex<=0){
			separatorIndex = duration.indexOf("工作");
			if(separatorIndex>0){
				isBusinessTime = true;
				quantityText = duration.substring(0, separatorIndex).trim().toLowerCase();
				unitText = duration.substring(separatorIndex + 2).trim().toLowerCase();		
			}			
		}
		//逐字符检查
		if(separatorIndex<=0){
			for(int i=0; i<duration.length(); i++){
				char ch = duration.charAt(i);
				if((48<=ch && ch<=57) || ch==46){
					continue;
				}
				//第一个非数字和.
				quantityText = duration.substring(0, i).trim().toLowerCase();	
				unitText = duration.substring(i).trim().toLowerCase();	
			}
			if(quantityText==null){
				quantityText = duration;
			}
			if(unitText==null || "".equals(unitText)){
				unitText = "H";
			}
		}
				
		double amount = Double.parseDouble(quantityText);
		Long unit = (Long) units.get(unitText);
		this.milliseconds = (long) (amount * unit.longValue());
	}

       static {
		units = new HashMap();
		units.put("second", new Long(SECOND));
		units.put("seconds", new Long(SECOND));
		units.put("秒", new Long(SECOND));
		units.put("S", new Long(SECOND));
		units.put("s", new Long(SECOND));
		units.put("business second", new Long(SECOND));
		units.put("business seconds", new Long(SECOND));
		units.put("minute", new Long(MINUTE));
		units.put("minutes", new Long(MINUTE));
		units.put("分钟", new Long(MINUTE));
		units.put("Min", new Long(MINUTE));
		units.put("min", new Long(MINUTE));
		units.put("business minute", new Long(MINUTE));
		units.put("business minutes", new Long(MINUTE));
		units.put("hour", new Long(HOUR));
		units.put("hours", new Long(HOUR));
		units.put("小时", new Long(HOUR));
		units.put("时", new Long(HOUR));
		units.put("H", new Long(HOUR));
		units.put("h", new Long(HOUR));
		units.put("business hour", new Long(HOUR));
		units.put("business hours", new Long(HOUR));
		units.put("day", new Long(DAY));
		units.put("days", new Long(DAY));
		units.put("天", new Long(DAY));
		units.put("日", new Long(DAY));
		units.put("D", new Long(DAY));
		units.put("d", new Long(DAY));
		units.put("business day", new Long(BUSINESS_DAY));
		units.put("business days", new Long(BUSINESS_DAY));
		units.put("week", new Long(WEEK));
		units.put("weeks", new Long(WEEK));
		units.put("星期", new Long(WEEK));
		units.put("周", new Long(WEEK));
		units.put("W", new Long(WEEK));
		units.put("w", new Long(WEEK));
		units.put("business week", new Long(BUSINESS_WEEK));
		units.put("business weeks", new Long(BUSINESS_WEEK));
		units.put("month", new Long(MONTH));
		units.put("months", new Long(MONTH));
		units.put("月", new Long(MONTH));
		units.put("M", new Long(MONTH));
		units.put("m", new Long(MONTH));
		units.put("business month", new Long(BUSINESS_MONTH));
		units.put("business months", new Long(BUSINESS_MONTH));
		units.put("year", new Long(YEAR));
		units.put("years", new Long(YEAR));
		units.put("年", new Long(YEAR));
		units.put("Y", new Long(YEAR));
		units.put("y", new Long(YEAR));
		units.put("business year", new Long(BUSINESS_YEAR));
		units.put("business years", new Long(BUSINESS_YEAR));
	}
}



BusinessCalendar配置文件范例
## 工作日及工作时间定义
hour.format=HH:mm   
#weekday ::= [<daypart> [& <daypart>]*]   
#daypart ::= <start-hour>-<to-hour>  
#start-hour and to-hour must be in the hour.format   
#dayparts have to be ordered   
weekday.common=9:00-12:00 & 12:30-17:00  #通用工作时间, 周末调休会用到
weekday.monday=9:00-12:00 & 12:30-17:00 
weekday.thuesday=9:00-12:00 & 12:30-17:00
weekday.wednesday=9:00-12:00 & 12:30-17:00
weekday.thursday=9:00-12:00 & 12:30-17:00
weekday.friday=9:00-12:00 & 12:30-17:00
weekday.saturday=
weekday.sunday=

## 节假日及加班调休定义
day.format=yyyy/MM/dd
# holiday syntax: <holiday>  
# holiday period syntax: <start-day>-<end-day>  
# below are the belgian official holidays 
#元旦  
holiday.1=2012/01/01-2012/01/03
#春节 
holiday.2=2012/01/22-2012/01/28
overtime.2=2012/01/21
overtime.2a=2012/01/29
#清明
holiday.3=2012/04/02-2012/04/04
overtime.3=2012/03/31-2012/04/01
#五一
holiday.4=2012/04/29-2012/05/01
overtime.4=2012/04/28
#端午   
holiday.5=2012/06/22-2012/06/24 
#中秋 
holiday.6=2012/09/30
#国庆
holiday.7=2012/10/01-2012/10/07
overtime.7=2012/09/29

## 其他参数定义  
business.day.expressed.in.hours=8
business.week.expressed.in.hours=40
business.month.expressed.in.business.days=21
business.year.expressed.in.business.days=220


jBPM BusinessCalendar.java
http://kickjava.com/src/org/jbpm/calendar/BusinessCalendar.java.htm
OBE1.0 BusinessCalendar.java
https://java2s.com/Open-Source/Java/Workflow-Engines/obe-1.0/org/obe/engine/calendar/BusinessCalendar.java.htm

你可能感兴趣的:(BPM,节假日,调休,工作日历)