Кодировка URL Objective-C и Swift - PullRequest
       37

Кодировка URL Objective-C и Swift

134 голосов
/ 11 ноября 2011

У меня есть NSString, как это:

http://www.

но я хочу преобразовать его в:

http%3A%2F%2Fwww.

Как я могу это сделать?

Ответы [ 12 ]

318 голосов
/ 11 ноября 2011

Убежать от нужных вам персонажей - немного больше работы.

Пример кода

iOS7 и выше:

NSString *unescaped = @"http://www";
NSString *escapedString = [unescaped stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLHostAllowedCharacterSet]];
NSLog(@"escapedString: %@", escapedString);

Вывод NSLog:

escapedString: http% 3A% 2F% 2Fwww

Ниже приведены полезные наборы символов кодирования URL:

URLFragmentAllowedCharacterSet  "#%<>[\]^`{|}
URLHostAllowedCharacterSet      "#%/<>?@\^`{|}
URLPasswordAllowedCharacterSet  "#%/:<>?@[\]^`{|}
URLPathAllowedCharacterSet      "#%;<>?[\]^`{|}
URLQueryAllowedCharacterSet     "#%<>[\]^`{|}
URLUserAllowedCharacterSet      "#%/:<>?@[\]^`

Создание набора символов, объединяющего все вышеперечисленное:

NSCharacterSet *URLCombinedCharacterSet = [[NSCharacterSet characterSetWithCharactersInString:@" \"#%/:<>?@[\\]^`{|}"] invertedSet];

Создание Base64

В случае набора символов Base64:

NSCharacterSet *URLBase64CharacterSet = [[NSCharacterSet characterSetWithCharactersInString:@"/+=\n"] invertedSet];

Для Swift 3.0:

var escapedString = originalString.addingPercentEncoding(withAllowedCharacters:.urlHostAllowed)

Для Swift 2.x:

var escapedString = originalString.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLHostAllowedCharacterSet())

Примечание: stringByAddingPercentEncodingWithAllowedCharacters также будет кодировать символы UTF-8, требующие кодирования.

Предварительно iOS7 использует Core Foundation
Использование Core Foundation с ARC:

NSString *escapedString = (NSString *)CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes(
    NULL,
   (__bridge CFStringRef) unescaped,
    NULL,
    CFSTR("!*'();:@&=+$,/?%#[]\" "),
    kCFStringEncodingUTF8));

Использование базовой основы без ARC:

NSString *escapedString = (NSString *)CFURLCreateStringByAddingPercentEscapes(
    NULL,
   (CFStringRef)unescaped,
    NULL,
    CFSTR("!*'();:@&=+$,/?%#[]\" "),
    kCFStringEncodingUTF8);

Примечание: -stringByAddingPercentEscapesUsingEncoding не даст правильную кодировку, в этом случае он не будет кодировать что-либо, возвращающее ту же строку.

stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding кодирует 14 символов:

`#% ^ {} [] | \" <> плюс символ пробела в процентах.

testString:

" `~!@#$%^&*()_+-={}[]|\\:;\"'<,>.?/AZaz"  

encodedString:

"%20%60~!@%23$%25%5E&*()_+-=%7B%7D%5B%5D%7C%5C:;%22'%3C,%3E.?/AZaz"  

Примечание: подумайте, соответствует ли этот набор символов вашим потребностям, если не измените их при необходимости.

RFC 3986 символов, требующих кодирования (добавлено%, поскольку это префикс кодировки):

"# $ & '() * +, /:;!? = @ []%"

Некоторые "незарезервированные символы" дополнительно кодируются:

"\ n \ r \"% -. <> \ ^ _ `{|} ~"

22 голосов
/ 11 ноября 2011

Это называется Кодировка URL .Подробнее здесь .

-(NSString *)urlEncodeUsingEncoding:(NSStringEncoding)encoding {
    return (NSString *)CFURLCreateStringByAddingPercentEscapes(NULL,
           (CFStringRef)self,
           NULL,
           (CFStringRef)@"!*'\"();:@&=+$,/?%#[]% ",
           CFStringConvertNSStringEncodingToEncoding(encoding));
}
7 голосов
/ 17 октября 2012

Это не мое решение.Кто-то еще написал в stackoverflow, но я забыл, как.

Каким-то образом это решение работает "хорошо".Он обрабатывает диакритические знаки, китайские и многое другое.

