我暂时还不清楚 HTTP 文件上传的具体细节, 但是我知道通过浏览器上传文件, 然后用 PHP 接收, 需要使用 PHP 的预定义变量 $_FILES. 最近我有一个应用需要使用 PHP 的预定义变量 $_POST 获取上传的文件(或者任何数据), 也就是通过 HTTP POST 参数传递二进制数据.
工作在 ContentType = "application/x-www-form-urlencoded" 的模式下时, HTTP 协议使用 ASCII 字符集的一个子集来编码要传输的字节流. 如字符串 "a@" 如果以 ASCII 编码存在于内存中(以字节数组的形式), 也就是内存按字节读取为 HEX 是: 0x61 0x40, 那么该字节数组被传输时会被编码(URL 编码)成为 0x61 0x25 0x34 0x30, 被编码后字节数组如果按照 ASCII 编码显示, 就是 "a%40". 也就是说, 当你想传输 0x61 0x40 时, HTTP 传输的是 0x61 0x25 0x34 0x30.
这种编码过程几乎所有的编程语言都提供支持, 在 C# 中你可以使用 HttpUtility.UrlEncode(), 输入一个字节数组, 返回一个 URL 编码的 字节数组.
在 PHP 脚本中, 当你使用 $_POST['data'] 获取数组时, 该数组已经被 PHP 自动解码了. 按照上面的例子, $_POST['data'] 所指向的内存内容为 0x61 0x40 -- 但是, 默认情况下并不总是, 如果该数据包含单引号等少数几个字符, 那么它们的前面会被 PHP 加上 \.
下面先给出 PHP 脚本, 保存一个通过 HTTP 参数传递的文件:
<?php $filename = $_POST['filename']; $data = $_POST['data']; if(get_magic_quotes_gpc()){ // 去掉斜杠 $data = stripslashes($data); } file_put_contents($filename, $data); ?>
C# 客户端为:
using System; using System.Text; using System.Net; using System.Web; using System.IO; class Test { public static void Main(){ try{ WebResponse response; HttpWebRequest request; request = (HttpWebRequest)HttpWebRequest.Create("http://localhost/test.php"); request.Method = "POST"; request.ContentType = "application/x-www-form-urlencoded"; String filename = "test.jpg"; // HTTP 参数名 byte[] keyBytes = Encoding.UTF8.GetBytes(String.Format("filename={0}&data=", filename)); FileStream fs = new FileStream(filename, FileMode.Open); byte[] buffer = new byte[8192]; int n = fs.Read(buffer, 0, buffer.Length); byte[] dataArray = new byte[n]; Array.Copy(buffer, dataArray, n); dataArray = HttpUtility.UrlEncodeToBytes(dataArray); request.ContentLength = keyBytes.Length + dataArray.Length; Stream dataStream = request.GetRequestStream(); // 发送参数名 dataStream.Write(keyBytes, 0, keyBytes.Length); // 发送名为 data 对应的值. dataStream.Write(dataArray, 0, dataArray.Length); dataStream.Close(); response = (HttpWebResponse)request.GetResponse(); // 读取服务器的返回, 在本应用中, 如果正常, 返回空字符串. StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.GetEncoding("UTF-8")); Console.Write(sr.ReadToEnd()); response.Close(); } catch (Exception e) { Console.Write("Error: " + e.ToString() + "\r\n"); } } }
如果你要测试的文件小于 8K, 可以直接使用上面的程序测试. 如果大于, 你可以加大 buffer 的容量.
上传成功后, 使用文件比较工具对比两个文件是否一致:
diff test.jpg up_test.jpg
这种方式的典型的应用是自定义文件上传, 你只需要更改少量的代码就可以实现支持断点续传功能的文件上传.