2

我有一个正在使用的 WCF 服务,并且到目前为止做得很好。

然而,在我们流量很大的生产系统上,我注意到在内存逐渐一致上升和下降之后(之间的时间逐渐拉长,增量逐渐增加),内存消耗呈上升趋势。

我想知道这是否可能是由于我使用 DAL Web 服务的方式:

例如:

    public static int GetUserTypeFromProfileID(int profileID)
    {
        try
        {
            memberServiceClient = new MemberServiceClient();                                // connect to the data service
            return memberServiceClient.GetUserTypeFromProfileID(profileID);                 // get the profileID associated with the sessionID
        }
        catch (Exception ex)
        {
            ErrorLogging.Instance.Fatal(ex);
            return 0;
        }
    }

如果我将其更改为以下内容,请使用以下using语句:

    public static int GetProfileIDFromSessionID(string sessionID)
    {
        try
        {
            using (memberServiceClient = new MemberServiceClient())                                // connect to the data service
            {
                return memberServiceClient.GetProfileIDFromSessionID(sessionID);                // get the profileID associated with the sessionID
            }

        }
        catch (Exception ex)
        {
            ErrorLogging.Instance.Fatal(ex);
            return 0;
        }
    }

在部分内执行返回是否是一种良好的形式using

4

1 回答 1

2

我相信使用语句没有特定于 WCF 的内容。它会在返回值之前处理您的 MemberServiceClient。

但是 WCF 服务客户端上的 Dispose() 方法调用内部的Close()方法,这可能会引发异常。所以最好直接调用 Close() 方法。当发生异常时,您还应该调用Abort()方法。这是推荐的实现。

var result;
try
{
    memberServiceClient = new MemberServiceClient();
    result = memberServiceClient.GetUserTypeFromProfileID(profileID);
    memberServiceClient.Close();
}
catch (FaultException e)
{
    //handle exception
    memberServiceClient.Abort();
}
catch (CommunicationException e)
{
    //handle exception
    memberServiceClient.Abort();
}
catch (TimeoutException e)
{
    //handle exception
    memberServiceClient.Abort();
}

注意:我编写了一个简单的基类来处理这些细节。它在NuGet上。

更新:

以下是 WcfClientBase 请求的示例:

public class MemberServiceManager : ServiceClientBase<MemberServiceClient>
{
    public int GetUserTypeFromProfileID(int profileID)
    {
        //makes a call to GetUserTypeFromProfileID operation, closes the channel and handles the exceptions
        //you may want to implement another base class for overriding exception handling methods
        //return value will be default of return type if any exceptions occur
        return PerformServiceOperation(item => item.GetUserTypeFromProfileID(profileID));
    }

    //or you can manually check if any exceptions occured with this overload
    public bool TryGetUserTypeFromProfileID(int profileID, out int userType)
    {
        return TryPerformServiceOperation(item => item.GetUserTypeFromProfileID(profileID), out userType);
    }


    //these exception handling methods should be overriden in another common subclass
    //they re-throw exceptions by default
    protected override void HandleCommunicationException(CommunicationException exception)
    {
        Console.WriteLine(exception.Message);
    }

    protected override void HandleFaultException(FaultException exception)
    {
        Console.WriteLine(exception.Message);
    }

    protected override void HandleTimeoutException(TimeoutException exception)
    {
        Console.WriteLine(exception.Message);
    }

}

你也可以看看GitHub 上的源代码

于 2012-06-26T20:40:46.547 回答