4

1.)我有一个主方法处理,它将字符串作为参数,并且该字符串包含一些 x 数量的任务。

2.) 我有另一种方法 Status,它通过使用两个变量 TotalTests 和 CurrentTest 来跟踪第一种方法。每次都会在第一种方法(处理)中循环修改。

3.) 当多个客户端并行调用我的 Web 服务以通过传递字符串来调用处理方法时,具有不同任务的处理将需要更多时间。因此,同时客户端将使用第二个线程调用 Web 服务中的 Status 方法来获取第一个方法的状态。

4.)当第 3 点完成时,所有客户端都应该并行获取变量(TotalTests,CurrentTest),而不会与其他客户端请求混淆。

5.) 当我将它们设置为静态时,我在下面提供的代码会使所有客户端的变量结果混合在一起。如果我删除了变量的静态变量,那么客户端只会得到这两个变量的全 0,我无法修复它。请看下面的代码。

[ServiceBehavior(InstanceContextMode = InstanceContextMode.PerSession)]
public class Service1 : IService1
{
    public int TotalTests = 0;
    public int CurrentTest = 0;

    public string Processing(string OriginalXmlString)
    {
                XmlDocument XmlDoc = new XmlDocument();
                XmlDoc.LoadXml(OriginalXmlString);
                this.TotalTests = XmlDoc.GetElementsByTagName("TestScenario").Count;  //finding the count of total test scenarios in the given xml string
                this.CurrentTest = 0;
                while(i<10)
                {
                        ++this.CurrentTest;
                         i++;
                }
    }

    public string Status()
    {
        return (this.TotalTests + ";" + this.CurrentTest);
    }
}

服务器配置

<wsHttpBinding>
    <binding name="WSHttpBinding_IService1" closeTimeout="00:10:00"
      openTimeout="00:10:00" receiveTimeout="00:10:00" sendTimeout="00:10:00"
      bypassProxyOnLocal="false" transactionFlow="false" hostNameComparisonMode="StrongWildcard"
      maxBufferPoolSize="524288" maxReceivedMessageSize="2147483647"
      messageEncoding="Text" textEncoding="utf-8" useDefaultWebProxy="true"
      allowCookies="false">
      <readerQuotas maxDepth="2147483647" maxStringContentLength="2147483647"
        maxArrayLength="2147483647" maxBytesPerRead="2147483647" maxNameTableCharCount="2147483647" />
      <reliableSession ordered="true" inactivityTimeout="00:10:00"
        enabled="true" />
      <security mode="Message">
        <transport clientCredentialType="Windows" proxyCredentialType="None"
          realm="" />
        <message clientCredentialType="Windows" negotiateServiceCredential="true"
          algorithmSuite="Default" establishSecurityContext="true" />
      </security>
    </binding>
  </wsHttpBinding>

客户端配置

<wsHttpBinding>
            <binding name="WSHttpBinding_IService1" closeTimeout="00:10:00"
                openTimeout="00:10:00" receiveTimeout="00:10:00" sendTimeout="00:10:00"
                bypassProxyOnLocal="false" transactionFlow="false" hostNameComparisonMode="StrongWildcard"
                maxBufferPoolSize="524288" maxReceivedMessageSize="2147483647"
                messageEncoding="Text" textEncoding="utf-8" useDefaultWebProxy="true"
                allowCookies="false">
                <readerQuotas maxDepth="2147483647" maxStringContentLength="2147483647"
                    maxArrayLength="2147483647" maxBytesPerRead="2147483647" maxNameTableCharCount="2147483647" />
                <reliableSession ordered="true" inactivityTimeout="00:10:00"
                    enabled="true" />
                <security mode="Message">
                    <transport clientCredentialType="Windows" proxyCredentialType="None"
                        realm="" />
                    <message clientCredentialType="Windows" negotiateServiceCredential="true"
                        algorithmSuite="Default" establishSecurityContext="true" />
                </security>
            </binding>
        </wsHttpBinding>

下面提到的是我的客户端代码

