7

昨晚我遇到了两个问题,现在我已经解决了,但我不能 100% 确定为什么我所做的事情已经解决了这些问题,并希望有人能提供一些见解,因为我已经翻了很多岩石,没有运气!

第一个问题

第一个问题是我有两个唯一命名的管道,它们位于两个单独的程序中:

  • net.pipe://localhost/superuniquepipe1
  • net.pipe://localhost/superuniquepipe2

但是,由于地址已在使用中,在打开 ServiceHost 时,要启动的第二个程序会引发异常(我相信是 AddressAlreadyInUseException)。

我实例化这些 ServiceHosts 的方式如下

Uri[] baseAddresses = new Uri[] { new Uri("net.pipe://localhost") };
this.host = new ServiceHost(this, baseAddresses);
this.host.AddServiceEndpoint(typeof(IHostType), new NetNamedPipeBinding(), "superuniquepipe1");
this.host.Open();

所以我会先指定localhost的基地址,然后在添加端点时指定其余部分,我解决这个问题的方法是更改​​代码如下:

this.host = new ServiceHost(this);
this.host.AddServiceEndpoint(typeof(IHostType), new NetNamedPipeBinding(), "net.pipe://localhost/superuniquepipe2");
this.host.Open();

我是否正确地说这有效的原因是因为它只检查基地址而不是我试图添加的端点?并且使用第二个代码示例是让多个程序在“localhost”上侦听的有效/安全方式吗?

第二个问题

为了解决上述问题,我将基地址从 localhost 更改为多个不同的唯一字符串,例如“net.pipe://rawrwhyisntthisworkingsadface”,但是在这样做时,我会收到来自客户端尝试的 InvalidCredentialException建立连接(见下面的代码)

我的印象是命名管道可以从字面上命名任何东西,任何人都可以对此有所了解吗?

ChannelFactory<IHostType> factory = new ChannelFactory<IHostType>(new NetNamedPipeBinding(), new EndpointAddress("net.pipe://rawrwhyisntthisworkingsadface/superuniquepipe2"));
IHostType proxy = factory.CreateChannel();
proxy.CallSomeMethodAndGetAnException();

任何输入将不胜感激,正如我所说,我已经解决了这个问题,只是想知道为什么我的解决方案有效,但是如果您发现我解决问题的方式存在缺陷并且可以提出更好的解决方法,请这样做:)

4

1 回答 1

13

重新问题1:

WCFNetNamedPipeBinding使用命名的共享内存部分向其客户端发布可以调用服务的管道的实际名称。管道名称本身是一个 GUID,每次打开服务主机时都会重新生成。它是用于发布依赖于服务 URL 的服务的共享内存部分的名称。如果定义了基地址,则基地址用于派生该名称。

这意味着您一次只能运行一个 WCF 服务应用程序,该应用程序使用特定的基地址作为其 NetNamedPipe 端点。如果您尝试启动第二个,它会失败并显示 AddressAlreadyInUseException,因为它发现 WCF 想要用于发布位置的名称(从基地址派生)已被另一个应用程序占用。

如果您不指定基地址,并为每个服务提供一个绝对的、唯一的服务 URL,那么发布位置的名称现在是从完整的绝对 URL 派生的,并且应用程序之间没有名称冲突。这是让多个 WCF 命名管道服务侦听的完全有效且安全的方法。

重新问题2:

在服务端,您可以对服务 URL 的主机名部分使用任何内容。这是由于HostNameComparisonModeNetNamePipeBinding 中默认应用的设置,因为 WCF 中派生共享内存发布位置名称的算法用通配符替换主机名(参见此处)以启用配置的主机名比较模式。

然而,在客户端,服务 URL 受到限制:主机部分必须真正解析为 localhost(即它是localhost,正确的 IP 地址或正确的机器名称)。

于 2012-04-29T14:23:59.260 回答