最近因为某些坏事,做了这么一个用来确认工作日的逻辑,前一阵子看到论坛上有些人也在问思路什么的,所以就在这里记录下
首先需要说明:因为这个判断只基于日期,未基于小时,所以这个判断标准不支持跨日,即不支持从某一天晚上到第二天早上,也就是俗称的夜班
其实判断某一天是否为工作日的逻辑非常简单:
1、按常规情况判断该日期是否是工作日
2、在第一步的基础上,判断该日期是否为特殊日期,所谓的特殊日期就是按常规情况应该是上班的,但实际这天是休息日;按常规情况应该是休息的,但实际情况是工作日,即常规判断标准取反
3、因为特殊日期是由国家每年公布,所以这需要有个维护的功能
下面是特殊日期维护小工具的一个截图,这里用xml作为持久化处理的载体,实际可以用数据库之类的进行持久化
这个小工具包含增删以及查,编辑就没有了,因为不需要,这里的常规判断标准如工具右上角备注:默认周六周日休息,其他时间为工作时间,另外2015年的特殊日期已经按国家规定进行了维护,小工具可维护的年份为当前年份及下一个年份,当然如果你要维护更多的年份,那只需要将相应的代码稍做处理即可
在小工具维护好特殊日期之后,那判断是否为工作日期的代码就仅仅如下几行即可
/// <summary>
/// 判断指定日期是否是工作时间
/// </summary>
/// <param name="dt"></param>
/// <returns>true表示工作日,false表示非工作日</returns>
private static bool IsWorkDay(DateTime dt)
{
bool isWorkDay = (int)dt.DayOfWeek > 0 && (int)dt.DayOfWeek < 6;//此处可按实际需要进行调整
string filePath = string.Format("{0}{1}{2}.xml", ConfigurationManager.AppSettings["xmlDirectory"], ConfigurationManager.AppSettings["xmlFileName"], dt.Year);
XElement root = XElement.Load(filePath);//如果有必要,可以将特殊日期集合做缓存,判断也就只需判断Date是否相同,而不是下面的转化成string
if (root.Elements("SpecialDate").Any(s => s.Value == dt.ToString("yyyy-MM-dd")))
{
isWorkDay = !isWorkDay;
}
//Logger.Write(dt.ToString("yyyy-MM-dd ") + (isWorkDay ? "属于工作时间" : "属于非工作时间"), LogCategories.WorkDay);
return isWorkDay;
}
补充下:如果要判断某日是否为法定节假日什么的,这个就支持不了了,当然要支持的话,可以在特殊日期部分增加标志位,然后判断标准适当的调整下