Как кодировать строку URL
У меня есть строка URL (NSString
) с пробелами и &
символами. Как я могу кодировать всю строку (включая символ и пробелы амперсанда &
)?
Ответы
Ответ 1
К сожалению, stringByAddingPercentEscapesUsingEncoding
не всегда работает на 100%. Он кодирует символы не URL-адреса, но оставляет зарезервированные символы (например, слэш /
и ampersand &
). По-видимому, это ошибка, о которой Apple знает, но поскольку они еще не исправили ее, я использовал эту категорию для кодирования строки:
@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 для кодирования строки?
Objective-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/
Ответ 2
Это может быть полезно
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.
Ответ 3
Добавлены новые API-интерфейсы после выбора ответа; Теперь вы можете использовать NSURLUtilities. Поскольку разные части URL-адресов допускают разные символы, используйте соответствующий набор символов. Следующий пример кодируется для включения в строку запроса:
encodedString = [myString stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLQueryAllowedCharacterSet];
Чтобы специально преобразовать '&', вам нужно удалить его из набора запросов url или использовать другой набор, например '&' разрешено в запросе URL:
NSMutableCharacterSet *chars = NSCharacterSet.URLQueryAllowedCharacterSet.mutableCopy;
[chars removeCharactersInRange:NSMakeRange('&', 1)]; // %26
encodedString = [myString stringByAddingPercentEncodingWithAllowedCharacters:chars];
Ответ 4
Пример 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
}
}
Ответ 5
Обновление ios 7
NSString *encode = [string stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];
NSString *decode = [encode stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
Ответ 6
Я решил использовать вызов 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));
Ответ 7
Попробуйте использовать метод 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
Ответ 8
После прочтения всех ответов на этот вопрос и принятого (неправильного), я хочу добавить свой вклад.
IF, целью является iOS7 +, а в 2017 году это должно произойти, поскольку XCode действительно сложно обеспечить совместимость в iOS8, лучший способ, потокобезопасный, быстрый, amd, будет полностью поддерживать UTF-8, чтобы сделать это:
(Objective C code)
@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, чтобы избежать множественных инициализаций в многопоточных средах.
Ответ 9
Вот готовый к работе гибкий подход в 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
Ответ 10
Используйте 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/
Ответ 11
Этот код помог мне для кодирования специальных символов
NSString* encPassword = [password stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet alphanumericCharacterSet]];
Ответ 12
быстрый код на основе chown objc ответ в этом потоке.
extension String {
func urlEncode() -> String {
return CFURLCreateStringByAddingPercentEscapes(
nil,
self,
nil,
"!*'();:@&=+$,/?%#[]",
CFStringBuiltInEncodings.UTF8.rawValue
)
}
}
Ответ 13
В 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)
}
Ответ 14
В быстром 3:
// exclude alpha and numeric == "full" encoding
stringUrl = stringUrl.addingPercentEncoding(withAllowedCharacters: .alphanumerics)!;
// exclude hostname and symbols &,/ and etc
stringUrl = stringUrl.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed)!;
Ответ 15
Рекомендации Apple в примечаниях к выпуску 10.11:
Если вам нужно процитировать всю строку URL-кода, вы можете использовать этот код для кодирования NSString, предназначенного для URL-адреса (в urlStringToEncode):
NSString *percentEncodedURLString =
[[NSURL URLWithDataRepresentation:[urlStringToEncode dataUsingEncoding:NSUTF8StringEncoding] relativeToURL:nil] relativeString];
Ответ 16
-(NSString *)encodeUrlString:(NSString *)string {
return CFBridgingRelease(
CFURLCreateStringByAddingPercentEscapes(
kCFAllocatorDefault,
(__bridge CFStringRef)string,
NULL,
CFSTR("!*'();:@&=+$,/?%#[]"),
kCFStringEncodingUTF8)
);
}
в соответствии со следующим блогом
Ответ 17
В моем случае, когда последний компонент был арабскими буквами, я сделал следующее в 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) {
...
}
}
Ответ 18
Для отдельных параметров запроса в форме страницы я я сделал категорию в NSString:
- (NSString*)WWWFormEncoded{
NSMutableCharacterSet *chars = NSCharacterSet.alphanumericCharacterSet.mutableCopy;
[chars addCharactersInString:@" "];
NSString* encodedString = [self stringByAddingPercentEncodingWithAllowedCharacters:chars];
encodedString = [encodedString stringByReplacingOccurrencesOfString:@" " withString:@"+"];
return encodedString;
}
Ответ 19
//Это без теста
NSMutableCharacterSet* set = [[NSCharacterSet alphanumericCharacterSet] mutableCopy];
[set addCharactersInString:@"-_.~"];
NSString *encode = [test stringByAddingPercentEncodingWithAllowedCharacters:set];
Ответ 20
Я столкнулся с аналогичной проблемой, передавая сложные строки как параметр POST. Мои строки могут содержать азиатские символы, пробелы, кавычки и всевозможные специальные символы. Решение, которое я в итоге нашел, состояло в том, чтобы преобразовать мою строку в соответствующую серию юникодов, например. "Hu0040Hu0020Hu03f5....", используя [NSString stringWithFormat: @ "Hu% 04x", [string characterAtIndex: i]], чтобы получить Unicode от каждого символа в исходной строке. То же самое можно сделать на Java.
Эта строка может быть безопасно передана как параметр POST.
На стороне сервера (PHP) я меняю все "H" на "\", и передаю результирующую строку json_decode. Заключительный шаг - избежать одиночных кавычек перед сохранением строки в MySQL.
Таким образом, я могу сохранить любую строку UTF8 на моем сервере.
Ответ 21
Этот работает для меня.
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/
Вы также можете использовать эту функцию с быстрым расширением. Пожалуйста, дайте мне знать, если есть какие-либо проблемы.