由于公司实施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);

在写这篇文章的时候,开发的功能还没有投入到正式运行,但是经过最近一段时候的测试来看,没有任何问题。

.net 动态编译解决考勤计算问题的更多相关文章

  1. Java_java动态编译整个项目,解决jar包找不到问题

    java动态编译整个项目,解决jar包找不到问题原文:http://itzyx.com/index.php/javac/ 动态将java文件编译为class文件解决方案:将temp\sdl\src目录 ...

  2. (转载)JAVA动态编译--字节代码的操纵

    在一般的Java应用开发过程中,开发人员使用Java的方式比较简单.打开惯用的IDE,编写Java源代码,再利用IDE提供的功能直接运行Java 程序就可以了.这种开发模式背后的过程是:开发人员编写的 ...

  3. JIT(动态编译)和AOT(静态编译)编译技术比较

    Java 应用程序的性能经常成为开发社区中的讨论热点.因为该语言的设计初衷是使用解释的方式支持应用程序的可移植性目标,早期 Java 运行时所提供的性能级别远低于 C 和 C++ 之类的编译语言.尽管 ...

  4. 分享基于.NET动态编译&Newtonsoft.Json封装实现JSON转换器(JsonConverter)原理及JSON操作技巧

    看文章标题就知道,本文的主题就是关于JSON,JSON转换器(JsonConverter)具有将C#定义的类源代码直接转换成对应的JSON字符串,以及将JSON字符串转换成对应的C#定义的类源代码,而 ...

  5. Java_动态编译总结

    不多说直接上代码: 动态编译的主类: package com.lkb.autoCode.util; import com.lkb.autoCode.constant.AutoCodeConstant; ...

  6. .NET中的动态编译

    代码的动态编译并执行是一个.NET平台提供给我们的很强大的工具用以灵活扩展(当然是面对内部开发人员)复杂而无法估算的逻辑,并通过一些额外的代码来扩展我们已有 的应用程序.这在很大程度上给我们提供了另外 ...

  7. Java_JAVA6动态编译的问题

    摘自:http://www.iteye.com/problems/14909 在使用JAVA6动态编译时遇到的一个问题,动态编译方法已经写就.通过main方法调用的动态编译时,编译通过,并可以使用编译 ...

  8. 关于cshtml中的js对动态编译支持的问题

    问题:MVC4中支持对ViewBag.ViewDate等的动态编译,但是在js中对它的支持就是有问题.虽然是可以动态编译,但是动态编译之后,断点无法获取. $.getJSON("/api/A ...

  9. ASP.NET 动态编译、预编译和 WebDeployment 项目(转)

    概述 在 Web 服务器上,既可以部署源文件,也可以部署编译后程序集. 若部署源文件,则当用户访问时,Web 应用程序会被动态编译,并缓存该程序集,以便下次访问. 否则,若部署程序集,Web 应用程序 ...

随机推荐

  1. 关于 profile文件(转)

    登录shell执行了两个特殊文件, 1个是:\etc\profile, 这个文件由系统管理员设置,通常做一些如检查是否有邮件,设置默认的创建文件的掩码,给某些表转到处变量赋值,已经任何管理员希望每当用 ...

  2. VMware10中的CentOS6.5命令行安装VMwaretools工具启用windows与虚拟机中Linux系统的共享目录

    VMware10中的CentOS6.5命令行安装VMwaretools工具启用windows与虚拟机中Linux系统的共享目录 一.描述 系统描述:win7旗舰版64位系统+VMware Workst ...

  3. table extraction

    http://yz.mit.edu/papers/webtables-vldb08.pdf http://www.vldb.org/pvldb/vol4/p528-venetis.pdf Alon H ...

  4. php 调用.net的webservice 需要注意的

    首先 SoapClient类这个类用来使用Web services.SoapClient类可以作为给定Web services的客户端.它有两种操作形式:* WSDL 模式* Non-WSDL 模式在 ...

  5. ionic localstorage

    angular.module('locals',[]) .factory('ls', ['$window', function($window) { return { set: function(ke ...

  6. myeclipse的一些设置

    常用快捷键 Ctrl + Shift + R 搜索文件Ctrl + Shift + T 搜索类型Ctrl + O 搜索成员   智能提示 Windows > Preferences > G ...

  7. openstack实例热迁移

    [DEFAULT]scheduler_default_filters=AllHostsFilterallow_resize_to_same_host=Trueallow_migrate_to_same ...

  8. 如何解决SWAT模型数据移动目录后出现的“SWAT2005.mdb database specified in your MasterProgress table does not exists. Please correct and try again”的问题

    方法: 1.用MS Access软件打开SWAT模型工程文件的数据文件,如“**流域模拟.mdb”,该文件一般存放在工程文件“**流域模拟.mxd”相同的路径: 2.打开以后,找到“MasterPro ...

  9. Linux下的多进程编程

    1.进程 1.1进程的定义 <计算机操作系统>这门课对进程有这样的描述:进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,是操作系统 ...

  10. WP8_区分滑动和点击(在图片列表中)

    在windows phone中,对于一个页面中 有图片列表的,滑动的时候,很容易被误认为是点击了图片,而打开图片详细信息等,原意是滑动列表,由此对图片添加2个事件,来控制其点击行为(滑动的时候,基本不 ...