NSURL с фигурными скобками - PullRequest
2 голосов
/ 01 декабря 2011

NSURL не поддерживает использование фигурных скобок (т. Е. {}) В URL.Мое приложение должно общаться с сервером, который требует использования фигурных скобок в URL.Я рассматриваю возможность использования мостов для написания сетевого кода на Python или C ++ или даже переписывания кода C для NSURL, чтобы он мог принимать фигурные скобки.Процентные побеги не принимаются моим удаленным сервером.

Есть ли у меня какие-либо другие хорошие альтернативы?

РЕДАКТИРОВАТЬ: Объяснил, почему addPercentEscapesUsingEncoding и тому подобное у меня не работают.

Ответы [ 3 ]

10 голосов
/ 01 декабря 2011

Будет ли это работать для вас, если вы избежите скобок?

Этот код:

// escape {} with %7B and %7D
NSURL *url = [NSURL URLWithString:@"http://somesite.com/%7B7B643FB915-845C-4A76-A071-677D62157FE07D%7D.htm"];
NSLog(@"%@", url);

// {} don't work and return null
NSURL *badurl = [NSURL URLWithString:@"http://somesite.com/{7B643FB915-845C-4A76-A071-677D62157FE07D}.htm"];
NSLog(@"%@", badurl);

Выходы:

2011-11-30 21:25:06.655 Craplet[48922:707] http://somesite.com/%7B7B643FB915-845C-4A76-A071-677D62157FE07D%7D.htm
2011-11-30 21:25:06.665 Craplet[48922:707] (null)

Итак, похоже, что экранирование работает

Вот как вы можете программно избежать URL:

NSString *escapedUrlString = [unescaped stringByAddingPercentEscapesUsingEncoding: NSASCIIStringEncoding];

РЕДАКТИРОВАТЬ:

В своем комментарии ниже вы сказаливаш сервер не будет принимать закодированные скобки, и были ли другие альтернативы.Сначала я попытался бы починить сервер.Если это невозможно ... Я не пробовал это с фигурными скобками и т. Д. ... но уровень ниже сетевых классов NS - CFNetworking.

См. Это:

http://developer.apple.com/library/ios/#documentation/Networking/Conceptual/CFNetwork/CFHTTPTasks/CFHTTPTasks.html#//apple_ref/doc/uid/TP30001132-CH5-SW2

Из этого документа:

 CFStringRef url = CFSTR("http://www.apple.com");
 CFURLRef myURL = CFURLCreateWithString(kCFAllocatorDefault, url, NULL);
 CFStringRef requestMethod = CFSTR("GET");

 ....

Еще раз, не пробовал, и у меня заканчивается.возможно, попробуйте позже, но если слой не работает, ваши первые варианты - переместиться вниз по стеку.

8 голосов
/ 01 декабря 2011

Ответ Брайана совершенно великолепен (+1 к нему!), Но другим решением было бы использование метода stringByAddingPercentEscapesUsingEncoding: NSString.

Надеюсь, эта информация поможет вам!

0 голосов
/ 08 июля 2014

Для звонков с использованием CFNetwork можно использовать следующее

-(void)getDataFromUrl{
    CFStringRef tempURLA = CFSTR("http://my.test.server/iostest/index.html?{\"a\":\"b\"}");
    CFStringRef tempUrlSting = CFURLCreateStringByAddingPercentEscapes(kCFAllocatorDefault, (CFStringRef)tempURLA,CFSTR("{}"), CFSTR("\""), CFStringConvertNSStringEncodingToEncoding(NSUTF8StringEncoding));

    CFURLRef myURL = CFURLCreateWithString(kCFAllocatorDefault, tempUrlSting, NULL);

    CFStringRef requestMethod = CFSTR("GET");
    CFHTTPMessageRef myRequest = CFHTTPMessageCreateRequest(kCFAllocatorDefault, requestMethod, myURL,kCFHTTPVersion1_1);

    CFStringRef headerFieldName = CFSTR("Accept");
    CFStringRef headerFieldValue = CFSTR("text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8");
    CFHTTPMessageSetHeaderFieldValue(myRequest, headerFieldName, headerFieldValue);


    [self performHTTPRequest:myRequest];

}

-(void)performHTTPRequest:(CFHTTPMessageRef)request {
    CFURLRef gotdatab = (__bridge CFURLRef)(CFBridgingRelease(CFHTTPMessageCopyRequestURL(request)));
//    NSLog(@"(CFHTTPMessageRef request  %@",gotdatab);

    CFReadStreamRef requestStream = CFReadStreamCreateForHTTPRequest(NULL, request);

    CFReadStreamOpen(requestStream);

    NSMutableData *responseBytes = [NSMutableData data];
    NSError *error;

    while (TRUE) {

        if (CFReadStreamHasBytesAvailable(requestStream)) {
            UInt8 streambuffer[1024];
            int readBytes = CFReadStreamRead (requestStream,streambuffer,sizeof(streambuffer));
            NSLog(@"Read: %d",readBytes);
            [responseBytes appendBytes:streambuffer length:readBytes];
        }

        if (CFReadStreamGetStatus(requestStream) == kCFStreamStatusError) {

            error = (NSError*)CFBridgingRelease(CFReadStreamCopyError (requestStream));
            if ([error code] == 61) {
                // connection refused
                NSLog(@"Error occured: %d",[error code]);
            }
            break;
        }
        if (CFReadStreamGetStatus(requestStream) == kCFStreamStatusAtEnd) {
            NSLog(@"Stream reached end!");
            error = nil;
            break;
        }

    }//
    CFHTTPMessageRef response = (CFHTTPMessageRef)CFReadStreamCopyProperty(requestStream, kCFStreamPropertyHTTPResponseHeader);

    if (response==NULL) {
        NSLog(@"response is null");
        return;
    }
}

Выше было сделано с использованием примеров из здесь и здесь

Но даже вышеописанный метод имеет ту же проблему. То есть: если {} не закодировано, URL не генерируется. Если закодированы {}, сервер не возвращает правильное значение.

Есть ли другие предложения по решению этой проблемы?

...