Workflow笔记3——BookMark和持久化

时间:2021-07-12 18:40:37

BookMark

我们在平时的工作流使用中,并不是直接这样一气呵成将整个工作流直接走完的,通常一个流程到了某一个节点,该流程节点的操作人,可能并不会马上去处理该流程,而只有当处理人处理了该流程,流程才会继续往下走。对于不同流程节点的处理人,他所能处理的是不同的流程节点。

怎么让流程停下来,等待其他用户对流程进行参与处理。并且必须能够保证流程能够在原有的处理现场情况进行保存,而且能够对流程进行继续启动和处理?那就是书签。

就好像我们看书,我们需要书签来标识,我现在已经看到哪个地方了,工作流也是一样的,我需要通过书签,来确定不同角色的人能处理的是哪一个流程。

要定义具备Bookmark的Activit,可从 NativeActivity继承,override [Execute 方法],使用[NativeActivityContext.CreateBookmark方法]添加Bookmark,需要override [ CanInduceIdle 属性],使其返回值为[True]。

1、在项目WindowsWorkFlowApp中,新建“代码活动” BookMarkCodeActivity

修改继承类为NativeActivity,Execute方法的参数类型变为NativeActivityContext类型了。代码如下:

    public sealed class BookMarkCodeActivity : NativeActivity
{
// 定义一个字符串类型的活动输入参数
public InArgument<string> BookMarkName { get; set; }
//定义一个输出参数,用来做流程判断,相当于模拟用户处理流程节点的操作
public OutArgument<int> Num { get; set; } // 创建一个BookMark,让流程停下来
protected override void Execute(NativeActivityContext context)
{
// 1.获取BookMark名称
string strBookMarkName = context.GetValue(BookMarkName);
// 2.创建BookMark
context.CreateBookmark(strBookMarkName,new BookmarkCallback(PreExecuteWorkFlow));
}
/// <summary>
/// 注意,一定要记得注意重写此属性,并返回true,否则后面运行会报错
/// </summary>
protected override bool CanInduceIdle
{
get
{
return true;// base.CanInduceIdle;
}
}
/// <summary>
/// 继续执行下一个状态前,必须先执行该方法。
/// </summary>
/// <param name="context"></param>
/// <param name="bookmark">书签</param>
/// <param name="value">传递过来的值</param>
public void PreExecuteWorkFlow(NativeActivityContext context, Bookmark bookmark, object value)
{
context.SetValue(Num, Convert.ToInt32(value));
}
}

2、生成项目WindowsWorkFlowApp

3、双击State1打开,将代码活动添加到State1中,并创建变量Vnum。

Workflow笔记3——BookMark和持久化

4、创建输入参数InputBookMarkName

Workflow笔记3——BookMark和持久化

5、改造Form1窗体

Workflow笔记3——BookMark和持久化

修改启动工作流的代码:

将以WorkflowApplication app;提取到类下面。

            app = new WorkflowApplication(new Activity1(), new Dictionary<string, object>() {
{"InputName","神刀张三"},{"InputBookMarkName",txtBookMarkName.Text}
});
app.Idle = delegate(WorkflowApplicationIdleEventArgs er)
{
Console.WriteLine("工作流 {0} 空闲.", er.InstanceId);
syncEvent.Set(); //这里要唤醒,不让的话,当创建了一个书签之后,界面就卡死了。
};

为“继续执行”按钮添加代码

        //唤醒BookMark执行流程
private void btnContinue_Click(object sender, EventArgs e)
{
//这里会调用PreExecuteWorkFlow方法,并将txtNum的值传过去
app.ResumeBookmark(txtBookMarkName.Text, int.Parse(txtNum.Text));
}

6、双击T1进行修改,添加条件判断

Workflow笔记3——BookMark和持久化

假设VNum变量的值等于5,则继续往下执行State2。

7、添加T3,当VNum变量的值不等于5,再回到State1。

Workflow笔记3——BookMark和持久化

双击T3,添加条件

Workflow笔记3——BookMark和持久化

8、运行结果如下:

Workflow笔记3——BookMark和持久化

工作流持久化

