我想在NSInputStream中添加三个“部分”:NSString,另一个流的输出,然后是另一个NSString。这个想法如下:
第一个和最后一个NSStrings表示SOAP请求的开头和结尾,而流的输出是加载一个非常大的文件并将其编码为Base64字符串的结果。所以,最后我会让最终的NSInputStream像这样保存整个SOAP请求:
<肥皂开始> <
我希望整个请求在NSInputStream中保存的原因有两个:
我认为这是强制在HTTP 1.1块中发送最终请求的唯一方法(我需要它,否则,如果请求变得太大,服务器将不接受它)。所以,我知道这样做:
NSInputStream *dataStream = ....;
[request setHTTPBodyStream:dataStream];
确保请求将作为HTTP 1.1块发送,而不是作为一个巨大的原始SOAP请求发送。
所以,我想知道如何实现这一点 - 即,如何将事物“排入”NSInputStream?甚至可以做到吗?还有另一种方法吗?
仅供参考,在Java中可以按照以下方式完成
Vector<InputStream> streamVec = new Vector<InputStream>();
BufferedInputStream fStream = new BufferedInputStream(fileData.getInputStream());
Base64InputStream b64stream = new Base64InputStream(fStream, true);
String[] SOAPBody = GenerateSOAPBody(fileInfo).split("CUT_HERE");
streamVec.add(new ByteArrayInputStream(SOAPBody[0].getBytes()));
streamVec.add(b64stream);
streamVec.add(new ByteArrayInputStream(SOAPBody[1].getBytes()));
SequenceInputStream seqStream = new SequenceInputStream(streamVec.elements());
因为Java有这些对象可用,但Objective-c中的NSStreams看起来像非常低级别的对象,并且非常难以使用。
注意:我完全重写了原始问题,就像我2天前提出的那样,因为我认为新编辑更清楚地解释了问题所在。我希望这将有助于更容易理解并回答
这是我迄今为止所能实现的目标:我使用临时文件来首先编写&lt;而不是尝试入队。 soap开始&gt;,然后我设置一个输入流来从块中读取文件,将每个块编码为Base64字符串并将其写入相同的临时文件,最后,当我的流关闭时,我写了&lt;肥皂结束&gt;到临时文件。然后我设置另一个输入流,其中包含我传递给NSMutableURLRequest的文件内容:
NSMutableURLRequest* request = [NSMutableURLRequest requestWithURL:url];
...
NSInputStream *dataStream = [NSInputStream inputStreamWithFileAtPath:_tempFilePath];
[request setHTTPBodyStream:dataStream];
这可确保HTTP 1.1分块传输文件内容。连接完成后,删除临时文件。
这似乎工作得很好,但当然这是一个烦人的工作。我不想写一个临时文件,因为它可以由流处理(理想情况下。)如果有人仍然有更好的建议,请告诉我:)
好的,另一个更新是有序的。虽然我的文件写入似乎有效,但我现在遇到了一些意外问题,我的一些请求无法上传到服务器。具体来说,一切都按照计划进行,我正在将临时文件的内容读入流中,并将我的请求的HTTP主体设置为此流,并开始按我的意愿传输HTTP 1.1块 - 但对于某些一些数据包丢失的原因和最后的请求 - 这是我的猜测 - 变得格格不入,因此失败。我认为丢弃数据包的问题是随机的,因为我在更大的请求上观察它 - 也就是说,问题只是有更多机会出现 - 而我的较小请求通常会很好。这当然是与此问题中的原始问题不同的问题。如果有人知道可能导致这种情况的原因,我在这里询问了这个问题:Packets dropped during chunked HTTP 1.1 request sent by NSURLConnection
答案 0 :(得分:0)
您的解决方案是一个不错的选择,但您可以使用流来完成。它意味着继承NSInputStream
,这不是一件容易的事,因为你需要实现一堆方法。
基本上你的子类最初会返回头字节,然后它会将“内部”流中的字节返回到文件内容,然后当它用完时它会返回页脚字节。这意味着要记录页眉和页脚的大小以及到目前为止已经处理了多少,但这不是一个大问题。
有一个创建子类here的示例,它显示了您需要实现的棘手的隐藏方法,以使流子类正常工作而不会抛出异常。