iOS通过http post上传图片

由于iOS无法通过html表单来上传图片,因此想要上传图片,必须实现http请求,而不能像其他语言那样通过html表单的post就能上传。

上传图片的http post请求的格式是这样的:

?
1
2
3
4
5
6
7
8
9
10
11
12
Content-type: multipart/form-data, boundary=AaB03x
 
--AaB03x
content-disposition: form-data; name= "field1"
 
Hello Boris!
--AaB03x
content-disposition: form-data; name= "pic" ; filename= "boris.png"
Content-Type: image/png
 
... contents of boris.png ...
--AaB03x--




第一行是指定了http post请求的编码方式为multipart/form-data(上传文件必须用这个)。
boundary=AaB03x说明了AaB03x为分界线。比如 --AaB03x 就是一个分界线的意思

content-disposition: form-data; name="field1"

Hello Boris!

  这句话声明了请求中的一个字段的名称,如field1  以及字段的值,如Hello Boris!
这里类似form表单中的<input name="field1" type="text" value="Hello Boris!"/>
中间的空行是必须的。

不同的字段之间用分界线分开,分界线需要单独一行,如 --AaB03x--

分界线的下一行,是下一个字段

content-disposition: form-data; name="pic"; filename="boris.png"
Content-Type: image/png

 ... contents of boris.png ...
 --AaB03x--

这里声明了变量pic,也就是我们要传的文件,上传文件的时候需要在后边指定file name:filename="boris.png"
并且需要在下一行指定文件的格式:Content-Type: image/png


 ... contents of boris.png ...  这里是boris.png的二进制内容,如 <89504e47 0d0a1a0a 0000000d 49484452 000000b4 000000b4 08020000 00b2af91 65000020 00494441 5478012c dd79b724 6b7616f6 8c888c88 8c9c8733 55ddb1d5 6a0db486 06218401 ......

在http post请求的结尾,需要有一个分界线,但是是前后都有--的:--AaB03x--

以上的这些格式,是http的规范,每个空行,空格都是必须的。

下边是iOS的实现代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
//分界线的标识符
         NSString *TWITTERFON_FORM_BOUNDARY = @ "AaB03x" ;
         //根据url初始化request
         NSMutableURLRequest * request = [ NSMutableURLRequest requestWithURL:[ NSURL URLWithString:url]
                                                                cachePolicy: NSURLRequestReloadIgnoringLocalCacheData
                                                            timeoutInterval:10];
         //分界线 --AaB03x
         NSString *MPboundary=[[ NSString alloc]initWithFormat:@ "--%@" ,TWITTERFON_FORM_BOUNDARY];
         //结束符 AaB03x--
         NSString *endMPboundary=[[ NSString alloc]initWithFormat:@ "%@--" ,MPboundary];
         //要上传的图片
         UIImage *image=[params objectForKey:@ "pic" ];
         //得到图片的data
         NSData * data = UIImagePNGRepresentation(image);
         //http body的字符串
         NSMutableString *body=[[ NSMutableString alloc]init];
         //参数的集合的所有key的集合
         NSArray *keys= [params allKeys];
         
         //遍历keys
         for ( int i=0;i<[keys count];i++)
         {
             //得到当前key
             NSString *key=[keys objectAtIndex:i];
             //如果key不是pic,说明value是字符类型,比如name:Boris
             if (![key isEqualToString:@ "pic" ])
             {
                 //添加分界线,换行
                 [body appendFormat:@ "%@\r\n" ,MPboundary];
                 //添加字段名称,换2行
                 [body appendFormat:@ "Content-Disposition: form-data; name=\"%@\"\r\n\r\n" ,key];
                 //添加字段的值
                 [body appendFormat:@ "%@\r\n" ,[params objectForKey:key]];           
             }
         }
         
         ////添加分界线,换行
         [body appendFormat:@ "%@\r\n" ,MPboundary];
         //声明pic字段,文件名为boris.png
         [body appendFormat:@ "Content-Disposition: form-data; name=\"pic\"; filename=\"boris.png\"\r\n" ];
         //声明上传文件的格式
         [body appendFormat:@ "Content-Type: image/png\r\n\r\n" ];
         
         //声明结束符:--AaB03x--
         NSString * end =[[ NSString alloc]initWithFormat:@ "\r\n%@" ,endMPboundary];
         //声明myRequestData,用来放入http body
         NSMutableData *myRequestData=[ NSMutableData data];
         //将body字符串转化为UTF8格式的二进制
         [myRequestData appendData:[body dataUsingEncoding: NSUTF8StringEncoding ]];
         //将image的data加入
         [myRequestData appendData:data];
         //加入结束符--AaB03x--
         [myRequestData appendData:[ end dataUsingEncoding: NSUTF8StringEncoding ]];
         
         //设置HTTPHeader中Content-Type的值
         NSString *content=[[ NSString alloc]initWithFormat:@ "multipart/form-data; boundary=%@" ,TWITTERFON_FORM_BOUNDARY];
         //设置HTTPHeader
         [request setValue:content forHTTPHeaderField:@ "Content-Type" ];
         //设置Content-Length
         [request setValue:[ NSString stringWithFormat:@ "%d" , [myRequestData length]] forHTTPHeaderField:@ "Content-Length" ];
         //设置http body
         [request setHTTPBody:myRequestData];
         //http method
         [request setHTTPMethod:@ "POST" ]; 
         
         //建立连接,设置代理
         NSURLConnection *conn = [[ NSURLConnection alloc] initWithRequest:request delegate: self ];
         
         //设置接受response的data
         if (conn) {
             mResponseData = [[ NSMutableData data] retain];
         }