持久化:工作流持久性是指独立于进程或计算机信息持续捕获工作流实例的状态。持久存储化,例如用磁盘进行存储,光盘存储等持久化的存储数据就是持久化。

为何持久化?财务审批中,提交了审批邀请后,财务总监过了一周才进行审批。那么数据必须进行持久化的保存,等待流程的继续相关处理。
工作流在长时间运行时难免会遇到一些问题,许多业务逻辑需要花费数日、数周乃至数月。在这段时间中,我们不能让工作流实例一直驻留在内存中。

工作流什么时候进行持久化?

  • 当 TransactionScope 活动完成时或 TransactedReceiveScope 活动完成时。
  • 当工作流实例变为空闲状态,且对工作流主机设置了 WorkflowIdleBehavior 时。 例如,当使用消息传递活动或 Delay 活动时会发生此情况。
  • 当 WorkflowApplication 变为空闲状态且将应用程序的 PersistableIdle 属性设置为 PersistableIdleAction.Persist 时。
  • 当指示主机应用程序持久化或卸载工作流实例时。
  • 当终止工作流实例或工作流实例结束时。
  • 当执行 Persist 活动时。
  • 当使用 Windows Workflow Foundation 的早期版本开发的工作流实例在可互操作执行过程中遇到持久点时。

1、通过创建一个数据库来持久保存工作流实例。新建数据库WorkFlowDB:

CREATE DATABASE [WorkFlowDB]
CONTAINMENT = NONE
ON PRIMARY
( NAME = N'WorkFlowDB', FILENAME = N'G:\DataBase\WorkFlowDB.mdf' , SIZE = 5120KB , MAXSIZE = UNLIMITED, FILEGROWTH = 1024KB )
LOG ON
( NAME = N'WorkFlowDB_log', FILENAME = N'G:\DataBase\WorkFlowDB_log.ldf' , SIZE = 2048KB , MAXSIZE = 2048GB , FILEGROWTH = 10%)
GO

2、然后新建表来存储工作流的实例数据,如何新建表?

到%WINDIR%\Microsoft.NET\Framework\v4.xxx\SQL\EN 文件夹下面去寻找脚本,按Win+R,运行%WINDIR%\Microsoft.NET\Framework

Workflow笔记3——BookMark和持久化Workflow笔记3——BookMark和持久化

找到这两个SQL脚本之后,在数据库WorkFlowDB中首先运行 SqlWorkflowInstanceStoreSchema.sql 文件,然后运行 SqlWorkflowInstanceStoreLogic.sql 文件。执行完成之后,就会在数据库WorkFlowDB中新建如下表。

Workflow笔记3——BookMark和持久化

InstancesTable表就是用来存储工作流实例的表。

3、在项目WindowsWorkFlowApp中,添加如下两个程序集的引用

Workflow笔记3——BookMark和持久化

4、修改工作流启动代码

引入命名空间

using System.Activities.DurableInstancing;  

修改btnStartWorkFlow_Click代码:

            SqlWorkflowInstanceStore store =
