在 WCF 中解决服务时出现问题
我正在使用 wcf 编写服务。 我创建了一个包含两个项目的解决方案:
- 库: 一个用于存储有关服务的文件的项目(包含该服务的接口和相应的实现)。该项目是一个图书馆项目。
- 托管应用程序 用于托管这些服务的项目(自托管)。因此,该项目是一个具有配置文件的可执行项目,我在其中放置了配置服务所需的信息。
我还编写了一个客户端来调用该服务。这将称为客户端应用程序。
我有一个服务。以下是接口和实现(库项目):
namespace EchoWcfLibrary {
/// <summary>
/// The interface specifies for those classes implementing it (services), the operation that the service will expose.
/// </summary>
[ServiceContract]
public interface IService1 {
// This does not use serialization (implicit serialization in considered: base types used).
[OperationContract]
string GetData(int value);
// This uses data contracts and serialization.
[OperationContract]
CompositeType GetDataUsingDataContract(CompositeType composite);
}
[DataContract]
public class CompositeType {
// Members not serialized
bool boolValue = true;
string stringValue = "Hello ";
// Serialized
[DataMember]
public bool BoolValue {
get { return boolValue; }
set { boolValue = value; }
}
// Serialized
[DataMember]
public string StringValue {
get { return stringValue; }
set { stringValue = value; }
}
}
}
以下是服务宿主应用程序(可执行项目)的启动:
namespace WcfServiceApplication {
public static class Program {
static void Main(string[] args) {
// Setting endpoints and setting the service to start properly.
// Base address specified: http://localhost:8081/Service1
Console.WriteLine("Beginning...");
using (ServiceHost host = new ServiceHost(typeof(Service1), new Uri("http://localhost:8081/Service1"))) {
Console.WriteLine("Opening host...");
host.Open();
Console.WriteLine("Waiting...");
System.Threading.Thread.Sleep(1000000);
Console.WriteLine("Closing...");
host.Close();
Console.WriteLine("Quitting...");
}
}
}
}
以下是可执行项目(宿主应用程序)中的 App.config
:
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<!-- When deploying the service library project, the content of the config file must be added to the host's
app.config file. System.Configuration does not support config files for libraries. -->
<system.serviceModel>
<services>
<service name="WcfServiceLibrary.Service1">
<host>
<baseAddresses>
<add baseAddress="http://localhost:8081/Service1" />
</baseAddresses>
</host>
<!-- Service Endpoints -->
<!-- Unless fully qualified, address is relative to base address supplied above -->
<endpoint address="/GoInto/Svc"
binding="basicHttpBinding"
contract="WcfServiceLibrary.IService1">
<!--
Upon deployment, the following identity element should be removed or replaced to reflect the
identity under which the deployed service runs. If removed, WCF will infer an appropriate identity
automatically.
-->
<identity>
<dns value="localhost"/>
</identity>
</endpoint>
<endpoint address="/GoInto/Sav"
binding="basicHttpBinding"
contract="WcfServiceLibrary.IService1">
<!--
Upon deployment, the following identity element should be removed or replaced to reflect the
identity under which the deployed service runs. If removed, WCF will infer an appropriate identity
automatically.
-->
<identity>
<dns value="localhost"/>
</identity>
</endpoint>
<!-- Metadata Endpoints -->
<!-- The Metadata Exchange endpoint is used by the service to describe itself to clients. -->
<!-- This endpoint does not use a secure binding and should be secured or removed before deployment -->
<endpoint address="GoInto/Mex" binding="mexHttpBinding" contract="IMetadataExchange"/>
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior>
<!-- To avoid disclosing metadata information,
set the value below to false and remove the metadata endpoint above before deployment -->
<serviceMetadata httpGetEnabled="True"/>
<!-- To receive exception details in faults for debugging purposes,
set the value below to true. Set to false before deployment
to avoid disclosing exception information -->
<serviceDebug includeExceptionDetailInFaults="False" />
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
</configuration>
以下是客户端可执行项目中的程序启动(已创建到库项目的链接),基本上这是客户端:
namespace WcfServiceClient {
class Program {
static void Main(string[] args) {
ServiceEndpoint httpEndpoint = new ServiceEndpoint(ContractDescription.GetContract(typeof(IService1)), new BasicHttpBinding(), new EndpointAddress("http://localhost:8081/Service1"));
ChannelFactory<IService1> channelFactory = new ChannelFactory<IService1>(httpEndpoint);
IService1 svc = channelFactory.CreateChannel();
Console.WriteLine(svc.GetData(121));
System.Threading.Thread.Sleep(10000);
}
}
}
嗯...我的问题如下:该应用程序可以工作!
为什么会出现问题???
问题是我在托管服务时在 App.config 文件中指定了三个端点:两个 basicHttp 和一个元数据端点。好吧,我想解决 http://localhost:8081/Service1/GoInto/Svc
。
好吧,不幸的是,在客户端中,我解决了这个端点:http://localhost:8081/Service1
,这只是基地址......为什么它有效???我想在客户端中指定此地址:
namespace WcfServiceClient {
class Program {
static void Main(string[] args) {
ServiceEndpoint httpEndpoint = new ServiceEndpoint(ContractDescription.GetContract(typeof(IService1)), new BasicHttpBinding(), new EndpointAddress("http://localhost:8081/Service1/GoInto/Svc"));
ChannelFactory<IService1> channelFactory = new ChannelFactory<IService1>(httpEndpoint);
IService1 svc = channelFactory.CreateChannel();
Console.WriteLine(svc.GetData(121));
System.Threading.Thread.Sleep(10000);
}
}
}
但是如果我这样做,则会引发不匹配错误:
带有“To”的消息 'http://localhost:8081/Service1/GoInto/Svc' 无法在接收方进行处理, 由于地址过滤器不匹配 端点调度程序。检查一下 发送者和接收者的 端点地址一致。
为什么不起作用?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
基地址必须在一处指定,可以在 ServiceHost 构造函数上,也可以在元素中。如果这两个地方都有,WCF 将抛出异常,表明您有同一方案 (HTTP) 的两个基地址。
可能发生的情况是,托管项目的 app.config 上的服务名称不匹配,因此未选择配置(并且您得到的是默认端点,其地址与基地址)。尝试在托管代码上添加 foreach 循环,您应该会看到您的服务正在侦听的端点的地址。
The base address must be specified in one place, either on the ServiceHost constructor, or in the element. If you have in both places, WCF will throw an exception saying that you have two base addresses for the same scheme (HTTP).
What is likely happening is that you have a mismatch on the service name on app.config of the hosting project, so that configuration is not being picked up (and what you get is a default endpoint, whose address is at the same one as the base address). Try adding the foreach loop on your hosting code, and you should see the addresses of the endpoints your service is listening at.