我暂时还不清楚 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
这种方式的典型的应用是自定义文件上传, 你只需要更改少量的代码就可以实现支持断点续传功能的文件上传.