Как мне URL кодировать строку


Ответы:


291

К сожалению, stringByAddingPercentEscapesUsingEncodingне всегда работает на 100%. Он кодирует не-URL-символы, но оставляет зарезервированные символы (такие как косая черта /и амперсанд &) в одиночку. Очевидно, это ошибка, о которой Apple знает, но поскольку они еще не исправили ее, я использовал эту категорию для URL-кодирования строки:

@implementation NSString (NSString_Extended)

- (NSString *)urlencode {
    NSMutableString *output = [NSMutableString string];
    const unsigned char *source = (const unsigned char *)[self UTF8String];
    int sourceLen = strlen((const char *)source);
    for (int i = 0; i < sourceLen; ++i) {
        const unsigned char thisChar = source[i];
        if (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;
}

Используется так:

NSString *urlEncodedString = [@"SOME_URL_GOES_HERE" urlencode];

// Or, with an already existing string:
NSString *someUrlString = @"someURL";
NSString *encodedUrlStr = [someUrlString urlencode];

Это также работает:

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

Некоторое хорошее чтение о предмете:

Objective-c iPhone проц кодирует строку?
Кодировка URL Objective-C и Swift

http://cybersam.com/programming/proper-url-percent-encoding-in-ios
https://devforums.apple.com/message/15674#15674 http://simonwoodside.com/weblog/2009/4/ 22 / how_to_really_url_encode /


12
С какой стати вы будете использовать сложную категорию, подобную этой, а не просто опускаться до CFURLCreateStringByAddingPercentEscapes () , где вы можете явно указать, какие символы вы всегда хотите экранировать.
Лили Баллард

8
@KevinBallard, потому что функция CF работает на инклюзивной модели («экранировать эти символы»), и обычно вы хотите работать с эксклюзивной моделью («экранировать все, кроме этих символов»)
Дейв ДеЛонг,


31
@DaveDeLong Это может быть, где я получил это. Это была стандартная категория во всех моих проектах в течение года или около того, так что я думаю, что вы, возможно, были первоисточником! Я отредактировал формулировку, так что не похоже, что я пытаюсь взять кредит на ее написание).
chown

4
Зачем? if (thisChar == '') {[output appendString: @ "+"]; } Без этого, если он будет кодировать пробелы с% 20, как я и ожидал
Крис Стивенс,

127

Это может быть полезно

NSString *sampleUrl = @"http://www.google.com/search.jsp?params=Java Developer";
NSString* encodedUrl = [sampleUrl stringByAddingPercentEscapesUsingEncoding:
 NSUTF8StringEncoding];

Для iOS 7+ рекомендуемый способ:

NSString* encodedUrl = [sampleUrl stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];

Вы можете выбрать разрешенный набор символов в соответствии с требованием компонента URL.


7
Это также неправильно кодирует разделители параметров, такие как «&», «?» в случае, когда вы передаете контент в API.
Бен Лахман

'&' не является допустимым символом для параметра строки запроса URL. Пожалуйста, попробуйте использовать & amp; вместо.
Nishant

1
stringByAddingPercentEscapesUsingEncoding is DEPRECATED "Использовать -stringByAddingPercentEncodingWithAllowedCharacters:вместо этого, который всегда использует рекомендованную кодировку UTF-8 и который кодирует для конкретного компонента или подкомпонента URL, поскольку каждый компонент или подкомпонент URL имеет разные правила для допустимых символов."
Михир Оза

89

Новые API были добавлены с момента выбора ответа; Теперь вы можете использовать NSURLUtilities. Поскольку разные части URL допускают разные символы, используйте соответствующий набор символов. В следующем примере кодируется для включения в строку запроса:

encodedString = [myString stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLQueryAllowedCharacterSet];

Чтобы специально преобразовать '&', вам нужно удалить его из набора URL-запросов или использовать другой набор, так как '&' разрешено в URL-запросе:

NSMutableCharacterSet *chars = NSCharacterSet.URLQueryAllowedCharacterSet.mutableCopy;
[chars removeCharactersInRange:NSMakeRange('&', 1)]; // %26
encodedString = [myString stringByAddingPercentEncodingWithAllowedCharacters:chars];

1
Мне потребовалось некоторое время, чтобы понять, что вы имели в виду под «использовать NSURLUtilities» - но теперь я вижу, что это название категории NSString, в которой Apple определила этот новый метод в iOS 7 и OS X 10.9.
Ричард Венейбл

1
Вот более подробный ответ, подобный этому, здесь: stackoverflow.com/a/20271177/456366
Ричард Венейбл,

Ага. Он также был указан как NSURLUtilities в списке новых выпущенных API.
Питер ДеВиз

'&', конечно, разрешено в строке запроса, поэтому я отредактирую свой ответ, используя ссылку Ричарда.
Питер ДеВиз

3
NSMakeRange('&', 1)не работает в Swift, потому что Swift не позволяет конвертировать char в int без взлома. Чтобы использовать это решение в коде Swift, используйте removeCharactersInString("&")вместо.removeCharactersInRange(...)
cbh2000

16

Пример Swift 2.0 (совместимо с iOS 9)

extension String {

  func stringByURLEncoding() -> String? {

    let characters = NSCharacterSet.URLQueryAllowedCharacterSet().mutableCopy() as! NSMutableCharacterSet

    characters.removeCharactersInString("&")

    guard let encodedString = self.stringByAddingPercentEncodingWithAllowedCharacters(characters) else {
      return nil
    }

    return encodedString

  }

}

2
@AlecThomas, кажется, тебе нужно прыгнуть через несколько обручей, чтобы попасть сюда, поэтому лучше спрятать его в расширении
Оливер Аткинсон,

@OliverAtkinson - для меня Int (String (Character ("&"))) возвращает nil, как и должно быть. Вместо characters.removeCharactersInString ("&").
ambientlight

14

обновление ios 7

NSString *encode = [string stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];

NSString *decode = [encode stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];

Это неточно. Вы хотите по-разному избегать различных частей URL. stackoverflow.com/questions/3423545/…
Стив Мозер

На самом деле это зависит от вашего сервера для допустимых символов, здесь на моем примере на сервере, который я использовал, он разрешал только буквенно-цифровой набор символов, вы можете изменить его на конкретные символы для удовлетворения ваших потребностей.
Underdog

1
Вопрос задает вопрос о кодировке символа «&».
Стив Мозер

8

Я решил использовать CFURLCreateStringByAddingPercentEscapesвызов в соответствии с принятым ответом, однако в новейшей версии XCode (и IOS) это привело к ошибке, поэтому вместо этого использовал следующее:

NSString *apiKeyRaw = @"79b|7Qd.jW=])(fv|M&W0O|3CENnrbNh4}2E|-)J*BCjCMrWy%dSfGs#A6N38Fo~";

NSString *apiKey = (NSString *)CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes(NULL, (CFStringRef)apiKeyRaw, NULL, (CFStringRef)@"!*'();:@&=+$,/?%#[]", kCFStringEncodingUTF8));