class Program
{
static void Main(string[] args)
{
    Program prog = new Program();
    Thread JavaClientCallThread = new Thread(new ThreadStart(prog.ClientCallThreadRun));
    Thread JavaStatusCallThread = new Thread(new ThreadStart(prog.StatusCallThreadRun));
    JavaClientCallThread.Start();
    JavaStatusCallThread.Start();
}

public void ClientCallThreadRun()
{
    XmlDocument doc = new XmlDocument();
    doc.Load(@"D:\t72CalculateReasonableWithdrawal_Input.xml");
    bool error = false;
    Service1Client Client = new Service1Client();
    string temp = Client.Processing(doc.OuterXml, ref error);
}

public void StatusCallThreadRun()
{
    int i = 0;
    Service1Client Client = new Service1Client();
    string temp;
    while (i < 10)
    {
        temp = Client.Status();
        Thread.Sleep(1500);
        Console.WriteLine("TotalTestScenarios;CurrentTestCase = {0}", temp);
        i++;
    }
}
}

任何人都可以帮忙。

4

2 回答 2

1

首先因为需要同时访问服务,当服务在处理第一个客户端调用(Processing)时,需要将服务并发模式改为Multiple。

此外,您还想维护每个客户端的处理状态,因此您需要将实例上下文模式设置为 PerSession。

[ServiceBehavior(ConcurrencyMode = ConcurrencyMode.Multiple, InstanceContextMode= InstanceContextMode.PerSession)]

笔记

  • 默认 InstanceContextMode 是 PerSession
  • 默认 ConcurrencyMode 为 Single

您可以执行以下操作以确保您的配置与 PerSession InstanceContextMode 兼容,使用此方法,WCF 将在必要时抛出运行时异常

[ServiceContract(SessionMode=SessionMode.Required)]

注意使用 InstanceContextMode.PerSession 您将获得每个您创建的代理的不同实例

因此,每个客户端只需要一个“Service1Client”实例,您将调用其 Process 方法并从中检索状态。

此外,对于虚拟繁重处理,您可以仅在“处理”方法(服务端)中使用 Thread.Sleep(millisecond) 进行测试建议。

对于客户端应用程序,如果要调用“Processing”方法,然后使用 Status 方法检索状态,则需要异步调用 Process 方法。

1.在solution-explorer中右击服务引用,选择“配置服务引用”,勾选“生成异步操作”,点击确定。

2.像这样更改您的客户端代码

static void Main(string[] args)
{
    StartProcessing();
    StatusReport();

    Console.ReadLine();
}

static ServiceClient Client = new ServiceClient();
private static bool Completed = false;

public static void StartProcessing()
{
    XmlDocument doc = new XmlDocument();
    doc.Load(@"D:\t72CalculateReasonableWithdrawal_Input.xml");
    bool error = false;

    Client.ProcessingCompleted += Client_ProcessingCompleted;
    Client.ProcessingAsync(doc.OuterXml);

    Console.WriteLine("Processing...");
}

static void Client_ProcessingCompleted(object sender, ProcessingCompletedEventArgs e)
{
    // processing is completed, retreive the return value of Processing operation
    Completed = true;
    Console.WriteLine(e.Result);
}

public static void StatusReport()
{
    int i = 0;
    string temp;
    while (!Completed)
    {
        temp = Client.Status();
        Console.WriteLine("TotalTestScenarios;CurrentTestCase = {0}", temp);
        Thread.Sleep(500);
        i++;
    }
}
于 2012-06-29T20:25:48.003 回答
0

PerSession不会使您的静态变量不在对象实例之间共享。PerSession上下文模式所做的唯一一件事就是控制对象的生命周期。

PerSession会话结束之前,WCF 不会破坏服务对象。会话可以由客户端或超时(默认为 10 分钟)显式关闭。来自具有相同会话 ID 的客户端的每个下一次调用都将由 WCF 路由到现有对象。

变量不应该是静态的,以防止在不同的服务实例之间共享。只要您使用InstanceContextMode.PerSession和绑定维护会话,变量的状态将由 WCF 维护。

public int TotalTests = 0;
public int CurrentTest = 0;

我还将添加SessionMode.Required合同以确保正确配置服务。

 [ServiceContract(SessionMode = SessionMode.Required )]
于 2012-06-29T16:34:25.167 回答