WCF中操作的分界于调用顺序和会话的释放操作分界实例停止

时间:2022-05-03
本文章向大家介绍WCF中操作的分界于调用顺序和会话的释放操作分界实例停止,主要内容包括操作分界、实例停止、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。

操作分界

在WCF操作契约的设计中,有时会有一些调用顺序的业务,有的操作不能最先调用,有的操作必须最后调用,比如在从一个箱子里拿出一件东西的时候,必须先要执行打开箱子的操作,而关上箱子的操作应该在一切工作完成之后再被执行。

public interface Box
{
    void Open(int boxId);
    int GetTotalFrenchletter();
    void Close();
}

为了解决这种实际需求,WCF在OperationContractAttribute中提供了IsInitiating和IsTerminating属性,IsInitiating的默认值为true,表示当前操作可以被第一个调用,IsTerminating属性默认为false,表示在这个方法执行完之后,服务对象不会被释放。可以通过这两个属性来控制上文中提供的需求。

此外,在修改了默认值之后,在装在服务的时候,WCF会去验证服务契约是否被定义为SessionMode.Required,如果不是,WCF会抛出InvalidOperationExample异常。

上文中的契约定义可以被重新设计:

[ServiceContract(SessionMode=SessionMode.Required)]
public interface Box
{
    [OperationContract(IsInitiating=true,IsTerminating=false)]
    void Open(int boxId);
    [OperationContract(IsInitiating = false, IsTerminating = false)]
    int GetTotalFrenchletter();
    [OperationContract(IsInitiating = false, IsTerminating = true)]
    void Close();
}

Open方法上的Attribute和不去添加它是一样的含义,只不过看起来更加清晰一点

有一点需要注意的是,参照以上的契约定义,在Close调用执行完之后,WCF会异步的释放对象并且关闭会话,客户端将不能再通过当前代理调用服务中的操作。


实例停止

在服务的生命周期中,上下文是一直伴随着服务实例的创建于释放的整个过程的,然后处于某些目的,WCF也提供了分离两者的选项,允许服务实例被单独的停止。方法就是设置OperationBehaviorAttribute的ReleaseInstanceMode属性,它是一个名为ReleaseInstanceMode的枚举类型,包含了AfterCall、BeforeCall、BeforeAndAfterCall和None四个值,默认值为None。

BeforeCall:将在调用当前操作之前,WCF会释放当前的服务实例,然后创建一个新的实例取代它,然后在这个新的实例上调用方法;

AfterCall:将在调用当前操作之后释放当前服务实例;

BefireAndAfterCall:它是对前两种设置的一种补充,OperationBehavior如果应用了这个值,那么当前方法可以在标记了BeforeCall或者None的方法之后调用,也可以在在标记了AfterCall或者None的方法之后调用.

而在上文中的示例中,我们可以做如下的定义

public class Box : IBox
{
    public void Open(int boxId)
    {
        throw new NotImplementedException();
    }

    public int GetTotalFrenchletter()
    {
        throw new NotImplementedException();
    }

    [OperationBehavior(ReleaseInstanceMode=ReleaseInstanceMode.AfterCall)]
    public void Close()
    {
        lockBox();
    }
}

即使如此,WCF仍然提供了一个直接停止服务实例的方法,以满足在以上设置中都没有找到一个完美的适用于您的需求的解决方案。然后再非不得已的情况下,应该尽量不要使用它,因为它破坏了业务逻辑和服务本身生命周期的分离。

方法很简单,在OperationContext中存在InstanceContext,而这个属性包含一个ReleaseServiceInstance方法,在这个方法调用之后服务将会被释放:

[OperationBehavior(ReleaseInstanceMode=ReleaseInstanceMode.AfterCall)]
public void Close()
{
    lockBox();
    OperationContext.Current.InstanceContext.ReleaseServiceInstance();
}

然后,以上所述的技术,仅仅是WCF提供的一些对于特殊需求的优化技术,通常情况下不必去可以使用它。