SharePoint工作流开发点滴(4) – 工作流中的自定义类与内部错误

最近在开发SharePoint工作流总是发生一个错误:工作流开始之后便显示“已完成”或者开始之后报错“内部错误”。查看当时的日志,发现下面的段落:

02/06/2007 10:31:03.92         w3wp.exe (0x0758)       

0x0F3C        Windows SharePoint Services         Workflow Infrastructure         72eo        Unexpected        

DehydrateInstance: System.Runtime.Serialization.SerializationException: 在分析完成之前就遇到流结尾。

在 System.Runtime.Serialization.Formatters.Binary.__BinaryParser.Run()

在 System.Runtime.Serialization.Formatters.Binary.ObjectReader.Deserialize(HeaderHandler handler, __BinaryParser serParser, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)

在 System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Deserialize(Stream serializationStream, HeaderHandler handler, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)

在 System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Deserialize(Stream serializationStream)

在 System.Workflow.ComponentModel.Activity.Load(Stream stream, Activity outerActivity, IFormatter formatter)…        

02/06/2007 10:31:03.92*        w3wp.exe (0x0758)         

0x0F3C        Windows SharePoint Services         Workflow Infrastructure         72eo        Unexpected        …

在 System.Workflow.ComponentModel.Activity.Load(Stream stream, Activity outerActivity)

在 System.Workflow.Runtime.Hosting.WorkflowPersistenceService.RestoreFromDefaultSerializedForm(Byte[] activityBytes, Activity outerActivity)

在 Microsoft.SharePoint.Workflow.SPWinOePersistenceService.LoadWorkflowInstanceState(Guid instanceId)

在 System.Workflow.Runtime.WorkflowRuntime.InitializeExecutor(Guid instanceId, CreationContext context, WorkflowExecutor executor, WorkflowInstance workflowInstance)

在 System.Workflow.Runtime.WorkflowRuntime.Load(Guid key, CreationContext context, WorkflowInstance workflowInstance) 在 System.Workflow.Runtime.WorkflowRuntime.GetWorkflow(Guid instanceId)

在 Microsoft.SharePoint.Workflow.SPWinOeHostServices.DehydrateInstance(SPWorkflowInstance wo…        

02/06/2007 10:31:03.92*        w3wp.exe (0x0758)         

0x0F3C        Windows SharePoint Services         Workflow Infrastructure         72eo        Unexpected        …rkflow)

02/06/2007 10:31:03.93         w3wp.exe (0x0758)         

0x0F3C        Windows SharePoint Services         Workflow Infrastructure         88xr        Unexpected        WinWF Internal Error, terminating workflow Id# 472dae03-5465-4f04-876f-d4cc4caa902a

 

看里边最长的一段中文描述:“在分析完成之前就遇到流结尾”,如果是SharePoint英文版,这段错误信息应该是“End of Stream encountered before parsing was completed”。

也就是说Workflow Runtime根本就没有完整的分析完整个流程。

再看这句中文之前的英文:“DehydrateInstance: System.Runtime.Serialization.SerializationException”。

原来工作流是在钝化实例的时候发生了序列化异常。

回想一下工作流的持久性,Workflow Runtime会把空闲的工作流数据序列化为XML形式,然后把工作流实例从内存中清除,等到需要的时候再将其反序列化加载到内存。

会不会是因为我在工作流项目中添加了自定义类,而这个类又不支持序列化,所以导致工作流序列化失败?

在工作流中使用InfoPath Initiation(或者Association)表单时需要为其生成一个类,观察这个类,发现这个用XSD生成的类有如下特性来修饰:

[System.CodeDom.Compiler.GeneratedCodeAttribute("xsd", "2.0.50727.42")]
[System.SerializableAttribute()]
[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(AnonymousType=true)]
[System.Xml.Serialization.XmlRootAttribute(Namespace="http://schemas.microsoft.com/office/infopath/2003/myXSD/2007-01-30T13:00:28", IsNullable=false)]

下面来逐行分析一下:

[System.CodeDom.Compiler.GeneratedCodeAttribute("xsd", "2.0.50727.42")]

表示这个类是由XSD工具生成的.

[System.SerializableAttribute()]

表示这个类可以被序列化,我想关键就在这里

[System.Diagnostics.DebuggerStepThroughAttribute()]

表示调试器会自动忽略被修饰的类内部的断点

[System.ComponentModel.DesignerCategoryAttribute("code")]

表示设计器的类别是“code”

[System.Xml.Serialization.XmlTypeAttribute(AnonymousType=true)]

表示系列化时生成的XSD架构是匿名类型

[System.Xml.Serialization.XmlRootAttribute(Namespace="http://schemas.microsoft.com/office/infopath/2003/myXSD/2007-01-30T13:00:28", IsNullable=false)]

设置序列化时根元素的命名空间

删去或修改我们不需要的特性(比如第一项),将其插入到我们的自定义类中,工作流就正常了。

其实只需要将自定义类标识为[Serializable]即可。

其实不仅是自定义类,只要是在工作流类中的类级别变量,都必须实现了ISerializable的,而方法内的局部变量则可以不支持序列化。否则工作流在 进行钝化是无法序列化某些变量,也会导致这个错误。比如在工作流累中定义了一个XmlDocument,就会发生此错误。

2008.12.25最后更新

5 Comments

  1. 端木

    楼主,借贵宝地问一个问题:

    我用WSS3.0建了一个网站,主要做文档管理。可现有文档库模板都是直接提供文档下载。即在文档列表上一点击,就直接弹出下载对话框。

    我希望的是:
    在页面最左边是个树形的层次结构目录,里面可以象资源管理器那样收缩。选中一个子目录后,在页面中间部分显示该目录的标题列表,点击标题后进入单个的文件页面,该页面上先是文档简介,再是提供附件下载,最后是文档评论。

    即如下规划

    -------------------------
    目录1 | 文章1.1.1
    子目录1.1 | 文章1.1.2
    子目录1.2 | 文章1.1.3
    目录2 | 文章1.1.4
    子目录2.1 |
    子目录2.2 |

    点击文章1.1.1后,如下界面:
    -------------------------
    目录1 | 文章1.1.1标题
    子目录1.1 | 文章1.1.2简介:
    子目录1.2 |
    目录2 | 文章附件下载链接
    子目录2.1 |
    子目录2.2 | 评论框

    如此简单的需求,sharepoint难道都搞不定?
    目前���用列表模板可以实现右边的功能
    启用树视图能实现左边的功能

    但这两者无法兼得!即列表没有层次化目录结构,文档库没有文档描述功能

    谁有解决办法?

  2. wjinwei

    你好!
    我现在正在做sharepoint 工作流相关方面的东西.
    我现在有一个问题想请教一下你 :

    在我的工作流程中有一个ReplicatorActivity,它里面包含了一个WssTaskActivity.流程部署到sharepoint上后,一启动“工作流程状态就是已完成”,但是任务创建都已成功!

    这个问题好像跟你在该篇文章中的描述一样,肯请帮忙解决一下,不甚感激!

    wangjinwei2003@hotmail.com

  3. 我同样碰到这个问题,我使用自定义的aspx表单,已经将所有的所有无关的类都删掉,依旧出现这个问题。

发表评论

电子邮件地址不会被公开。 必填项已用*标注