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

113

У меня есть NSString:

http://www.

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

http%3A%2F%2Fwww.

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

  • 1
    У меня есть зашифрованная строка типа ùÕ9y^VêÏÊEØ®.ú/V÷ÅÖêú2Èh~ - ни одно из приведенных ниже решений, похоже, не решает эту проблему!
Теги:
nsstring
urlencode

12 ответов

282
Лучший ответ

Чтобы избежать символов, которые вы хотите, это немного больше.

Пример кода

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, требующие кодирования.

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

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

Использование Core Foundation без 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 \" % -. < > \^ _ `{|} ~"

  • 1
    Также обратите внимание, что вы можете использовать метод NSString -stringByAddingPercentEscapesUsingEncoding .
  • 0
    Этот код также __bridge_retained память, если вы не используете приведение __bridge_retained в ARC или не вызываете CFRelease в какой-то момент для возвращаемого значения из CFURLCreateStringByAddingPercentEscapes .
Показать ещё 10 комментариев
21

Он назвал кодировку URL. Подробнее здесь.

-(NSString *)urlEncodeUsingEncoding:(NSStringEncoding)encoding {
    return (NSString *)CFURLCreateStringByAddingPercentEscapes(NULL,
           (CFStringRef)self,
           NULL,
           (CFStringRef)@"!*'\"();:@&=+$,/?%#[]% ",
           CFStringConvertNSStringEncodingToEncoding(encoding));
}
  • 3
    Это было бы намного полезнее, если бы в ответ был включен какой-либо контент из размещенных вами ссылок.
  • 1
    CFURLCreateStringByAddingPercentEscapes() устарела. Используйте [NSString stringByAddingPercentEncodingWithAllowedCharacters:] .
7

Это не мое решение. Кто-то написал в 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, а не+. Это все.

  • 0
    Оригинальный код: stackoverflow.com/questions/3423545/…
  • 0
    что такое [self UTF8String]?
Показать ещё 1 комментарий
4
 NSString * encodedString = (NSString *)CFURLCreateStringByAddingPercentEscapes(NUL,(CFStringRef)@"parameter",NULL,(CFStringRef)@"!*'();@&+$,/?%#[]~=_-.:",kCFStringEncodingUTF8 );

NSURL * url = [[NSURL alloc] initWithString:[@"address here" stringByAppendingFormat:@"?cid=%@",encodedString, nil]];
  • 1
    освободить encodedString и URL. этот код о параметре кодирования. Для кодирования всего адреса передайте строку вместо «параметра».
  • 0
    Это сработало для меня ... закодировало даже символ &, с которым у меня были проблемы.
1

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

+ (NSString *)encodeUrlString:(NSString *)string {
return CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes
                         (kCFAllocatorDefault,
                          (__bridge CFStringRef)string,
                          NULL,
                          CFSTR("!*'();:@&=+$,/?%#[]"),
                          kCFStringEncodingUTF8)
                         );}
1

Swift iOS:

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

extension String {

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

}// end extension String
1
NSString *str = (NSString *)CFURLCreateStringByAddingPercentEscapes(
                             NULL,
                             (CFStringRef)yourString, 
                             NULL, 
                             CFSTR("/:"), 
                             kCFStringEncodingUTF8);

Вам нужно будет освободить или autorelease str самостоятельно.

0

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

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

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

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

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

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

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

    func decodeURIComponent() -> String {
        return self.componentsSeparatedByString("+").joinWithSeparator(" ").stringByRemovingPercentEncoding!
    }
}
0

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

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

-2

//используем метод экземпляра NSString следующим образом:

+ (NSString *)encodeURIComponent:(NSString *)string
{
NSString *s = [string stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
return s;
}

+ (NSString *)decodeURIComponent:(NSString *)string
{
NSString *s = [string stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
return s;
}

помните, что вы должны только кодировать или декодировать значение вашего параметра, а не весь URL-адрес, который вы запрашиваете.

  • 2
    stringByReplacingPercentEscapeusingencoding: только экранирует & и = :-(
  • 1
    Правильно, поэтому такие вещи, как +, не кодируются, когда они должны быть. Так что не используйте ответ выше
-7
int strLength = 0;
NSString *urlStr = @"http://www";
NSLog(@" urlStr : %@", urlStr );
NSMutableString *mutableUrlStr = [urlStr mutableCopy];
NSLog(@" mutableUrlStr : %@", mutableUrlStr );
strLength = [mutableUrlStr length];
[mutableUrlStr replaceOccurrencesOfString:@":" withString:@"%3A" options:NSCaseInsensitiveSearch range:NSMakeRange(0, strLength)];
NSLog(@" mutableUrlStr : %@", mutableUrlStr );
strLength = [mutableUrlStr length];
[mutableUrlStr replaceOccurrencesOfString:@"/" withString:@"%2F" options:NSCaseInsensitiveSearch range:NSMakeRange(0, strLength)];
NSLog(@" mutableUrlStr : %@", mutableUrlStr );

Ещё вопросы

Сообщество Overcoder
Наверх
Меню