Это не пространство кодирования. Пример @ "string string" и @ "string & string" закодирован правильно. Пожалуйста, дайте мне знать ваш вклад в том же.
Йогеш Лолусаре

2
для меня большую часть времени, потраченного на решение этой проблемы, было периодом отказа, когда я отказывался верить, что среда не содержит чего-то похожего на 'urlencode', что делает это без
возни

6

Попробуй использовать stringByAddingPercentEncodingWithAllowedCharacters метод с [NSCharacterSet URLUserAllowedCharacterSet]ним охватит все случаи

Цель С

NSString *value = @"Test / Test";
value = [value stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLUserAllowedCharacterSet]];

быстрый

var value = "Test / Test"
value.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLUserAllowedCharacterSet())

Вывод

Test%20%2F%20Test


6

Прочитав все ответы на эту тему и ( неправильный ) принятый, я хочу добавить свой вклад.

ЕСЛИ целью является iOS7 +, и в 2017 году это должно произойти, так как XCode делает действительно трудным обеспечение совместимости под iOS8, лучший способ, многопоточный, быстрый, amd будет полностью поддерживать UTF-8, чтобы сделать это:

(Объективный код C)

@implementation NSString (NSString_urlencoding)

- (NSString *)urlencode {
    static NSMutableCharacterSet *chars = nil;
    static dispatch_once_t pred;

    if (chars)
        return [self stringByAddingPercentEncodingWithAllowedCharacters:chars];

    // to be thread safe
    dispatch_once(&pred, ^{
        chars = NSCharacterSet.URLQueryAllowedCharacterSet.mutableCopy;
        [chars removeCharactersInString:@"!*'();:@&=+$,/?%#[]"];
    });
    return [self stringByAddingPercentEncodingWithAllowedCharacters:chars];
}
@end

Это расширит NSString, исключит запрещенные символы RFC, поддержит символы UTF-8 и позволит вам использовать такие вещи, как:

NSString *myusername = "I'm[evil]&want(to)break!!!$->àéìòù";
NSLog(@"Source: %@ -> Dest: %@", myusername, [myusername urlencode]);

Это напечатает на вашей консоли отладки:

