简介
上传文件到服务器是一个比较常用的操作,最基本的方式是通过POST上传,文件以二进制形式,作为一个参数传递,但是这个POST的结构相当复杂,且必须完全符合HTTP标准。
文件上传的POST格式
该POST主要由下面几个部分构成。
- 请求头
1.Content-Length(请求体的二进制大小)
- 注意这里的二进制大小应该根据请求体计算
2.Content-Type multipart/form-data; boundary=(分隔符)
- 注意这里的分隔符与请求体的分隔符有关,但不完全一致。
- 请求体
基本格式示例
- 这里的分隔符就是上面的分隔符,但是前面要多加两个
'-'
。
--分隔符
Content-Disposition: form-data; name="uploadFile"; filename="button.png"
Content-Type: image/png
(此处空两行)
<二进制内容>
--分隔符
Content-Disposition: form-data; name="submit"
(此处空两行)
Submit
--分隔符--
(此处空一行)
- 注意到最后的分隔符后面跟了
--
,这个代表结束符,并且后面要跟一个空行。
格式说明
- 普通参数
普通参数的构成如下:
Content-Disposition: form-data; name="参数名"
(此处空两行)
参数值
--分隔符
多个参数可以连续拼接。
- 文件参数
文件参数与普通参数类似,只是多了一行MineType的说明,该说明告诉服务器文件的类型。
Content-Disposition: form-data; name="uploadFile"; filename="button.png"
Content-Type: image/png
(此处空两行)
<二进制内容>
--分隔符
将二者连起来就构成了完整的文件上传POST信息,到这里我们可以理解,该POST不仅发送了文件数据,还发送了一个参数。
文件上传的服务器php脚本
- 要让post请求发挥作用,必须借助php脚本实现对post的处理,换句话说,我们的post请求应该发送给该php脚本,脚本的代码如下:(注意修改uploadPath为自己服务器想要接收文件的路径)
<?php
header("Content-type: text/html; charset=utf-8");
// 配置文件需要上传到服务器的路径,需要允许所有用户有可写权限,否则无法上传!
$uploadPath = '../uploads/';
$IOS_forKey=$_FILES["uploadFile"];
if ($IOS_forKey["error"] > 0) {
echo "传入参数错误:" . $IOS_forKey["error"] . "<br />";
} else {
echo "文件: " . $IOS_forKey["name"] . "<br />";
echo "类型: " . $IOS_forKey["type"] . "<br />";
echo "大小: " . ($IOS_forKey["size"] / 1024) . " Kb<br />";
echo "临时文件: " . $IOS_forKey["tmp_name"] . "<br />";
chmod($uploadPath . $IOS_forKey["name"], 0666);
if (file_exists($uploadPath . $IOS_forKey["name"])) {
echo $IOS_forKey["name"] . "文件已经存在!";
} else {
move_uploaded_file($IOS_forKey["tmp_name"], $uploadPath . $IOS_forKey["name"]);
echo "上传文件保存在: " . $uploadPath . $IOS_forKey["name"];
}
}
?>
通过iOS设备上传文件
想要通过iOS设备上传文件,一般的做法是根据上面的结构创建URLRequset,然后发送该request到服务器请求上面的php脚本,实现文件的上传,具体的代码如下。
- 为了方便插入二进制文件数据,我们直接使用data拼接,因此对于每一段字符串都需要转为data,这就是DataWithStr宏的作用。
- 为了适配各个系统的换行符,使用\r\n。
- FileBoundary就是上文提到的分隔符。
- HTTP请求头的contentLength需要待请求体拼接完毕后才能得到,因此最后才赋值。
#import "ViewController.h"
#import "UploadFile.h"
#define FileBoundary @"-----------------------------test"
#define EndLine @"-----------------------------test--\r\n"
#define NewLine @"\r\n"
#define DataWithStr(str) [str dataUsingEncoding:NSUTF8StringEncoding]
@interface ViewController ()
@end
@implementation ViewController
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
[self upload];
}
- (void)upload{
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:@"http://127.0.0.1/lesson2/upload.php"]];
request.HTTPMethod = @"POST";
// 设置请求头
NSString *contentType = [NSString stringWithFormat:@"multipart/form-data; boundary=%@",FileBoundary];
[request setValue:contentType forHTTPHeaderField:@"Content-Type"];
// 设置请求体
NSMutableData *body = [NSMutableData data];
[body appendData:DataWithStr(@"--")];
[body appendData:DataWithStr(FileBoundary)];
[body appendData:DataWithStr(NewLine)];
[body appendData:DataWithStr(@"Content-Disposition: form-data; name=\"uploadFile\"; filename=\"test.png\"")];
[body appendData:DataWithStr(NewLine)];
[body appendData:DataWithStr(@"Content-Type: image/png")];
[body appendData:DataWithStr(NewLine)];
[body appendData:DataWithStr(NewLine)];
UIImage *img = [UIImage imageNamed:@"test.png"];
NSData *imgData = UIImagePNGRepresentation(img);
[body appendData:imgData];
[body appendData:DataWithStr(NewLine)];
// 其他参数
[body appendData:DataWithStr(@"--")];
[body appendData:DataWithStr(FileBoundary)];
[body appendData:DataWithStr(@"Content-Disposition: form-data; name=\"param1\"")];
[body appendData:DataWithStr(NewLine)];
[body appendData:DataWithStr(NewLine)];
[body appendData:DataWithStr(@"value1")];
[body appendData:DataWithStr(NewLine)];
[body appendData:DataWithStr(@"--")];
[body appendData:DataWithStr(EndLine)];
[request setValue:[NSString stringWithFormat:@"%ld",body.length] forHTTPHeaderField:@"Content-Length"];
request.HTTPBody = body;
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
NSString *result = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
NSLog(@"%@",result);
}];
}
@end