我正在尝试构建一个WCF服务,允许我的WPF桌面客户端将文件上传到服务器。
我修改了代码项目中的代码示例(WCF Streaming: Upload/Download Files Over HTTP),我也查看了几个SO帖子,但似乎无法实现这一点。
当我执行代码时,在服务器尝试读取已通过接口传递的流时,它会因空引用异常而失败。
此时,我很遗憾,不知道如何解决这个问题。任何建议都表示赞赏。
代码示例如下:
CustomerDocumentModel是我通过WCF接口传输的数据元素,用于读取客户端文件:
[DataContract]
[KnownType(typeof(System.IO.FileStream))]
public class CustomerDocumentModel : IDisposable
{
public CustomerDocumentModel()
{
}
public CustomerDocumentModel(string documentName, string path)
{
DocumentName = documentName;
Path = path;
}
[DataMember]
public string DocumentName;
[DataMember]
public string Path;
[DataMember]
public System.IO.Stream FileByteStream;
public void Dispose()
{
if (FileByteStream != null)
{
FileByteStream.Close();
FileByteStream = null;
}
}
}
IBillingService是我的WCF服务的接口定义:
[ServiceContract]
public interface IBillingService
{
// other methods redacted...
[OperationContract]
void UploadCustomerDocument(CustomerDocumentModel model);
}
BillingService类实现WCF服务:
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
public class BillingService : IBillingService
{
// Other methods redacted ...
public void UploadCustomerDocument(CustomerDocumentModel model)
{
string path = HttpContext.Current.Server.MapPath(
String.Format("/Documents/{1}",
model.DocumentName));
using (FileStream stream = new FileStream(path, FileMode.Create, FileAccess.Write, FileShare.None))
{
const int bufferSize = 4096;
byte[] buffer = new byte[bufferSize];
int size = 0;
try
{
// The following Read() fails with a NullReferenceException
while ((size = model.FileByteStream.Read(buffer, 0, bufferSize)) > 0)
{
stream.Write(buffer, 0, size);
}
}
catch
{
throw;
}
finally
{
stream.Close();
model.FileByteStream.Close();
}
}
}
}
我的WCF Web服务器上的web.config中的一些相关位:
<system.web>
<compilation debug="true" targetFramework="4.0" />
<httpRuntime maxRequestLength="2097151" useFullyQualifiedRedirectUrl="true" executionTimeout="360"/>
</system.web>
<system.serviceModel>
<serviceHostingEnvironment
aspNetCompatibilityEnabled="true"
multipleSiteBindingsEnabled="true" />
<bindings>
<basicHttpBinding>
<binding name="userHttps" transferMode="Streamed" maxReceivedMessageSize="2147483647" maxBufferSize="2147483647">
<readerQuotas maxDepth="2147483647" maxStringContentLength="2147483647" maxArrayLength="2147483647" maxBytesPerRead="2147483647" maxNameTableCharCount="2147483647" />
<security mode="None" />
</binding>
</basicHttpBinding>
</bindings>
<behaviors>
<serviceBehaviors>
<behavior name="">
<dataContractSerializer maxItemsInObjectGraph="2147483646"/>
<serviceDebug includeExceptionDetailInFaults="true" />
<serviceMetadata httpGetEnabled="true" />
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
客户端是WPF / MVVM应用程序,它创建CustomerDocumentModel模型,使用OpenFileDialog打开()文件流,然后将模型传递给WCF服务上的UploadCustomerDocument方法。
如果我遗漏任何相关细节,请询问。
答案 0 :(得分:4)
我知道这个问题的回复非常迟,我相信你一定也解决了你的问题。这可能对其他人有帮助: - )
对Datacontract使用Messagecontract,只有一个MessageBodyMember,数据类型为Stream,其余所有参数都是MessageHeader。 这是一个例子:
[MessageContract]
public class CustomerDocumentModel : IDisposable
{
public CustomerDocumentModel(string documentName, string path)
{
DocumentName = documentName;
Path = path;
}
[MessageHeader]
public string DocumentName{get;set;}
[MessageHeader]
public string Path{get;set;}
[MessageBodyMember]
public System.IO.Stream FileByteStream{get;set;}
public void Dispose()
{
if (FileByteStream != null)
{
FileByteStream.Close();
FileByteStream = null;
}
}
}
注意:确保您的配置传输模式是StreamedResponse,您也可以将MessageEncoding更改为MTOM以获得更好的性能。
public void UploadCustomerDocument(CustomerDocumentModel model)
{
var filename = //your file name and path;
using (var fs = new FileStream(filename, FileMode.Create))
{
model.FileByteStream.CopyTo(fs);
}
}
答案 1 :(得分:2)
您的数据类型正在使流式传输失败。这在MSDN上有记录:http://msdn.microsoft.com/en-us/library/ms731913.aspx 相关段落是:
流式传输的限制
使用流式传输模式会导致运行时间强制执行 其他限制。
流式传输中发生的操作可以签订合同 最多有一个输入或输出参数。该参数对应 消息的整个主体,必须是一个消息,一个派生 Stream的类型,或IXmlSerializable实现。有回报 操作的值等同于具有输出参数。
一些WCF功能,例如可靠的消息传递,事务和SOAP 消息级安全性,依赖缓冲消息进行传输。 使用这些功能可能会降低或消除性能优势 通过使用流媒体获得。要确保流式传输,请使用 仅限传输级安全性或使用传输级安全性 仅验证消息安全性。
即使设置了传输模式,SOAP标头也始终被缓冲 流式传输邮件的标头不得超过邮件的大小 MaxBufferSize传输配额。有关此内容的更多信息 设置,请参阅运输配额。