Источник: Я [зло] и хочу (чтобы) сломаться !!! $ -> аи -> Dest: I% 27m% 5Bevil% 5D% 26want% 28to% 29break% 21% 21% 21% 24-% 3E% C3 % A0% C3% A9% C3% AC% C3% B2% C3% В9

... обратите внимание также на использование dispatch_once, чтобы избежать множественных инициализаций в многопоточных средах.


5

Вот готовый к работе гибкий подход в Swift 5.x :

public extension CharacterSet {

    static let urlQueryParameterAllowed = CharacterSet.urlQueryAllowed.subtracting(CharacterSet(charactersIn: "&?"))

    static let urlQueryDenied           = CharacterSet.urlQueryAllowed.inverted()
    static let urlQueryKeyValueDenied   = CharacterSet.urlQueryParameterAllowed.inverted()
    static let urlPathDenied            = CharacterSet.urlPathAllowed.inverted()
    static let urlFragmentDenied        = CharacterSet.urlFragmentAllowed.inverted()
    static let urlHostDenied            = CharacterSet.urlHostAllowed.inverted()

    static let urlDenied                = CharacterSet.urlQueryDenied
        .union(.urlQueryKeyValueDenied)
        .union(.urlPathDenied)
        .union(.urlFragmentDenied)
        .union(.urlHostDenied)


    func inverted() -> CharacterSet {
        var copy = self
        copy.invert()
        return copy
    }
}



public extension String {
    func urlEncoded(denying deniedCharacters: CharacterSet = .urlDenied) -> String? {
        return addingPercentEncoding(withAllowedCharacters: deniedCharacters.inverted())
    }
}

Пример использования:

print("Hello, World!".urlEncoded()!)
print("You&Me?".urlEncoded()!)
print("#Blessed 100%".urlEncoded()!)
print("Pride and Prejudice".urlEncoded(denying: .uppercaseLetters)!)

Вывод:

Hello,%20World!
You%26Me%3F
%23Blessed%20100%25
%50ride and %50rejudice

Пожалуйста, включите комментарий, объясняющий причину ваших отрицательных голосов, чтобы я знал, как написать лучший ответ в будущем 🙂
Бен Легжеро

4

Используйте NSURLComponents для кодирования параметров HTTP GET:

    var urlComponents = NSURLComponents(string: "https://www.google.de/maps/")!
    urlComponents.queryItems = [
        NSURLQueryItem(name: "q", value: String(51.500833)+","+String(-0.141944)),
        NSURLQueryItem(name: "z", value: String(6))
    ]
    urlComponents.URL     // returns https://www.google.de/maps/?q=51.500833,-0.141944&z=6

http://www.ralfebert.de/snippets/ios/encoding-nsurl-get-parameters/


1
Он не кодирует амперсанды, косые черты или точки с запятой.
Фабио Оливейра

4

Этот код помог мне для кодирования специальных символов

NSString* encPassword = [password stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet alphanumericCharacterSet]];

3

быстрый код, основанный на ответе chown 's objc в этой теме.

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

4
эта функция устарела в iOS 9 :(
Оливер Аткинсон

3

В Swift 3 , пожалуйста, попробуйте ниже:

let stringURL = "YOUR URL TO BE ENCODE";
let encodedURLString = stringURL.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed)
print(encodedURLString)

Так как stringByAddingPercentEscapesUsingEncodingкодирует не URL-символы, но оставляет зарезервированные символы (например !*'();:@&=+$,/?%#[]), вы можете кодировать URL-адрес, как показано ниже:

let stringURL = "YOUR URL TO BE ENCODE";
let characterSetTobeAllowed = (CharacterSet(charactersIn: "!*'();:@&=+$,/?%#[] ").inverted)
if let encodedURLString = stringURL.addingPercentEncoding(withAllowedCharacters: characterSetTobeAllowed) {
    print(encodedURLString)
}

2

В быстром 3:

// exclude alpha and numeric == "full" encoding
stringUrl = stringUrl.addingPercentEncoding(withAllowedCharacters: .alphanumerics)!;

// exclude hostname and symbols &,/ and etc
stringUrl = stringUrl.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed)!;

2

Совет Apple в примечаниях к выпуску 10.11:

Если вам нужно процентное кодирование всей строки URL, вы можете использовать этот код для кодирования строки NSString, предназначенной для URL (в urlStringToEncode):

NSString *percentEncodedURLString =
  [[NSURL URLWithDataRepresentation:[urlStringToEncode dataUsingEncoding:NSUTF8StringEncoding] relativeToURL:nil] relativeString];

кажется, не работает, символы как & все еще остаются одни
kjyv

