由于公司实施SAP HR项目,但是SAP HR对考勤功能真的太弱化了,直接从考勤机上读取的原始打卡记录不能直接传输到HR系统里面,因为SAP HR不能识别那些多余的打卡记录,而且必须把打卡记录进行成组标记(P10,P20),以上标红色的是SAP HR顾问给我的反馈信息。
这样以来,必须开发一套算法来把多余的打卡记录进行过滤掉,然后标记上是P10还是P20,这样以来,HR系统在做时间评估时才不会出现异常情况。
需求已经明确,那么就是设计开发的问题,要开发该功能,需要用到的资源:
1、考勤的排班数据
2、原始打卡数据
3、取原始打卡数据的算法
然后就是确定该功能在我们的外围基础数据平台开发(外围系统是用.net开发的)
由于我们公司的考勤计算方式可能会经常有变化,因为不能把打卡数据的算法写死到代码里面,这样有两种解决方案:1是用侟储过程 2是用动态编译
考虑到以后数据库的迁移问题,最终采取.net动态编译方式来处理该问题
具体的实现思想如下:
1、把算法代码存储在数据库表中
2、新建一个计算考勤的静态类,类里面定义一个计算考勤的静态方法,传入参数为两上Table类型,一个为排班表,一个为原始打卡记录表,返回参数也
为DataTable内型,为计算完成后的结果数据,把该段代码保存在一个静态变量里面。
3、调用计算方法时,把静态类的代码加上代算法代码编译成动态类。
4、利用反射技术,调用计算方法,返回结果。
关键性代码如下:
动态编译类:
DynCompiler.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.CodeDom.Compiler;
using Microsoft.CSharp;
using System.Reflection;
using System.Data; namespace EDICLibrary
{
public class DynCompiler:IDisposable
{ CodeDomProvider _codeprovide;
CompilerParameters _parameters;
CompilerResults _result = null;
MethodInfo _method = null;
Object _instance = null; public DynCompiler()
{
_codeprovide = new CSharpCodeProvider();
_parameters = new CompilerParameters();
_parameters.GenerateExecutable = false;
_parameters.GenerateInMemory = true;
_parameters.ReferencedAssemblies.Add("System.dll");
_parameters.ReferencedAssemblies.Add("System.Data.dll");
_parameters.ReferencedAssemblies.Add("System.Xml.dll");
} public bool SourceCompiler(string source, out string outputMsg)
{
outputMsg = "";
StringBuilder sbout = new StringBuilder();
StringBuilder sb = new StringBuilder();
if (string.IsNullOrEmpty(source))
{
outputMsg = "源代码不能为空!";
return false;
}
else
{ sb.AppendLine("using System;");
sb.AppendLine("using System.Collections.Generic;");
sb.AppendLine("using System.Text;");
sb.AppendLine("using System.Data;");
sb.AppendLine ("using System.Xml;");
sb.AppendLine("namespace JMCompiler");
sb.AppendLine("{");
sb.AppendLine("public class DynCompilerHelper");
sb.AppendLine("{");
sb.AppendLine("public DataTable Calculate(DataTable dtShiftInfo,DataTable dtTimes)");
sb.AppendLine("{");
sb.Append(source);
sb.Append("}");
sb.AppendLine("}");
sb.AppendLine("}");
}
_result = _codeprovide.CompileAssemblyFromSource(_parameters, sb.ToString());
if (_result.Errors.HasErrors)
{
foreach (string str in _result.Output)
{
sbout.AppendLine(str);
}
outputMsg = sbout.ToString();
return false;
}
else
{
Type _type = _result.CompiledAssembly.GetType("JMCompiler.DynCompilerHelper");
_instance = Activator.CreateInstance(_type);
_method = _type.GetMethod("Calculate",new Type[]{typeof(DataTable),typeof (DataTable)});
outputMsg = "编译成功";
return true;
} } public object GetReturnResult(List<DataTable> parameters)
{ if (_method == null)
{
throw new Exception ("未进行代码编译");
}
return _method.Invoke(_instance, parameters.ToArray());
} #region IDisposable 成员 public void Dispose()
{
_codeprovide = null;
_parameters = null;
_result = null;
_method = null;
_instance = null;
} #endregion
}
}
计算考勤类:
AttdCalculate.cs
public class AttdCalculate
{
/// <summary>
/// 计算考勤
/// </summary>
/// <param name="source">计算考勤算法代码</param>
/// <param name="dtShiftInfo">排班信息</param>
/// <param name="dtTimes">原始打卡的数据</param>
/// <param name="msg">输出信息</param>
/// <returns>计算后的结果表</returns>
public static DataTable Calculate(string source, DataTable dtShiftInfo, DataTable dtTimes, out string msg)
{
if (string.IsNullOrEmpty(source))
{
msg = "考勤计算逻辑不能为空";
return new DataTable();
}
DynCompiler complier = new DynCompiler();
string output;
bool result = complier.SourceCompiler(source, out output);
List<DataTable> lstData = new List<DataTable>();
lstData.Add(dtShiftInfo);
lstData.Add(dtTimes);
if (result)
{
msg = "计算成功";
return (DataTable)complier.GetReturnResult(lstData);
}
else
{ msg = output;
return new DataTable();
}
} }
考勤调用代码
//计算公式
string Source = string.Empty;
//获取计算工式
Source = <获取计算工式的方法>
string msg;
//获取考勤数据 DataTable dtResult = MIS.Util.AttdCalculate.Calculate(Source, dtShiftInfo, dtTimes, out msg);
在写这篇文章的时候,开发的功能还没有投入到正式运行,但是经过最近一段时候的测试来看,没有任何问题。