在开发网络相关的应用,比如微博应用时,就必然需要使用到HTTP请求来发送或者接收数据。最主要的就是使用GET方法或者POST方法。本文将详细介绍HTTP请求在iOS开发中的编程实现。
1、对于HTTP请求的基本介绍
这里不对原理做过多的介绍,大家可以Google之。对于完全不了解的童鞋,这里作个基本的说明。举新浪开放平台为例,我们要从手机发布一条新的状态,需要通过网络向新浪微博的服务器发送请求,这个请求中包含了我们要发送的状态信息。请求的URL必须是新浪开放平台指定的才行。比如这个发送状态,URL是https://api.weibo.com/2/statuses/update.json。大家可以自己到开放平台查看,然后使用的HTTP请求方式是POST。然后要附带一些请求参数。HTTP请求有GET,POST,DELETE等等方式,不同的方式它的URL request格式不同,对于大多数使用情况,使用GET和POST就行。然后如果发送请求成功的话,就可以从服务器接受到相关的信息。
GET是比较简单的请求方式,在URL中直接就包含了要发送的所有参数。还是以新浪开放平台的请求为例。我们要获取获取某个用户最新发表的微博列表。那么这个基本的URL是https://api.weibo.com/2/statuses/user_timeline.json ,然后由于这个请求主要是接收数据,要附带的请求参数最基本的就两个,一个是access_token,一个uid就是用户的编号。
那么一个基本的GET请求的URL就是
https://api.weibo.com/2/statuses/user_timeline.json?access_token=xxxxxxxxxxxxxxxxxxxxxxxxxxx&uid=xxxxxxxxxxxxxxx
那么在iOS开发中,确定了URL,然后就是建立URLRequest,NSURLConnection去实现请求。当然,在实际中,GET方法也有它相关的报文格式,如下:
GET /api.weibo.com/2/statuses/user_timeline.json?access_token=xxxxxxxxxxxxxxxxxxxxxxxxxxx&uid=xxxxxxxxxxxxxxx HTTP/1.1
Host: api.weibo.com
HTTP/1.1 200 OK
Content-Type: text/xml; charset=utf-8Content-Length: length 
这里的很多参数都是默认,在Xcode中创建NSURLConnection的时候会自动加上。
下面说一下POST。POST可以这样理解主要用于向服务器上传数据,简单一点的是上传文字,复杂一点的是上传图片声音等等。那么对于这种需求显然如果使用GET的话,那么这URL未免会太长。因此有POST方式的存在,可以在基本的URL的情况下,再附加一个数据包,包含要上传的数据,然后和URL一起发送给服务器。
POST方法有两种格式,一种是基本的格式,一般用于发送文字信息。
Post请求基本格式如下:
POST /login.asp HTTP/1.1
Host: www.example.com
Content-Type: application/x-www-form-urlencoded
Content-Length: 35
username=wantsoft&password=password   //post的数据
另一种是multipart/form-data,格式如下
POST /upload_file/UploadFile HTTP/1.1 
Accept: text/plain, */* 
Accept-Language: zh-cn 
Host: www.example.com 
Content-Type:multipart/form-data;boundary=---------------------------7d33a816d302b6 
User-Agent: Mozilla/4.0 (compatible; OpenOffice.org) 
Content-Length: 424 
Connection: Keep-Alive -----------------------------7d33a816d302b6 
Content-Disposition:form-data; 
name="userfile"; 
filename="userfile"
Content-Type: application/octet-stream abbXXXccc 
-----------------------------7d33a816d302b6 
Content-Disposition: form-data; 
name="text1" foo 
-----------------------------7d33a816d302b6 
Content-Disposition: form-data; 
name="password1" bar 
-----------------------------7d33a816d302b6-- 
看起来比较复杂,其实就是把请求数据通过分界线也就是Boundary分开,至于开头的一些内容,很多都是默认,无需考虑。
2、Xcode编程实现
2.1 HTTP请求的基本方法
STEP 1: 创建URL
一般我们用于URL的字符串。通过字符串创建URL
 NSString URLString;
NSURL *URL = [NSURLURLWithString:[URLStringstringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]];
上面附加的stringByAddingPercentEscapesUsingEncoding:是为了保证URL字符串有效。
STEP 2:创建URLRequest
NSMutableURLRequest *URLRequest = [[NSMutableURLRequestalloc]initWithURL:finalURLcachePolicy:NSURLRequestReloadIgnoringLocalCacheDatatimeoutInterval:SR_TIME_OUT_INTERVAL];
//如果是GET
    [URLRequest setHTTPMethod:@"GET"];
//如果是POST
[URLRequest setHTTPBody:[HTTPBodyStringdataUsingEncoding:NSUTF8StringEncoding]];
[URLRequestsetHTTPMethod:@"POST"];
STEP 3:通过NSURLConnection实现请求
一般有两种方式,如果不需要知道请求的进度,比如文字上传,那么用简单的异步请求。如下:
NSOperationQueue *queue = [[NSOperationQueuealloc]init];
[NSURLConnectionsendAsynchronousRequest:self.URLRequestqueue:queuecompletionHandler:^(NSURLResponse *response,NSData *data,NSError *error){...};
如果要知道请求的进度,比如图片发送的情况。那么就应该创建一个NSURLConnection的实例,并设置其delegate,通过其delegate方法来获取请求的实时状态,具体使用方法详见开发文档。
2.2 HTTP GET方法的实现
主要讲如何创建NSURLRequest
这里我直接将我写的方法放下面:
- (NSURLRequest *)HTTPGETRequestForURL:(NSURL *)url parameters:(NSDictionary *)parameters
{
   NSString *URLFellowString = [@"?"stringByAppendingString:[selfHTTPBodyWithParameters:parameters]];
    
    NSString *finalURLString = [[url.absoluteStringstringByAppendingString:URLFellowString]stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    
   NSURL *finalURL = [NSURLURLWithString:finalURLString];
    
    NSMutableURLRequest *URLRequest = [[NSMutableURLRequestalloc]initWithURL:finalURLcachePolicy:NSURLRequestReloadIgnoringLocalCacheDatatimeoutInterval:TIME_OUT_INTERVAL];
    [URLRequestsetHTTPMethod:@"GET"];
    
   return URLRequest;
    
}
- (NSString *)HTTPBodyWithParameters:(NSDictionary *)parameters
{
   NSMutableArray *parametersArray = [[NSMutableArrayalloc]init];
    
   for (NSString *keyin [parametersallKeys]) {
       id value = [parameters objectForKey:key];
       if ([value isKindOfClass:[NSStringclass]]) {
            [parametersArrayaddObject:[NSStringstringWithFormat:@"%@=%@",key,value]];
        }
        
    }
    
   return [parametersArray componentsJoinedByString:@"&"];
}
基本上就是通过方法把请求参数与URL连接起来,然后创建NSURLRequest
2.3 HTTP POST 基本方法的实现
- (NSURLRequest *)HTTPPOSTNormalRequestForURL:(NSURL *)url parameters:(NSDictionary *)parameters
{
     NSMutableURLRequest *URLRequest = [[NSMutableURLRequestalloc]initWithURL:urlcachePolicy:NSURLRequestReloadIgnoringLocalCacheDatatimeoutInterval:TIME_OUT_INTERVAL];
   NSString *HTTPBodyString = [selfHTTPBodyWithParameters:parameters];
    [URLRequest setHTTPBody:[HTTPBodyStringdataUsingEncoding:NSUTF8StringEncoding]];
    [URLRequestsetHTTPMethod:@"POST"];
   return URLRequest;
    
}
这个和GET极其类似,只不过是把请求参数放在了HTTPBody中了。
2.4 HTTP POST Multipart
- (NSURLRequest *)HTTPPOSTMultipartRequestForURL:(NSURL *)url parameters:(NSDictionary *)parameters
{
    NSMutableURLRequest *URLRequest = [[NSMutableURLRequestalloc]initWithURL:urlcachePolicy:NSURLRequestReloadIgnoringLocalCacheDatatimeoutInterval:TIME_OUT_INTERVAL];
    
    NSString *contentType = [NSStringstringWithFormat:@"multipart/form-data;boundary=%@",SR_POST_BOUNDARY];
// 设置内容的类型 contentType,这完全是按照POST的格式来设置
    [URLRequestsetValue:contentType forHTTPHeaderField:@"Content-Type"];
    [URLRequestsetHTTPMethod:@"POST"];
    [URLRequest setHTTPBody:[selfHTTPPOSTMultipartBodyWithParameters:parameters]];
   return URLRequest;
}
// 用于创建Multipart Body,需要分割线,也是完全参考POST格式
- (NSData *)HTTPPOSTMultipartBodyWithParameters:(NSDictionary *)parameters
{
    NSMutableData *body = [NSMutableDatadata];
    
    // Add Body Prefix String
    [body appendData:[[NSStringstringWithFormat:@"--%@\r\n",SR_POST_BOUNDARY]dataUsingEncoding:NSUTF8StringEncoding]];
    
    // Add Main Body
   for (NSString *keyin [parametersallKeys]) {
       id value = [parameters objectForKey:key];
        
       if ([value isKindOfClass:[NSStringclass]]){
            [body appendData:[[NSStringstringWithFormat:@"Content-Disposition: form-data; name=\"%@\"\r\n\r\n%@\r\n",key,value]dataUsingEncoding:NSUTF8StringEncoding]];
            [body appendData:[[NSStringstringWithFormat:@"--%@\r\n",SR_POST_BOUNDARY]dataUsingEncoding:NSUTF8StringEncoding]];
        }else {
            NSLog(@"please use addMultiPartData:withName:type:filename: Methods to implement");
        }
    }
   return body;
}
// 在最后要附加上请求参数中的图片等等大的数据!
- (void)addMultiPartData:(NSData *)data
                withName:(NSString *)name
                    type:(NSString *)type
                filename:(NSString *)filename
{
   NSMutableURLRequest *URLRequest = [self.URLRequestmutableCopy];
   NSMutableData *body = [URLRequest.HTTPBodymutableCopy];
    
    // Step 1
   NSString *disposition = [[NSStringalloc]init];
   if (!filename) {
        disposition =
        [NSStringstringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, filename];
    }else {
        disposition =
        [NSStringstringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, name];
    }
    [body appendData:[dispositiondataUsingEncoding:NSUTF8StringEncoding]];
    
    // Step 2
   NSString *contentType = [NSStringstringWithFormat:@"Content-Type: %@\r\n\r\n",type];
    [body appendData:[contentTypedataUsingEncoding:NSUTF8StringEncoding]];
    
    // Step 3
    [bodyappendData:data];
    
    // Step 4 Add suffix boundary
   NSString *boundary = [NSStringstringWithFormat:@"\r\n--%@--\r\n",SR_POST_BOUNDARY];
    [body appendData:[boundarydataUsingEncoding:NSUTF8StringEncoding]];
    
    // Step 5
    [URLRequestsetHTTPBody:body];
   self.URLRequest = URLRequest;
}
3、下面说一下接受数据
一般我们要把NSData数据变成JSON数据,大都是情况下获得的JSON Object是NSDictionary或NSArray
基本方法是
id receivedData = [NSJSONSerializationJSONObjectWithData:dataoptions:NSJSONReadingMutableContainers |NSJSONReadingMutableLeaveserror:nil];
然后就可以进行相关的处理了!

你可能感兴趣的:(iOS通过http post上传图片)