今天,做一个C# 进行HTTP POST 发送文件的功能,将一个XML文件中的内容发送到指定服务器,以下是主要代码:
HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);
request.Method = "POST";
request.ContentType = "multipart/form-data;boundary=" + boundary;
request.ContentLength = Encoding.UTF8.GetByteCount(postDataStr);
Stream myRequestStream = request.GetRequestStream();
StreamWriter myStreamWriter = new StreamWriter(myRequestStream, Encoding.GetEncoding("UTF-8"));
myStreamWriter.Write(postDataStr);
myStreamWriter.Close();
报错的是:“myStreamWriter.Close()” 这一行,意思是说在调用close()方法的时候还没有写完全部字节,怎么会这样呢?
按理说,C#提供Write()、Close()这类方法,应该是非常成熟可靠的,怎么写字节,写多少字节,这两个方法应该都是照章办事,如果出错,那一定是其他地方的问题。
仔细想想,很可能是Wirte()方法在执行到某个点的时候,虽然还没写完,但Close()方法已经按照它收到的命令进行关闭操作了,那么,Close()方法接到的指令可能是什么呢?
仔细观察上下文代码,这个关闭的依据很可能就是“request.ContentLength” 这个点。
于是,先把“request.ContentLength = Encoding.UTF8.GetByteCount(postDataStr);” 给注释掉,进行测试:
//request.ContentLength = Encoding.UTF8.GetByteCount(postDataStr);
结果这次没报错,发送成功了!
看来问题果然是这个HTTP请求的 ContentLength 的问题。接着做一个测试:
1. 记下postDataStr内容的长度,结果是 31317,
label1.Text = Encoding.UTF8.GetByteCount(postDataStr) + "";
2. 在服务器端观察实际收到的内容长度,结果是 31320,,
很显然,是实际发出的长度更大,所以如果把request.contentLength 设置成一个更小的数字,那么也许Close()方法就会提前调用,导致报错。
因此解决办法就是不要手动指定request的contentLength,让程序按默认的规则办事。