new SqlWorkflowInstanceStore(@"Server=.\MSSQLSERVER2012;database=WorkFlowDB;uid=sa;pwd=yujie1127);
app.InstanceStore = store;

只需要这两行代码,就可以执行持久化工作。那么当下次重新打开工作流的时候,我需要从数据库中找到是那一条工作流实例数据,为了演示简单,我这里就将工作流实例的主键直接放到From窗体界面展示,而通常在工作中,我们是会用数据表来专门存储这些数据信息的。

5、改造Form1代码,修改btnContinue_Click

using System;
using System.Activities;
using System.Collections.Generic;
using System.Threading;
using System.Windows.Forms; using System.Activities.DurableInstancing; namespace WindowsWorkFlowApp
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
static readonly string ConnStr=@"Server=.\MSSQLSERVER2012;database=WorkFlowDB;uid=sa;pwd=yujie1127";
//WorkflowApplication app;
AutoResetEvent syncEvent = new AutoResetEvent(false);
private void btnStartWorkFlow_Click(object sender, EventArgs e)
{
WorkflowApplication app = new WorkflowApplication(new Activity1(), new Dictionary<string, object>() {
{"InputName","神刀张三"},{"InputBookMarkName",txtBookMarkName.Text}
}); SqlWorkflowInstanceStore store =
new SqlWorkflowInstanceStore(ConnStr);
app.InstanceStore = store;
txtID.Text = app.Id.ToString(); WorkFlowEvent(app, syncEvent); app.Run(); syncEvent.WaitOne();
} private static void WorkFlowEvent(WorkflowApplication app, AutoResetEvent syncEvent)
{
#region 工作流生命周期事件
app.Unloaded = delegate(WorkflowApplicationEventArgs er)
{
Console.WriteLine("工作流 {0} 卸载.", er.InstanceId);
};
app.Completed = delegate(WorkflowApplicationCompletedEventArgs er)
{
Console.WriteLine("工作流 {0} 完成.", er.InstanceId);
syncEvent.Set();
};
app.Aborted = delegate(WorkflowApplicationAbortedEventArgs er)
{
Console.WriteLine("工作流 {0} 终止.", er.InstanceId);
};
app.Idle = delegate(WorkflowApplicationIdleEventArgs er)
{
Console.WriteLine("工作流 {0} 空闲.", er.InstanceId);
syncEvent.Set(); //这里要唤醒,不让的话,当创建了一个书签之后,界面就卡死了。
};
app.PersistableIdle = delegate(WorkflowApplicationIdleEventArgs er)
{
Console.WriteLine("持久化");
return PersistableIdleAction.Unload;
};
app.OnUnhandledException = delegate(WorkflowApplicationUnhandledExceptionEventArgs er)
{
Console.WriteLine("OnUnhandledException in Workflow {0}\n{1}",
er.InstanceId, er.UnhandledException.Message);
return UnhandledExceptionAction.Terminate;
};
#endregion
} //唤醒BookMark执行流程
private void btnContinue_Click(object sender, EventArgs e)
{
#region old code
//这里会调用PreExecuteWorkFlow方法,并将txtNum的值传过去
//app.ResumeBookmark(txtBookMarkName.Text, int.Parse(txtNum.Text));
#endregion WorkflowApplication app = new WorkflowApplication(new Activity1()); SqlWorkflowInstanceStore store =
new SqlWorkflowInstanceStore(ConnStr);
app.InstanceStore = store; WorkFlowEvent(app, syncEvent); app.Load(Guid.Parse(txtID.Text)); //加载工作流实例
//继续执行此工作流实例
app.ResumeBookmark(txtBookMarkName.Text, int.Parse(txtNum.Text));
}
}
}

Workflow笔记3——BookMark和持久化

6、我们看数据表中已经多了一条工作流实例数据

Workflow笔记3——BookMark和持久化

7、然后关闭应用程序,再重新启动

从数据库中找到这个ID,然后填写上。

Workflow笔记3——BookMark和持久化

我们看到整个工作流执行完成了,在来看数据表中的工作流实例数据已经删除了。

Workflow笔记3——BookMark和持久化

源码下载:WorkflowConsoleApp3.zip