- (NSString *) URLEncodedString {
    NSMutableString * output = [NSMutableString string];
    const char * source = [self UTF8String];
    int sourceLen = strlen(source);
    for (int i = 0; i < sourceLen; ++i) {
        const unsigned char thisChar = (const unsigned char)source[i];
        if (false && thisChar == ' '){
            [output appendString:@"+"];
        } else if (thisChar == '.' || thisChar == '-' || thisChar == '_' || thisChar == '~' ||
                   (thisChar >= 'a' && thisChar <= 'z') ||
                   (thisChar >= 'A' && thisChar <= 'Z') ||
                   (thisChar >= '0' && thisChar <= '9')) {
            [output appendFormat:@"%c", thisChar];
        } else {
            [output appendFormat:@"%%%02X", thisChar];
        }
    }
    return output;
}

Если кто-нибудь скажет мне, кто написал этот код, я буду очень признателен.В основном у него есть какое-то объяснение, почему эта закодированная строка будет декодироваться именно так, как она хочет.

Я немного изменил его решение.Мне нравится, когда пространство обозначается% 20, а не +Вот и все.

4 голосов
/ 30 августа 2012
 NSString * encodedString = (NSString *)CFURLCreateStringByAddingPercentEscapes(NUL,(CFStringRef)@"parameter",NULL,(CFStringRef)@"!*'();@&+$,/?%#[]~=_-.:",kCFStringEncodingUTF8 );

NSURL * url = [[NSURL alloc] initWithString:[@"address here" stringByAppendingFormat:@"?cid=%@",encodedString, nil]];
3 голосов
/ 10 ноября 2016

Это может работать в Objective C ARC. Используйте CFBridgingRelease для приведения объекта базового стиля в качестве объекта Objective-C и передачи владения объектом в ARC. См. Функция CFBridgingRelease здесь.

+ (NSString *)encodeUrlString:(NSString *)string {
return CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes
                         (kCFAllocatorDefault,
                          (__bridge CFStringRef)string,
                          NULL,
                          CFSTR("!*'();:@&=+$,/?%#[]"),
                          kCFStringEncodingUTF8)
                         );}
2 голосов
/ 17 апреля 2015

Swift iOS:

Только для информации: я использовал это:

extension String {

    func urlEncode() -> CFString {
        return CFURLCreateStringByAddingPercentEscapes(
            nil,
            self,
            nil,
            "!*'();:@&=+$,/?%#[]",
            CFStringBuiltInEncodings.UTF8.rawValue
        )
    }

}// end extension String
1 голос
/ 11 ноября 2011
NSString *str = (NSString *)CFURLCreateStringByAddingPercentEscapes(
                             NULL,
                             (CFStringRef)yourString, 
                             NULL, 
                             CFSTR("/:"), 
                             kCFStringEncodingUTF8);

Вам нужно будет самостоятельно разблокировать или выпустить str.

0 голосов
/ 15 июля 2016

Вот что я использую.Обратите внимание, что вы должны использовать функцию @autoreleasepool, иначе программа может аварийно завершить работу или заблокировать IDE.Мне пришлось три раза перезапускать свою IDE, пока я не понял, что исправить.Похоже, что этот код соответствует ARC.

Этот вопрос задавался много раз, и было дано много ответов, но, к сожалению, все выбранные (и несколько других предложенных) неверны.

Вот тестовая строка, которую я использовал: This is my 123+ test & test2. Got it?!

Это мои методы класса Objective C ++:

static NSString * urlDecode(NSString *stringToDecode) {
    NSString *result = [stringToDecode stringByReplacingOccurrencesOfString:@"+" withString:@" "];
    result = [result stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    return result;
}

static NSString * urlEncode(NSString *stringToEncode) {
    @autoreleasepool {
        NSString *result = (NSString *)CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes(
                NULL,
                (CFStringRef)stringToEncode,
                NULL,
                (CFStringRef)@"!*'\"();:@&=+$,/?%#[]% ",
                kCFStringEncodingUTF8
            ));
        result = [result stringByReplacingOccurrencesOfString:@"%20" withString:@"+"];
        return result;
    }
}
0 голосов
/ 16 апреля 2016

Вот как я делаю это по-быстрому.

extension String {
    func encodeURIComponent() -> String {
        return self.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLQueryAllowedCharacterSet())!
    }

    func decodeURIComponent() -> String {
        return self.componentsSeparatedByString("+").joinWithSeparator(" ").stringByRemovingPercentEncoding!
    }
}
0 голосов
/ 04 февраля 2016

Google реализует это в своих Google Toolbox для Mac . Так что это хорошее место, чтобы показать, как они это делают. Другой вариант - включить Toolbox и использовать их реализацию.

Оформить заказ здесь . (Который сводится к тому, что люди публикуют здесь).

...