4

我有以下类型,我在 WCF 中用作消息协定:

[MessageContract(IsWrapped = true, 
                 WrapperNamespace = "http://example.com/services", 
                 WrapperName = "EchoRequest")]
public class EchoRequest
{
    public EchoRequest() { }
    public EchoRequest(String value)
    {
        Value = value;
    }

    [MessageBodyMember(Name = "Value", 
                       Namespace = "http://example.com/services", 
                       Order = 0)]
    public String Value { get; set; }
}

当我使用svcutil.exe生成此类型的代理时,我得到了一个客户端,该客户端能够与托管它的服务进行通信,并且元素上的 XML 命名空间根据消息合同属性是正确的。

当我Message.CreateMessage(...)在它的实例上使用时,命名空间恢复为默认值(http://schemas.datacontract.org/2004/07/..)。当我使用 的实例时DataContractSerializer,会发生同样的事情。我尝试将命名空间传递给DataContractSerializer构造函数,并且只有包装器包含在命名空间中:

var requestMessage = new EchoRequest("hello, world!");
var serializer = new DataContractSerializer(typeof(EchoRequest), 
                                            "EchoRequest", 
                                            "http://example.com/services");
var stream = new MemoryStream();
serializer.WriteObject(stream, requestMessage);
var data = Encoding.UTF8.GetString(stream.ToArray());

此时,“数据”是:

<EchoRequest xmlns="http://example.com/services"
             xmlns:a="http://schemas.datacontract.org/2004/07/TestClient"
             xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <a:Value>hello, world!</a:Value>
</EchoRequest>

为什么DataContractSerializer出现忽略MessageContract属性?svcutil如何获得这项工作?

4

1 回答 1

5

这是因为消息合约不是数据合约,数据合约使用不同的属性来标记它们的类。尝试使用类型化消息转换器;

EchoRequest echoRequest = new EchoRequest{ value = "Hello" };

TypedMessageConverter echoMessageConverter = TypedMessageConverter.Create(
                 typeof(echoRequest),
                 "YourActionNameHere",
                 "http://example.com/services");
Message request = echoMessageConverter.ToMessage(
    echoRequest,MessageVersion.Soap11);

然后,您将准备好一条消息,并且可以在需要时将请求正文拉出。

于 2009-07-16T14:06:01.820 回答