Workflow笔记3——BookMark和持久化的更多相关文章

  1. Workflow笔记2——状态机工作流

    状态机工作流 在上一节Workflow笔记1——工作流介绍中,介绍的是流程图工作流,后来微软又推出了状态机工作流,它比流程图功能更加强大. 状态机工作流:就是将工作流系统中的所有的工作节点都可以看做成 ...

  2. Workflow笔记2——状态机工作流(转)

    出处:http://www.cnblogs.com/jiekzou/p/6192813.html 在上一节Workflow笔记1——工作流介绍中,介绍的是流程图工作流,后来微软又推出了状态机工作流,它 ...

  3. hibernate学习笔记(2)持久化类测试

    持久化类的创建: 创建一个共有的不带参数的构造方法: public void Students(){ } 创建一个带参数的构造方法: (快捷键创建) 生成get,set方法: *可以不用此方法创建持久 ...

  4. 11&sol;6笔记 补充(Redis持久化,RDB&amp&semi;&amp&semi;AOF)

    11/6补充笔记 修改redis-6379.conf里面的save10秒2个数据发生改变 (save 10 2) 修改一次数据不发生改变,修改2次数据才发生改变 继续修改数据,发现还是一样的规律 增删 ...

  5. Redis6&period;x学习笔记(二)持久化之RDB

    前言 最近学习Redis6.x,特做笔记以备忘,与大家共学.课程是从私塾在线下载的,他们把架构师课程都放出来了,大家可以去下载学习,不要钱的,地址是http://t.hk.uy/eK7,课程很不错,值 ...

  6. Workflow笔记1——工作流介绍

    什么是工作流? 工作流(Workflow),是对工作流程及其各操作步骤之间业务规则的抽象.概括.描述.BPM:是Business Process Management的英文字母缩写.即业务流程管理,是 ...

  7. docker学习笔记(四)-持久化数据,安装docker-compose

    Docker 持久化数据 实战 compose 安装docker-compose

  8. K8S学习笔记之Kubernetes数据持久化方案

    在开始介绍k8s持久化存储前,我们有必要了解一下k8s的emptydir和hostpath.configmap以及secret的机制和用途. 0x00 Emptydir EmptyDir是一个空目录, ...

  9. &lbrack;学习笔记&rsqb;FHQ-Treap及其可持久化

    感觉范浩强真的巨 博主只刷了模板所以就讲基础 fhq-treap 又形象的称为非旋转treap 顾名思义 保留了treap的随机数堆的特点,并以此作为复杂度正确的条件 并且所有的实现不用旋转! 思路自 ...

随机推荐

  1. 【转】Caffe初试(九)solver及其设置

    solver算是caffe的核心的核心,它协调着整个模型的运作.caffe程序运行必带的一个参数就是solver配置文件.运行代码一般为 #caffe train --solver=*_solver. ...

  2. Hibernate POJO在序列化(JSON)时遇到的若干问题

    假设某 POJO 有属性如下: private Set<User> users = new HashSet<>(0); @OneToMany(fetch = FetchType ...

  3. 《超级IP》:伪理论,没能比现有的市场营销理论更高明,只敢勉强去解释已经发生的事情,不敢去预测未来。2星。

    超级IP是作者造出来的一个词.作者尝试把“超级IP”作为一种理论来解释2015年以来的各种网红现象.读完全书后,我的感觉是这个理论不怎么样: 1:作者完全不提现有的市场营销理论.我的问题是:现有的理论 ...

  4. php get set方法深入理解

    在类当中,设计通用的set和get方法,可以简化对属性的读写,这种方法不同于针对于独立的属性的普通的get和set方法,后者针对每个属性,都必须提供一对方法,前者针对所有属性,因此,可以看作是批量定义 ...

  5. PLSQL&lowbar;性能优化工具系列02&lowbar;SQL Tuning Health-Check Script &lpar;SQLHC&rpar;

    2014-08-23 Created By BaoXinjian

  6. hibernate关联对象的增删改查------增

    本文可作为,北京尚学堂马士兵hibernate课程的学习笔记. 这一节,我们看看hibernate关联关系的增删改查 就关联关系而已,咱们在上一节已经提了很多了,一对多,多对一,单向,双向... 其实 ...

  7. Git使用入门笔记

    1. 创建并初始化一个 代码仓库 (repository) $ git init 2.查看当前状态 $ git status 3. 将修改后的文件推入缓冲区 $ git add <filenam ...

  8. 感知机、logistic回归 损失函数对比探讨

    感知机.logistic回归 损失函数对比探讨 感知机 假如数据集是线性可分的,感知机学习的目标是求得一个能够将正负样本完全分开的分隔超平面 \(wx+b=0\) .其学习策略为,定义(经验)损失函数 ...

  9. 用GraphX分析伴生网络(二)

    8. 过滤噪声边 在当前的伴生关系中,边的权重是基于一对概念同时出现在一篇论文中的频率来计算的.这种简单的权重机制的问题在于:它并没有对一对概念同时出现的原因加以区分,有时一对概念同时出现是由于它们具 ...

  10. JS进阶之---执行上下文,变量对象,变量提升

    一.结构顺序大体介绍 JavaScript代码的整个执行过程,分为两个阶段,代码编译阶段与代码执行阶段. 编译阶段由编译器完成,将代码翻译成可执行代码,这个阶段作用域规则会确定. 执行阶段由引擎完成, ...