2023年2月

我们在一般的接口函数开发中,为了安全性,我们都需要对传入的参数进行验证,确保参数按照我们所希望的范围输入,如果在范围之外,如空值,不符合的类型等等,都应该给出异常或错误提示信息。这个参数的验证处理有多种方式,最为简单的方式就是使用条件语句对参数进行判断,这样的判断代码虽然容易理解,但比较臃肿,如果对多个参数、多个条件进行处理,那么代码就非常臃肿难以维护了,本篇随笔通过分析几种不同的参数验证方式,最终采用较为优雅的方式进行处理。

通常会规定类型参数是否允许为空,如果是字符可能有长度限制,如果是整数可能需要判断范围,如果是一些特殊的类型比如电话号码,邮件地址等,可能需要使用正则表达式进行判断。参考随笔《
C# 中参数验证方式的演变
》中文章的介绍,我们对参数的验证方式有几种。

1、常规方式的参数验证

一般我们就是对方法的参数使用条件语句的方式进行判断,如下函数所示。

public bool Register(string name, intage)
{
if (string.IsNullOrEmpty(name))
{
throw new ArgumentException("name should not be empty", "name");
}
if (age < 10 || age > 70)
{
throw new ArgumentException("the age must between 10 and 70","age");
}
//insert into db }

或者

public void Initialize(string name, intid)
{
if (string.IsNullOrEmpty(value))throw new ArgumentException("name");if (id < 0)throw new ArgumentOutOfRangeException("id");//Do some work here. }

如果复杂的参数校验,那么代码就比较臃肿

void TheOldFashionWay(int id, IEnumerable<int>col, 
DayOfWeek day)
{
if (id < 1)
{
throw new ArgumentOutOfRangeException("id",
String.Format(
"id should be greater" + "than 0. The actual value is {0}.", id));
}
if (col == null)
{
throw new ArgumentNullException("col","collection should not be empty");
}
if (col.Count() == 0)
{
throw newArgumentException("collection should not be empty", "col");
}
if (day >= DayOfWeek.Monday &&day<=DayOfWeek.Friday)
{
throw newInvalidEnumArgumentException(
String.Format(
"day should be between" + "Monday and Friday. The actual value" + "is {0}.", day));
}
//Do method work }

有时候为了方便,会把参数校验的方法,做一个通用的辅助类进行处理,如在我的公用类库里面提供了一个:参数验证的通用校验辅助类 ArgumentValidation,使用如下代码所示。

     public classTranContext:IDisposable   
{
private readonly TranSetting setting=null;private IBuilder builder=null;private ILog log=null;private ManuSetting section=null;public eventEndReportEventHandler EndReport;publicTranContext()
{
}
publicTranContext(TranSetting setting)
{
ArgumentValidation.CheckForNullReference (setting,
"TranSetting");this.setting =setting;
}
public TranContext(string key,string askFileName,stringoperation)
{
ArgumentValidation.CheckForEmptyString (key,
"key");
ArgumentValidation.CheckForEmptyString (askFileName,
"askFileName");
ArgumentValidation.CheckForEmptyString (operation,
"operation");
setting
=new TranSetting (this,key,askFileName,operation);
}

但是这样的方式还是不够完美,不够流畅。

2、基于第三方类库的验证方式

在GitHub上有一些验证类库也提供了对参数验证的功能,使用起来比较简便,采用一种流畅的串联写法。如
CuttingEdge.Conditions
等。CuttingEdge.Condition 里面的例子代码我们来看看。

public ICollection GetData(Nullable<int> id, string xml, IEnumerable<int>col)
{
//Check all preconditions: Condition.Requires(id, "id")
.IsNotNull()
//throws ArgumentNullException on failure .IsInRange(1, 999) //ArgumentOutOfRangeException on failure .IsNotEqualTo(128); //throws ArgumentException on failure Condition.Requires(xml,"xml")
.StartsWith(
"<data>") //throws ArgumentException on failure .EndsWith("</data>") //throws ArgumentException on failure .Evaluate(xml.Contains("abc") || xml.Contains("cba")); //arg ex Condition.Requires(col,"col")
.IsNotNull()
//throws ArgumentNullException on failure .IsEmpty() //throws ArgumentException on failure .Evaluate(c => c.Contains(id.Value) || c.Contains(0)); //arg ex//Do some work//Example: Call a method that should not return null object result =BuildResults(xml, col);//Check all postconditions: Condition.Ensures(result, "result")
.IsOfType(
typeof(ICollection)); //throws PostconditionException on failure return(ICollection)result;
}
public static int[] Multiply(int[] left, int[] right)
{
Condition.Requires(left,
"left").IsNotNull();//You can add an optional description to each check Condition.Requires(right, "right")
.IsNotNull()
.HasLength(left.Length,
"left and right should have the same length");//Do multiplication }

这种书写方式比较流畅,而且也提供了比较强大的参数校验方式,除了可以使用其IsNotNull、IsEmpty等内置函数,也可以使用Evaluate这个扩展判断非常好的函数来处理一些自定义的判断,应该说可以满足绝大多数的参数验证要求了,唯一不好的就是需要使用这个第三方类库吧,有时候如需扩展就麻烦一些。而且一般来说我们自己有一些公用类库,如果对参数验证也还需要引入一个类库,还是比较麻烦一些的(个人见解)

3、Code Contract

Code Contracts
是微软研究院开发的一个编程类库,我最早看到是在
C# In Depth
的第二版中,当时.NET 4.0还没有出来,当时是作为一个第三方类库存在的,到了.NET 4.0之后,已经加入到了.NET BCL中,该类存在于System.Diagnostics.Contracts 这个命名空间中。

这个是美其名曰:契约编程

C#代码契约起源于微软开发的一门研究语言Spec#(参见http://mng.bz/4147)。

• 契约工具:包括:ccrewrite(二进制重写器,基于项目的设置确保契约得以贯彻执行)、ccrefgen(它生成契约引用集,为客户端提供契约信息)、cccheck(静态检查器,确保代码能在编译时满足要求,而不是仅仅检查在执行时实际会发生什么)、ccdocgen(它可以为代码中指定的契约生成xml文档)。

• 契约种类:前置条件、后置条件、固定条件、断言和假设、旧式契约。

• 代码契约工具下载及安装:下载地址Http://mng.bz/cn2k。(代码契约工具并不包含在Visual Studio 2010中,但是其核心类型位于mscorlib内。)

• 命名空间:System.Diagnostics.Contracts.Contract

Code Contract 使得.NET 中契约式设计和编程变得更加容易,Contract中的这些静态方法方法包括

  1. Requires:函数入口处必须满足的条件
  2. Ensures:函数出口处必须满足的条件
  3. Invariants:所有成员函数出口处都必须满足的条件
  4. Assertions:在某一点必须满足的条件
  5. Assumptions:在某一点必然满足的条件,用来减少不必要的警告信息

Code Contract 的使用文档您可以从
官网下载
到。为了方便使用Visual Studio开发。我们可以安装一个
Code Contracts for .NET
插件。安装完了之后,点击Visual Studio中的项目属性,可以看到如下丰富的选择项:

Contract和Debug.Assert有些地方相似:

  1. 都提供了运行时支持:这些Contracts都是可以被运行的,并且一旦条件不被满足,会弹出类似Assert的一样的对话框报错,如下:
  2. 都可以在随意的在代码中关闭打开。

但是Contract有更多和更强大的功能:

  1. Contracts的意图更加清晰,通过不同的Requires/Ensures等等调用,代表不同类型的条件,比单纯的Assert更容易理解和进行自动分析
  2. Contracts的位置更加统一,将3种不同条件都放在代码的开始处,而非散见在函数的开头和结尾,便于查找和分析。
  3. 不同的开发人员、不同的小组、不同的公司、不同的库可能都会有自己的Assert,这就大大增加了自动分析的难度,也不利于开发人员编写代码。而Contracts直接被.NET 4.0支持,是统一的。
  4. 它提供了静态分析支持,这个我们可以通过配置面板看到,通过静态分析Contracts,静态分析工具可以比较容易掌握函数的各种有关信息,甚至可以作为Intellisense

Contract中包含了三个工具:

  • ccrewrite, 通过向程序集中些如二进制数据,来支持运行时检测
  • cccheck, 运行时检测
  • ccdoc, 将Contract自动生成XML文档

前置条件的处理,如代码所示。

       /// <summary>
        ///实现“前置条件”的代码契约/// </summary>
        /// <param name="text">Input</param>
        /// <returns>Output</returns>
        public static int CountWhiteSpace(stringtext)
{
//命名空间:using System.Diagnostics.Contracts; Contract.Requires<ArgumentNullException>(text != null, "Paramter:text");//使用了泛型形式的Requires return text.Count(char.IsWhiteSpace);
}

后置条件(postcondition):表示对方法输出的约束:返回值、out或ref参数的值,以及任何被改变的状态。Ensures();

        /// <summary>
        ///实现“后置条件”的代码契约/// </summary>
        /// <param name="text">Input</param>
        /// <returns>Output</returns>
        public static int CountWhiteSpace(stringtext)
{
//命名空间:using System.Diagnostics.Contracts; Contract.Requires<ArgumentNullException>(!string.IsNullOrEmpty(text), "text"); //使用了泛型形式的Requires Contract.Ensures(Contract.Result<int>() > 0); //1.方法在return之前,所有的契约都要在真正执行方法之前(Assert和Assume除外,下面会介绍)。//2.实际上Result<int>()仅仅是编译器知道的”占位符“:在使用的时候工具知道它代表了”我们将得到那个返回值“。 return text.Count(char.IsWhiteSpace);
}
public static bool TryParsePreserveValue(string text, ref intvalue)
{
Contract.Ensures(Contract.Result
<bool>() || Contract.OldValue(value) == Contract.ValueAtReturn(out value)); //此结果表达式是无法证明真伪的。 return int.TryParse(text, out value); //所以此处在编译前就会提示错误信息:Code Contract:ensures unproven: XXXXX }

这个代码契约功能比较强大,不过好像对于简单的参数校验,引入这么一个家伙感觉麻烦,也不见开发人员用的有多广泛,而且还需要提前安装一个工具:
Code Contracts for .NET

因此我也不倾向于使用这个插件的东西,因为代码要交付客户使用,要求客户安装一个插件,并且打开相关的代码契约设置,还是比较麻烦,如果没有打开,也不会告诉客户代码编译出错,只是会在运行的时候不校验方法参数。

4、使用内置的公用类库处理

基于CuttingEdge.Conditions 的方式,其实我们也可以做一个类似这样的流畅性写法的校验处理,而且不需要那么麻烦引入第三方类库。

例如我们在公用类库里面增加一个类库,如下代码所示。

    /// <summary>
    ///参数验证帮助类,使用扩展函数实现/// </summary>
    /// <example>
    ///eg:///ArgumentCheck.Begin().NotNull(sourceArray, "需要操作的数组").NotNull(addArray, "被添加的数组");/// </example>
    public static classArgumentCheck
{
#region Methods /// <summary> ///验证初始化/// <para> ///eg:///ArgumentCheck.Begin().NotNull(sourceArray, "需要操作的数组").NotNull(addArray, "被添加的数组");/// </para> /// <para> ///ArgumentCheck.Begin().NotNullOrEmpty(tableName, "表名").NotNullOrEmpty(primaryKey, "主键");</para> /// <para> ///ArgumentCheck.Begin().CheckLessThan(percent, "百分比", 100, true);</para> /// <para> ///ArgumentCheck.Begin().CheckGreaterThan&lt;int&gt;(pageIndex, "页索引", 0, false).CheckGreaterThan&lt;int&gt;(pageSize, "页大小", 0, false);</para> /// <para> ///ArgumentCheck.Begin().NotNullOrEmpty(filepath, "文件路径").IsFilePath(filepath).NotNullOrEmpty(regexString, "正则表达式");</para> /// <para> ///ArgumentCheck.Begin().NotNullOrEmpty(libFilePath, "非托管DLL路径").IsFilePath(libFilePath).CheckFileExists(libFilePath);</para> /// <para> ///ArgumentCheck.Begin().InRange(brightnessValue, 0, 100, "图片亮度值");</para> /// <para> ///ArgumentCheck.Begin().Check&lt;ArgumentNullException&gt;(() => config.HasFile, "config文件不存在。");</para> /// <para> ///ArgumentCheck.Begin().NotNull(serialPort, "串口").Check&lt;ArgumentException&gt;(() => serialPort.IsOpen, "串口尚未打开!").NotNull(data, "串口发送数据");/// </para> /// </summary> /// <returns>Validation对象</returns> public staticValidation Begin()
{
return null;
}
/// <summary> ///需要验证的正则表达式/// </summary> /// <param name="validation">Validation</param> /// <param name="checkFactory">委托</param> /// <param name="argumentName">参数名称</param> /// <returns>Validation对象</returns> public static Validation Check(this Validation validation, Func<bool> checkFactory, stringargumentName)
{
return Check<ArgumentException>(validation, checkFactory, string.Format(Resource.ParameterCheck_Match2, argumentName));
}
/// <summary> ///自定义参数检查/// </summary> /// <typeparam name="TException">泛型</typeparam> /// <param name="validation">Validation</param> /// <param name="checkedFactory">委托</param> /// <param name="message">自定义错误消息</param> /// <returns>Validation对象</returns> public static Validation Check<TException>(this Validation validation, Func<bool> checkedFactory, stringmessage)whereTException : Exception
{
if(checkedFactory())
{
return validation ?? newValidation()
{
IsValid
= true};
}
else{
TException _exception
= (TException)Activator.CreateInstance(typeof(TException), message);throw_exception;
}
}
......

上面提供了一个常规的检查和泛型类型检查的通用方法,我们如果需要对参数检查,如下代码所示。

ArgumentCheck.Begin().NotNull(sourceArray, "需要操作的数组").NotNull(addArray, "被添加的数组");

而这个NotNull就是我们根据上面的定义方法进行扩展的函数,如下代码所示。

        /// <summary>
        ///验证非空/// </summary>
        /// <param name="validation">Validation</param>
        /// <param name="data">输入项</param>
        /// <param name="argumentName">参数名称</param>
        /// <returns>Validation对象</returns>
        public static Validation NotNull(this Validation validation, object data, stringargumentName)
{
return Check<ArgumentNullException>(validation, () => (data != null), string.Format(Resource.ParameterCheck_NotNull, argumentName));
}

同样道理我们可以扩展更多的自定义检查方法,如引入正则表达式的处理。

ArgumentCheck.Begin().NotNullOrEmpty(libFilePath, "非托管DLL路径").IsFilePath(libFilePath).CheckFileExists(libFilePath);

它的扩展函数如下所示。

        /// <summary>
        ///是否是文件路径/// </summary>
        /// <param name="validation">Validation</param>
        /// <param name="data">路径</param>
        /// <returns>Validation对象</returns>
        public static Validation IsFilePath(this Validation validation, stringdata)
{
return Check<ArgumentException>(validation, () => ValidateUtil.IsFilePath(data), string.Format(Resource.ParameterCheck_IsFilePath, data));
}
/// <summary> ///检查指定路径的文件必须存在,否则抛出<see cref="FileNotFoundException"/>异常。/// </summary> /// <param name="validation">Validation</param> /// <param name="filePath">文件路径</param> /// <exception cref="ArgumentNullException">当文件路径为null时</exception> /// <exception cref="FileNotFoundException">当文件路径不存在时</exception> /// <returns>Validation对象</returns> public static Validation CheckFileExists(this Validation validation, stringfilePath)
{
return Check<FileNotFoundException>(validation, () => File.Exists(filePath), string.Format(Resource.ParameterCheck_FileNotExists, filePath));
}

我们可以根据我们的正则表达式校验,封装更多的函数进行快速使用,如果要自定义的校验,那么就使用基础的Chek函数即可。

测试下代码使用,如下所示。

        /// <summary>
        ///应用程序的主入口点。/// </summary>
[STAThread]static void Main(string[] args)
{
ArgumentCheck.Begin().NotNull(args,
"启动参数");string test = null;
ArgumentCheck.Begin().NotNull(test,
"测试参数").NotEqual(test, "abc", "test");

这个ArgumentCheck作为公用类库的一个类,因此使用起来不需要再次引入第三方类库,也能够实现常规的校验处理,以及可以扩展自定义的参数校验,同时也是支持流式的书写方式,非常方便。

在较早博客随笔里面写过文章《
Winform开发框架之简易工作流设计
》之后,很久没有对工作流部分进行详细的介绍了,本篇继续这个主题,详细介绍其中的设计、实现及效果给大家,这个工作流在好几年前就应用在了市行业审批系统上,经过不断的改造适合更广泛的审批流程处理,从最初的Web上扩展到WInform上,并从WInform框架到混合框架上都实现了不错的处理。

1、工作流模块的表设计分析

在工作流处理表中,首先我们区分流程模板和流程实例两个部分,这个其实就是类似模板和具体文档的概念,我们一份模板可以创建很多个类似的文档,文档样式结构类似的。同理,流程模板实例为流程实例后,就是具体的一个流程表单信息了,其中流程模板和流程实例表单都包括了各个流程步骤。在流程实例的层次上,我们运行的时候,需要记录一些日志方便跟踪,如流程步骤的处理日志,流程实例表单的处理日志等这些信息。

当然实际的流程实例里面需要记录很多信息,其中流程步骤日志、申请单处理日志等信息是必须要记录的,方便我们跟踪相关的处理记录。因此工作流业务表包含多两个日志记录的表,如下所示。

一旦流程实例根据模板创建后,流程先根据模板初始化后,在处理过程还可以动态增加一些审批步骤,使得我们的处理更加弹性化。

当然,为了更好的处理流程的相关信息,还需要记录流程处理人,流程会签人、流程阅办人,以及常用审批意见等相关辅助表,以便对流程的各个处理信息进行合理处理和展示。

下面是具体表单的查看信息,包含了相关的处理步骤信息,以及相关的流程日志信息。

详细表单查看界面如下所示。

流程日志分为几个部分:申请单处理日志、申请单处理历史信息、申请单系统日志等几个部分

2、工作流步骤处理

对于一个流程处理操作,我们知道一般有审批通过、拒绝、退回到某步骤、转发到内部阅读、阅读,以及包括起草者能撤销表单呢等操作,当然如果还有一些具体的业务,可能还会有一些流程的处理才操作,不过基本上也可以归结为上面几种,只是他们每步处理的数据内容不同而已。因此审批的操作步骤分类如下所示。

如审批界面如下所示,里面包含了通过、拒绝,跳回到某步骤,增加步骤等功能集合。

WInform开发框架之工作流系列文章:

Winform开发框架之简易工作流设计


Winform开发框架中工作流模块的表设计分析


Winform开发框架中工作流模块的业务表单开发

Winform开发框架中工作流模块之审批会签操作

Winform开发框架中工作流模块之审批会签操作(2)

在我们开发工作流的时候,往往需要设计到具体业务表单信息的编辑,有些是采用动态编辑的,有些则是在开发过程中处理的,各有各的优点,动态编辑的则方便维护各种各样的表单,但是数据的绑定及处理则比较麻烦,而自定义开发的,则数据弹性很大,方便修改调整。本篇随笔基于表单的开发设计过程,介绍在工作流中如何新增一个业务表单,以便快速的实现审批业务的上线处理。

1、业务表单的基类继承

首先我们来了解一下业务表单的对应关系,一般创建一个业务流程处理,都需要有一个具体的创建业务表单的界面,以及一个查看处理表单的界面。

为了方便,我们尽可能减少代码编写,我们需要把大多数的逻辑处理放在基类实现,这样我们在新增一个业务表单的时候就可以减少很多代码编写及维护了。

如对于FrmAddApply类定义如下,我们定义一些抽象接口用于下面的业务表单实现

    /// <summary>
    ///创建申请单的窗体基类/// </summary>
    public partial classFrmAddApply : BaseForm
{
/// <summary> ///表单ID/// </summary> public string FormID { get; set; }/// <summary> ///申请单ID/// </summary> public string ApplyId { get; set; }publicFrmAddApply()
{
InitializeComponent();
}
/// <summary> ///显示数据的函数(子类必须实现)/// </summary> public virtual voidDisplayData() { }/// <summary> ///实现控件输入检查的函数(子类必须实现)/// </summary> /// <returns></returns> public virtual bool CheckInput() { return true; }/// <summary> ///编辑状态下的数据保存(子类必须实现)/// </summary> /// <returns></returns> public virtual bool SaveUpdated() { return true; }/// <summary> ///新增状态下的数据保存(子类必须实现)/// </summary> /// <returns></returns> public virtual bool SaveAddNew() { return true; }

...................

这样我们创建一个新的业务表单,只需要利用代码生成工具,生成所需要的各层框架代码,然后再生成Winform窗体代码,复制部分界面处理代码过来这个业务表单的子类即可。

下面是一个请假申请的业务表单设计,如下所示。

我们看到这个表单可以使用代码生成工具Database2Sharp快速生成后进行一定调整的,而这个编辑表单的界面,我们只需要使用自动生成的部分代码即可。

相关代码只需要复制上面的新增、更新、显示数据的代码即可。

对于查看申请单的基类FrmViewApply类,我们更加简单,我们需要把它的自定义界面控件加载出来即可。

下面是查看申请单的基类,封装了相关的处理逻辑。

    /// <summary>
    ///本窗体是通用的查看申请单界面基类。///为减少开发相关页面的工作量,只需要创建一个新窗体,并继承本窗体,然后在子窗体Form_Load函数里面,初始化对应的申请单显示控件即可。/// </summary>
    public partial classFrmViewApply : BaseDock
{
/// <summary> ///申请单ID/// </summary> public string ApplyId { get; set; }/// <summary> ///申请单自定义控件/// </summary> public BaseUserControl ApplyControl { get; set; }/// <summary> ///默认构造函数/// </summary> publicFrmViewApply()
{
InitializeComponent();
}
private void FrmViewApply_Load(objectsender, EventArgs e)
{
if (!this.DesignMode)
{
InitToolBar();
}
}
/// <summary> ///初始化申请单控件/// </summary> protected virtual voidInitApplyControl(BaseUserControl control)
{
if (control != null)
{
this.ApplyControl =control;this.ApplyControl.Dock =DockStyle.Fill;this.Controls.Add(control);
}
}
/// <summary> ///打印申请单控件内容(默认调用窗体打印)/// </summary> protected virtual voidPrintApplyControl()
{
if(this.ApplyControl != null)
{
PrintFormHelper.Print(
this.ApplyControl, false);
}
}
/// <summary> ///表单另存为/// </summary> protected virtual voidApplySaveAs()
{
}
/// <summary> ///初始化工具栏的按钮和状态/// </summary> protected virtual voidInitToolBar()
{
................
//基类实现,控制什么时候该做什么审批处理,以及一些常见按钮 }

.................

查看请假申请单的窗口就是继承这个FrmViewApply即可,如下所示。

    /// <summary>
    ///查看请假申请单的窗体/// </summary>
    public partial classFrmViewLeave : FrmViewApply
{
private LeaveControl control = null;publicFrmViewLeave()
{
InitializeComponent();
}
private void FrmViewLeave_Load(objectsender, EventArgs e)
{
//初始化控件并展示在基类窗体里面 control = newLeaveControl();
control.ApplyId
= this.ApplyId;
control.DisplayData();
base.InitApplyControl(control);
}
}

这个就是全部的窗体源码了,主要的内容我们看到是在LeaveControl这个用户控件类里面的了,

而这个控件主要就是上面编辑请假申请单的界面设计,并复制相关的显示数据代码即可。

相关界面代码如下所示。

    /// <summary>
    ///查看请假申请单的内容显示控件/// </summary>
    public partial classLeaveControl : BaseUserControl
{
/// <summary> ///申请单ID/// </summary> public string ApplyId { get; set; }publicLeaveControl()
{
InitializeComponent();

SetReadOnly();
}
/// <summary> ///设置整个窗体布局为只读并设置只读的背景颜色/// </summary> private voidSetReadOnly()
{
this.layoutControl1.OptionsView.IsReadOnly =DevExpress.Utils.DefaultBoolean.True;this.layoutControl1.Appearance.ControlReadOnly.BackColor =Color.SeaShell;
}
private void LeaveControl_Load(objectsender, EventArgs e)
{
this.applyInfoControl1.ApplyId = this.ApplyId;this.applyInfoControl1.BindData();
}
/// <summary> ///初始化数据字典/// </summary> private voidInitDictItem()
{
//初始化代码 }/// <summary> ///数据显示的函数/// </summary> public voidDisplayData()
{
InitDictItem();
//数据字典加载(公用)//由于申请单一般是用申请表单入口,而非业务数据表,因此只能传入ApplyId获取信息 if (!string.IsNullOrEmpty(ApplyId))
{
....................
}

通过上面定义的对应表单的窗体基类,可以减少我们重复编码的需要,我们只需要利用最有效率的生成界面,然后复制代码后调整即可快速生成我们所需要的不同表单界面。

每个表单我们放在一个目录上,这样我们就可以很好管理它们了。

2、业务表单的动态展示处理

上面介绍了业务表单的填写、查看两个不同的窗口,我们在申请单的审批界面里面,统一显示不同的表单,以及创建不同的业务表单界面,这种动态的处理可以实现不同业务表单的创建及显示界面。

如我的审批工作中,表单的显示界面如下所示,查看具体表单后,可以动态展示不同的业务窗口界面。

另外我们在创建业务表单的时候,根据数据库的配置信息,动态展示所有可以展示的创建入口,单击相关的按钮,可以动态调用创建对应的表单界面。

创建流程业务表单的入口如下所示。

在我的审批工作界面,动态创建对应的查看表单窗体代码如下所示。

        /// <summary>
        ///分页控件编辑项操作/// </summary>
        private void winGridViewPager1_OnEditSelected(objectsender, EventArgs e)
{
//获取记录ID和表单ID string ID = this.winGridViewPager1.gridView1.GetFocusedRowCellDisplayText("ID");string FormId = string.Concat(this.winGridViewPager1.gridView1.GetFocusedRowCellValue("FormId"));if (!string.IsNullOrEmpty(ID) && !string.IsNullOrEmpty(FormId))
{
var formInfo = BLLFactory<BLL.Form>.Instance.FindByID(FormId);if (formInfo != null && !string.IsNullOrEmpty(formInfo.ApplyWin2))
{
try{//根据配置的查看窗体,动态构建查看申请单对象 FrmViewApply dlg = Assembly.GetExecutingAssembly().CreateInstance(formInfo.ApplyWin2) asFrmViewApply;if (dlg != null)
{
dlg.ApplyId
=ID;
dlg.OnDataSaved
+= newEventHandler(dlg_OnDataSaved);if (DialogResult.OK ==dlg.ShowDialog())
{
BindData();
}
}
}
catch(Exception ex)
{
LogHelper.Error(ex);
MessageDxUtil.ShowError(ex.Message);
}
}
}
}

这个代码替代了需要手动创建不同对象的处理

    var dlg = newFrmViewAssignWork();
dlg.ApplyId
=ID;
dlg.OnDataSaved
+= newEventHandler(dlg_OnDataSaved);if (DialogResult.OK ==dlg.ShowDialog())
{
BindData();
}

同理,对于创建编辑界面,我们也可以同样的方法动态创建相关的编辑表单界面,如下代码所示。

WInform开发框架之工作流系列文章:

Winform开发框架之简易工作流设计


Winform开发框架中工作流模块的表设计分析


Winform开发框架中工作流模块的业务表单开发

Winform开发框架中工作流模块之审批会签操作

Winform开发框架中工作流模块之审批会签操作(2)

在前面介绍了框架中工作流的几个开发过程,本篇随笔重点介绍一下日常审批环节中的具体处理过程,从开始创建表单,以及各个审批、会签过程的流转过程,希望大家对其中流程的处理有一个大概的印象。

1、请假申请表单

这里介绍的具体功能,是在业务表单已经完成好,具体的使用过程,我们以请假申请单为例,在我们创建对应的流程步骤如下所示后,

我们就可以开始创建具体的业务审批流程了,在业务受理列表里面选择对应的请假申请,弹出相应的创建业务流程表单窗体。

单击【发送】按钮,会要求选择下一步的审批人,如下所示。

这样对应的审批人员进去后,在我的待办业务列表里面,就可以看到刚才的表单了,双击可以进行查看,以及相关的审批处理工作。

单击表单左上角的工具栏按钮【审批】,可以进行流程的审批处理操作了。

在处理完成相关的审批操作后,流程就会自动流转到了表单定义的下一流程人了,这里为了方便选择了管理员的角色来处理而已。

流程人登陆后查看具体业务申请单后,可以看到前面的审批意见等信息,如下所示。

继续【审批】处理环节,这样两个环节的流程就处理完成了。

最后我们查看完成的业务申请单,可以看到相关的审批信息了。

当然也可以查看具体的业务流程日志。

整个流程步骤全部完成了。

2、报销申请单

为了增加复杂一点的表单我们引入一个含有主从表的业务表单,明细表包括报销的具体列表内容。

具体的业务表单设计如下所示。

首先我们定义好业务流程,如下所示。

创建一个业务表单,如下所示。

其中几个流程的审批和上面的处理操作差不多,最后完成表单后,那么这个总共参与的步骤是三个步骤,包括了总经理审批环节了,和下面的表单过程类似,界面效果如下所示。

如果是报销费用少于条件值(3000)的,那么审批的环节就不包括总经理审批环节了,如下表单所示。

以上就是两类表单,一个是普通的单表表单、一个是包含明细表的主从表表单的流程处理,以及步骤中包含相关的过滤条件,满足条件的则包含该条件的审批处理,否则系统自动跳过这个步骤,直接跳转到下一步或者完成整个申请单的处理。

有时候为了方便查看对应的业务表单,一般提供一个所有表单的入口给管理查看 ,方便进行维护管理,如下所示。

管理员可以在这里对业务表单进行删除、撤回等特殊处理,另外,我们也可以提供一个业务数据的入口方便查询统计,如下所示的界面。

WInform开发框架之工作流系列文章:

Winform开发框架之简易工作流设计


Winform开发框架中工作流模块的表设计分析


Winform开发框架中工作流模块的业务表单开发

Winform开发框架中工作流模块之审批会签操作

Winform开发框架中工作流模块之审批会签操作(2)

前面随笔介绍了请假申请单和报销申请单两个不同的业务表单的流程处理,一个是单表信息,一个包含明细的主从表信息,后者包含了条件流程的处理,在流程审批中,一般还有一种流程处理就是会签的操作,会签处理是几个审批步骤中审批人同时处理是否通过的,一般同时通过即为通过。本篇随笔介绍工作流中的会签处理过程。

1、会签流程定义

会签是指创建一个或多个子流程供相关人员进行审批,等待全部人员完成处理后再次回到主流程上,然后决定是否继续流转到下一个流程步骤上去,一般的申请单的主流程如下所示。

这里设置的会签处理就是其中一个步骤,一旦会签处理步骤发起会签,就会构建多个可供审批的子流程了,如下所示。

在会签发起的步骤,指定参与具体流程会签审批的人员,然后流程则会流转到不同人员进行相关的处理【待办事项】。

我在工作流中定义会签完成后,由会签发起人审核(会签结果审核),决定是否进入下一步流程,在审核过程中决定如何处理这个申请单。

在流程定义里面,我们创建一个会签的流程步骤,我们以请假单为例,加入我们要求请假需要由各组长会签通过,然后在继续下面的部门审批、总经理审批步骤,如下所示。

增加会签后的流程步骤如下所示。

完成后可以在流程步骤列表中看到会签的步骤了,如下所示。

2、会签流程处理

了解了会签的处理过程,并完成了上面的会签定义后,我们创建一个请假申请单,用来发起会签处理,介绍会签的步骤说明。

完成请假单后提交给相关处理人,处理人员在待办事项中查看申请单,如下界面所示。

那么他会发起【发起会签】的处理操作,把相关的投票权发送给各个组长进行会签处理。

会签发起后,各个参与会签的人员在【待办事项】里面处理会签意见,如下所示。

各个待审批的人员进行处理后,最后返回给会签发起人决定是否进入下一步流程,如下所示。

我们查看相关的流程日志,看到会签的流程步骤已经完成了,其他步骤还需要进行处理。

这样这个会签流程就算整体完成了,剩下的就是其他步骤的处理,按正常的审批处理即可。

WInform开发框架之工作流系列文章:

Winform开发框架之简易工作流设计


Winform开发框架中工作流模块的表设计分析


Winform开发框架中工作流模块的业务表单开发

Winform开发框架中工作流模块之审批会签操作

Winform开发框架中工作流模块之审批会签操作(2)