Як я кодую 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 );

Деякі хороші читання з цього приводу:

Об'єктивний-процентний відсоток iPhone кодує рядок?
Об'єктивне кодування C та Swift URL

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 Можливо, це я і отримав. Це було стандартною категорією у всіх моїх проектах протягом року або близько того, тому я думаю, що ви, можливо, були першоджерелом! Я відредагував формулювання, тому, схоже, я намагаюся не брати до уваги його написання).
чауна

4
Чому? if (thisChar == '') {[вихід 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.
Бен Лачман

'&' не є дійсним символом для параметра рядка рядка запиту. Спробуйте скористатися "& amp;" замість цього.
Нішант

1
stringByAddingPercentEscapesUsingEncoding ВИЗНАЧЕНО "Використовувати -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, що випускаються.
Peter DeWeese

"&", звичайно, дозволено в рядку запиту, тому я відредагую свою відповідь, використовуючи посилання Річарда.
Peter DeWeese

3
NSMakeRange('&', 1)не працює в Swift, тому що Swift не дозволяє char вступити в конверсію без злому. Щоб використовувати це рішення в коді 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 ("&")) "повертає нуль, як слід. Замість символів.removeCharactersInString ("&").
ambientlight

14

оновлення ios 7

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

NSString *decode = [encode stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];

Це неточно. Ви захочете по-різному уникати різних частин URL-адреси. stackoverflow.com/questions/3423545/…
Стів Мозер

Це насправді залежить від вашого сервера для дозволених символів, тут на моєму прикладі на сервері я використовував лише дозволений буквено-цифровий набір символів, ви можете змінити його на конкретні символи, щоб задовольнити ваші потреби.
Андерґог

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]він охопить усі випадки

Мета C

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]);

Це буде надруковано на консолі налагодження:

Джерело: Я [злий] і хочу (до) зламати !!! $ -> àéìòù -> Призначення: I% 27м% 5Бевіль% 5D% 26бала% 28 до% 29пар% 21% 21% 21% 24-% 3E% C3 % A0% C3% A9% C3% AC% C3% B2% C3% B9

... зверніть увагу також на використання 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

Будь ласка, додайте коментар, який пояснює причину ваших подій, тому я знаю, як написати кращу відповідь у майбутньому 🙂
Ben Leggiero

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

swift-код, заснований на відповіді obnc від chown в цій темі.

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

Стільки відповідей, але не спрацювали для мене, тому я спробував наступне:

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.