1

В моем случае, когда последний компонент был арабскими буквами, я сделал следующее Swift 2.2:

extension String {

 func encodeUTF8() -> String? {

    //If I can create an NSURL out of the string nothing is wrong with it
    if let _ = NSURL(string: self) {

        return self
    }

    //Get the last component from the string this will return subSequence
    let optionalLastComponent = self.characters.split { $0 == "/" }.last


    if let lastComponent = optionalLastComponent {

        //Get the string from the sub sequence by mapping the characters to [String] then reduce the array to String
        let lastComponentAsString = lastComponent.map { String($0) }.reduce("", combine: +)


        //Get the range of the last component
        if let rangeOfLastComponent = self.rangeOfString(lastComponentAsString) {
            //Get the string without its last component
            let stringWithoutLastComponent = self.substringToIndex(rangeOfLastComponent.startIndex)


            //Encode the last component
            if let lastComponentEncoded = lastComponentAsString.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.alphanumericCharacterSet()) {


            //Finally append the original string (without its last component) to the encoded part (encoded last component)
            let encodedString = stringWithoutLastComponent + lastComponentEncoded

                //Return the string (original string/encoded string)
                return encodedString
            }
        }
    }

    return nil;
}
}

использование:

let stringURL = "http://xxx.dev.com/endpoint/nonLatinCharacters"

if let encodedStringURL = stringURL.encodeUTF8() {

    if let url = NSURL(string: encodedStringURL) {

      ...
    }

} 

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

в соответствии со следующим блогом


Но это не рекомендуется в iOS 9. Какой обновленный код для этого?
Суджит Начан

1

Так много ответов, но у меня не получилось, поэтому я попробовал следующее:

fun simpleServiceCall(for serviceUrl: String, appendToUrl: String) { 
    let urlString: String = serviceUrl + appendToUrl.addingPercentEncoding(withAllowedCharacters: .urlPathAllowed)!     

    let finalUrl = URL(string: urlString)!

    //continue to execute your service call... 
}

Надеюсь, это кому-нибудь поможет. Спасибо


0

Для отдельных параметров запроса в кодировке www я создал категорию на NSString:

- (NSString*)WWWFormEncoded{
     NSMutableCharacterSet *chars = NSCharacterSet.alphanumericCharacterSet.mutableCopy;
     [chars addCharactersInString:@" "];
     NSString* encodedString = [self stringByAddingPercentEncodingWithAllowedCharacters:chars];
     encodedString = [encodedString stringByReplacingOccurrencesOfString:@" " withString:@"+"];
     return encodedString;
}

0

// Это без теста

NSMutableCharacterSet* set = [[NSCharacterSet alphanumericCharacterSet] mutableCopy];
[set addCharactersInString:@"-_.~"];
NSString *encode = [test stringByAddingPercentEncodingWithAllowedCharacters:set];

0

Я столкнулся с аналогичной проблемой передачи сложных строк в качестве параметра POST. Мои строки могут содержать азиатские символы, пробелы, кавычки и всевозможные специальные символы. Решение, которое я в итоге нашел, состояло в том, чтобы преобразовать мою строку в совпадающую последовательность юникодов, например «Hu0040Hu0020Hu03f5 ....», используя [NSString stringWithFormat: @ "Hu% 04x", [string characterAtIndex: i]], чтобы получать Unicode из каждого символ в исходной строке. То же самое можно сделать в Java.

Эта строка может быть безопасно передана в качестве параметра POST.

На стороне сервера (PHP) я меняю все буквы "H" на "\" и передаю полученную строку в json_decode. Последний шаг - экранирование одинарных кавычек перед сохранением строки в MySQL.

Таким образом, я могу сохранить любую строку UTF8 на моем сервере.


0

Этот работает на меня.

func stringByAddingPercentEncodingForFormData(plusForSpace: Bool=false) -> String? {
    let unreserved = "*-._"
    let allowed = NSMutableCharacterSet.alphanumericCharacterSet()
    allowed.addCharactersInString(unreserved)

    if plusForSpace {
        allowed.addCharactersInString(" ")
    }

    var encoded = stringByAddingPercentEncodingWithAllowedCharacters(allowed)
    if plusForSpace {
        encoded = encoded?.stringByReplacingOccurrencesOfString(" ",
                                                                withString: "+")
    }
    return encoded
}

Я нашел вышеуказанную функцию по этой ссылке: http://useyourloaf.com/blog/how-to-percent-encode-a-url-string/

Вы также можете использовать эту функцию с быстрым расширением. Пожалуйста, дайте мне знать, если есть какие-либо проблемы.

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.