1

我正在编写一个控制台应用程序,它必须通过 Web 服务发送数据来将大型 SQL 表从一个数据库复制到另一个数据库。因为表可能非常大,并且它们不能(也不应该)完全存储在内存中,所以我使用 DataReader 顺序读取数据并希望将其流式传输到 Web 服务。

为了做到这一点,我将我的 Web 服务声明如下:

<OperationContract()>
<WebInvoke(UriTemplate:="{Id}/SendData", method:="PUT", BodyStyle:=WebMessageBodyStyle.WrappedRequest)>
Sub UploadData(ByVal Id As String, ByVal DataStream As Stream)

在我的控制台应用程序中,我正在创建这样的请求:

    Dim tRequest As HttpWebRequest = HttpWebRequest.Create(URL_WEBSERVICES & "DataLoad/" & _Id & "/SendData")
    tRequest.SendChunked = True
    tRequest.Method = "PUT"
    tRequest.ContentType = "application/octet-stream"

然后我打开请求流并像这样写入数据:

While tSqlReader.Read
    ' Check if a new chunk of data must be created
    If tRows Is Nothing Then
        tRows = New List(Of Object)
    End If
    ' Fill chunk with data
    tRowValues = New Object(tSqlReader.FieldCount - 1) {}
    tSqlReader.GetValues(tRowValues)
    tRows.Add(tRowValues)
    ' Check if chunk is full and must be sent
    If tRows.Count = CHUNK_ROWS_COUNT Then
        ' Write data to the request stream
        tDataChunk = SerializeToByteArray(tRows)
        tStream.Write(tDataChunk, 0, tDataChunk.Length)
        tRows = Nothing
        tDataChunk = Nothing
    End If
End While
' Write final data to the request stream
If tRows IsNot Nothing Then
    tDataChunk = SerializeToByteArray(tRows)
    tStream.Write(tDataChunk, 0, tDataChunk.Length)
End If

然后我关闭请求流并调用 tRequest.GetResponse()。

SerializeToByArray 是以下函数:

Private Function SerializeToByteArray(ByVal pObject As Object) As Byte()
    If pObject Is Nothing Then Return Nothing
    Dim tBinaryFormatter As New Runtime.Serialization.Formatters.Binary.BinaryFormatter
    Dim tMemoryStream As New MemoryStream()
    tBinaryFormatter.Serialize(tMemoryStream, pObject)
    Return tMemoryStream.ToArray
End Function

我的问题是网络服务只接收第一块数据,例如如果 CHUNK_ROWS_COUNT = 5 那么它只接收 5 行数据。(我知道这不是 HttpWebRequest 将产生的块的大小)。

我唯一的猜测是我正在通过流发送许多序列化数据包,每个数据包都由一些序列化标头封装,因此 Web 服务中的反序列化过程只会找到并解封装第一个,但这只是一个想法,我我不知道如何以不同的方式做到这一点。我无法序列化整个内容并分块发送写入,因为整个内容不适合内存。

有什么建议吗?

非常感谢提前

4

1 回答 1

0

我不确定您是否可以在 WCF 中使用手动 HttpWebRequest 执行此操作。WCF 将自己处理块,不要重新发明轮子。

一些伟大的文章:

不要忘记在 wcf 配置中更改阅读器配额

于 2012-09-17T11:11:30.743 回答