[iPhone] HTTP Request/Response Wrapper 클래스 만들기

간단하게 HTTP 통신을 할 수 있는 방법을 찾던중에 매우 귀중한 자료를 보게 되었습니다.

iphoneos 커뮤니티의 강좌게시판에 있는 자료인데, 종이비행기님이 작성하신 [링크] 글입니다.

매우 설명을 친절하게 잘 써주셨습니다. 지금 제가 쓰는 글은 종이비행기님이 기 작성하신 글을 참고하여 조금 수정한 것임을 미리 알립니다.

종이비행기님이 작성하신 강좌를 조금 더 쓰기 편하게 추가해 보았습니다. 추가 된 사항은 델리게이트를 설정하여 데이터 수신이 끝나는 시점에 내가 원하는 메서드를 호출하여 데이터를 받아 볼 수 있게 하였고요, POST로 데이터를 전송할 때 함께 보낼 Body 데이터를 NSDictionary를 사용하면 자동으로 변환되도록 하였습니다.

HTTPRequestAppDelegate.m
[code]- (void)applicationDidFinishLaunching:(UIApplication *)application {
    // 접속할 주소 설정
    NSString *url = @”http://your.webpage.url”;
   
    // HTTP Request 인스턴스 생성
    HTTPRequest *httpRequest = [[HTTPRequest alloc] init];
   
    // POST로 전송할 데이터 설정
    NSDictionary *bodyObject = [NSDictionary dictionaryWithObjectsAndKeys:@”eye”,@”name”,@”http://theeye.pe.kr”, @”home”, nil];
   
    // 통신 완료 후 호출할 델리게이트 셀렉터 설정
    [httpRequest setDelegate:self selector:@selector(didReceiveFinished:)];
   
    // 페이지 호출
    [httpRequest requestUrl:url bodyObject:bodyObject];
   
    [window makeKeyAndVisible];
}[/code]
HTTPRequest.h
[code]#import <Foundation/Foundation.h>
 
@interface HTTPRequest : NSObject
{
    NSMutableData *receivedData;
    NSURLResponse *response;
    NSString *result;
    id target;
    SEL selector;
}
 
– (BOOL)requestUrl:(NSString *)url bodyObject:(NSDictionary *)bodyObject;
– (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)aResponse;
– (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data;
– (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error;
– (void)connectionDidFinishLoading:(NSURLConnection *)connection;
– (void)setDelegate:(id)aTarget selector:(SEL)aSelector;
 
@property (nonatomic, retain) NSMutableData *receivedData;
@property (nonatomic, retain) NSURLResponse *response;
@property (nonatomic, assign) NSString *result;
@property (nonatomic, assign) id target;
@property (nonatomic, assign) SEL selector;
 
@end[/code]
HTTPRequest.m
[code]#import “HTTPRequest.h”

@implementation HTTPRequest

@synthesize receivedData;
@synthesize response;
@synthesize result;
@synthesize target;
@synthesize selector;

– (BOOL)requestUrl:(NSString *)url bodyObject:(NSDictionary *)bodyObject
{
    // URL Request 객체 생성
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:url]
                                                           cachePolicy:NSURLRequestUseProtocolCachePolicy
                                                       timeoutInterval:5.0f];
   
    // 통신방식 정의 (POST, GET)
    [request setHTTPMethod:@”POST”];
   
    // bodyObject의 객체가 존재할 경우 QueryString형태로 변환
    if(bodyObject)
    {
        // 임시 변수 선언
        NSMutableArray *parts = [NSMutableArray array];
        NSString *part;
        id key;
        id value;
       
        // 값을 하나하나 변환
        for(key in bodyObject)
        {
            value = [bodyObject objectForKey:key];
            part = [NSString stringWithFormat:@”%@=%@”, [key stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding],
                                                        [value stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]];
            [parts addObject:part];
        }
       
        // 값들을 &로 연결하여 Body에 사용
        [request setHTTPBody:[[parts componentsJoinedByString:@”&”] dataUsingEncoding:NSUTF8StringEncoding]];
    }
   
    // Request를 사용하여 실제 연결을 시도하는 NSURLConnection 인스턴스 생성
    NSURLConnection *connection = [[[NSURLConnection alloc] initWithRequest:request delegate:self] autorelease];
   
    // 정상적으로 연결이 되었다면
    if(connection)
    {
        // 데이터를 전송받을 멤버 변수 초기화
        receivedData = [[NSMutableData alloc] init];
        return YES;
    }
   
    return NO;
}

– (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)aResponse
{
    // 데이터를 전송받기 전에 호출되는 메서드, 우선 Response의 헤더만을 먼저 받아 온다.
    //[receivedData setLength:0];
    self.response = aResponse;
}

– (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
    // 데이터를 전송받는 도중에 호출되는 메서드, 여러번에 나누어 호출될 수 있으므로 appendData를 사용한다.
    [receivedData appendData:data];
}

– (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
    // 에러가 발생되었을 경우 호출되는 메서드
    NSLog(@”Error: %@”, [error localizedDescription]);
}

– (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    // 데이터 전송이 끝났을 때 호출되는 메서드, 전송받은 데이터를 NSString형태로 변환한다.
    result = [[NSString alloc] initWithData:receivedData encoding:NSUTF8StringEncoding];
   
    // 델리게이트가 설정되어있다면 실행한다.
    if(target)
    {
        [target performSelector:selector withObject:result];
    }
}

– (void)setDelegate:(id)aTarget selector:(SEL)aSelector
{
    // 데이터 수신이 완료된 이후에 호출될 메서드의 정보를 담고 있는 셀렉터 설정
    self.target = aTarget;
    self.selector = aSelector;
}

– (void)dealloc
{
    [receivedData release];
    [response release];
    [result release];
    [super dealloc];
}

@end[/code]
위의 예제를 잠깐 다시 살펴보면 requestUrl:bodyObject: 가 호출될 때 bodyObject를 자동으로 queryString으로 변환하여 Request를 보내게 됩니다.

이후에 setDelegate:selector:를 통해 설정한 셀렉터 정보를 가지고 데이터 수신이 끝나 connectionDidFinishLoading: 가 호출되는 시점에 해당 메서드를 호출하게 됩니다. 테스트는 인터넷이 되는 환경에서 해야하고 다음과 같이 잘 되는것을 알 수 있습니다.
사용자 삽입 이미지1391674733.zip
참고 : http://iphoneos.co.kr/zbxe/5718

  • white10048

    안녕하세요~

    글 매우 잘 보았습니다~

    질문이 있는데요 제가 이 파일을 다운받아 실행시켜봤는데 위의 캡쳐화면처럼 나오지않고

    HTML소스만 나오더라구요 혹시 실행하기전에 설정해줘야 하는부분이있나요? 아니면 제가 뭘 잘못알고 실행했는지 ^^;; 답변 부탁드리겠습니다!

    • 아이

      위와 같은 표기 방법은 NSDictionary와 같은 key-value 변수를 그냥 %@로 찍어보면 저렇게 표기됩니다.

    • white10048

      음 그럼 에뮬레이터에는 아무것도 안뜨게 되는건가요?ㅎ

    • 아이

      아니요? 위의 화면 다 에뮬레이터 캡춰한건데…흠..

  • 안개속으로

    찾던 자료인데 감사합니다..
    connectionDidFinishLoading 델리게이트 함수에서 받아온
    HTML 데이터를 어떻게 넘겨야 하나 고민중 이었습니다.

    정말 감사합니다.

  • 리팩토링

    오픈해주신 샘플소스 정말 잘 쓰고 있습니다만
    궁금한게 있어 댓글 답니다.
    위와 같이 사용하고 있습니다만.
    [httpRequest release]
    로 객체를 해제하게 되면 뻗습니다.
    해제는 안해줘도 상관 없는건가요??

  • 안녕하세요? 정말 좋은 설명과 소스까지 감사드립니다.
    http://www.test.co.kr/a.php?a=“aaa”&b=”bbbb”

    저런식으로 만들어서 connection을 하고싶은데요
    그러려면 왠지 bodyObject 부분에서 ? 이후의 작업을 할것 같은데요..정확하게 어떻게 만들어지는지 궁금해서요..

    혹시 post할때 bodyObject까지 붙은 전체적인 url log 을 보여주실수 있으신지요?
    아직 개인적으로 연결할 서버가 전혀 없어서요..

    • HTTPRequest.m 파일의 bodyObject 쪼갰다가 붙이는부분 있죠?
      그부분이 QueryString으로 변환하는 부분이예요. 이미 원하시는데로 동작하는 소스지요.

  • 안되용;;

    올려주신 소스를 돌려봤는데요.

    빈 화면만 나타납니다.;;

    링크 수정하라는 곳을 수정 했는데도 그러네요.

    다른 곳도 수정해야 하나요?!

    • 당연히 안되겠죠??
      서버에서 데이터를 가져오는데 서버 정보는 수정하셨나요?
      서버에 데이터를 가진 페이지는 존재하나요?
      JSON이나 Ajax/REST에 대한 이해가 부족하시다면 이 예제가 어렵게 느껴지실수 있습니다.

  • yeNJOholic

    유용한 자료 감사합니다. 이 자료로 서버쪽 응답을 받는 것에 성공했습니다.

    다만, 서버쪽 응답 화면 처리 부분을 간략하게나마 조언해주시면 좋겠습니다.

    제가 받아오고자 하는 것은, 간단한 코드나 String 문자 정도 입니다.

    예를들어

    Login_OK
    Login_Fail
    User Not Found

    등입니다.

    현재는 응답페이지 쪽에 아무것도 없이 그냥
    (PHP 환경입니다.)

    echo “Login_OK”;

    정도로 표시해 놨는데… 그랬더니

    result 에 저 글자만 나오는게 아니라 엔터도 몇개 쳐 지고 그렇습니다.

    좀 더 세련되게 result 를 받아오는 방법이 있다면 알려주시면 정말 감사하겠습니다.

    • 빈칸이 들어간다면 PHP에서 처리상의 문제입니다. 파일에 빈줄이 들어가 있거나 해서 그게 그대로 나오는 것이겠죠. 그냥 편하게 하실려면 서버에서 데이터를 받고 트림 처리를 하시면 되겠네요.

      NSString *trimmedString = [description stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];

    • yeNJOholic

      아아…정말 감사합니다. ㅜㅜ 이 은혜를 어찌 갚을지…

      이 블로그 저에겐 정말 가뭄에 단비 같습니다.

  • 이미혜

    글 좀 퍼가겠습니다. 출처는 같이 게시하겠습니다 ^^
    문제 되실경우 자삭 하겠습니다 ^^
    제 메일 주소는 soundeshop@gmail.com 입니다
    감사합니다.

  • messfilm

    안녕하세요, 위 소스를 응용해서 구현중에 막히는 부분이 있어서 질문드립니다.

    제가 테이블 뷰를 구성할 때
    url 을 던져서 xml Data 를 가져오고… 그 후에 가져온 데이터를 파싱해서 사용하는데요..

    님께서 구현해 주신 HttpRequest 를 앞에 넣으려고 합니다.

    기존의 구현방식은

    tableview 컨트롤러에서 viewdidload 시에 특정 메서드를 수행해서…

    xml 파싱값을 리턴받아서 사용했거든요

    NSMutableArray *arr = [xxx.getXML :url];
    이런식으로요.

    그런데 저 getXML 메서드에 httpRequest 를 집어 넣으려니… 얘는 리퀘스트해서 얻어온 xml String 을… 델리게이트 셀렉터를 통해서 result 로 던져주더군요..

    그렇게 되면 애초에 제가 불러온 getXML 이 아니라
    리턴값을 안주더라구요…

    여기서 딱 막혀버렸습니다. 위 소스를 이용하고 싶은데
    어떻게 해야 파싱이 끝난 MutableArray 를 리턴할 수 있을까요?

  • 서산대사

    아주 유용한 내용이었습니다.
    감사합니다.

  • 탄돌

    감사합니다.
    아이군님의 방법을 아주 요긴하게 사용해 보고 있습니다. ^^

    그런데 다음과 같은 경고 메세지가 뜨는군요

    warning: local declaration of ‘httpRequest’ hides instance variable

    iOS 4.1 한테 무얼 해주어야 이 징징거림을 잠재울까요?

    • httpRequest라는 이름으로 지역변수와 맴버변수를 중복 선언하신건 아니신지요?

  • 얼렁뚱땅

    안녕하세요~~ 위 소스 정말 유용하게 쓰고있습니다.
    사용중에 궁금증이 생겨 문의 드립니다..

    예를 들어
    위에 소스를 사용하여 네이버버 같은데는 로긴할려고 하면
    로그인 부분이 다른 프레임으로 되어있어..
    url : http://static.nid.naver.com/login.nhn?20091104&url=http%3A%2F%2Fwww.naver.com
    설정하였습니다..
    그렇게 해서 post 메세지 작성하여 id, 와 pw값을 넘기면 결과는 url로 넘겼던 홈페이지 소스가 나옵니다
    결과를 http://www.naver.com 이쪽에서 받을려면 어떻게 해야할지…
    많은 가르침 부탁드립니다.

    • 무슨 말씀이신지;;; url로 넘기면 결과가 그리로 넘어가는거 아닌가요?
      질문을 이해하기가 어렵네요;

  • 얼렁뚱땅

    ㅎㅎ 제가 설명을 잘 못했나 보네요..
    네이버 로그인을 위의 소스로 하려고했습니다.
    그런데 네이버는 로그인 부분이 다른 프레임으로 작성되어있어서 url값을 로그인 부분의 프레임 값으로
    아래와 같이 설정하였고,
    url : http://static.nid.naver.com/login.nhn?20091104&url=http%3A%2F%2Fwww.naver.com

    그렇게 해서 post 메세지 작성하여 id, 와 pw값을 넘기면
    결과로 받는 recive data 는 url로 넘겼던 프레임 홈페이지 소스가 나옵니다
    결과를 http://www.naver.com 이쪽에서 로그인된 상태의 결과를 받을려면 어떻게 해야할지…
    많은 가르침 부탁드립니다.

  • 아이님 ~ ? 이렇게 불러야하나요 ㅋㅋ?
    안녕하세요 블로그 게시물 잘 보고가요 ^ㅡ^

  • 박카스소년

    안녕하세요 objective C에서 전혀 감을 못잡고 있어서 메모리 관리 때문에 좀 헤매고 있는데요.

    HTTPRequestAppDelegate.m 에서 보면 httpRequest를 alloc하고 해제해 주지 않고 있는데, requets를 보낸 다음 즉시 해제를 하면 connectionDidFinishLoading 을 부르지 못하고 뻗는것 같더라고요. 그래서 궁여지책으로 connectionDidFinishLoading 안에다가 [self release]; 라는 구문을 집어넣었는데 그렇게 쓰다보니 영 코드가 이상해보여서 -_-;; release 시점을 어떻게 잡아야 할까요?

    • 저라면 멤버변수로 잡아놓고 뷰컨트롤러가 제거될때(dealloc) release할꺼 같네요.

  • groove

    좋은 정보 감사드립니다! 😀

  • co

    감사합니다 =)