iPhone - Reg Exp для валидности URL
У меня есть вид чата, где пользователи могут отправлять URL друг другу. В случае URL, я хочу позволить пользователю нажать на ссылку и открыть веб-просмотр.
Я использую IFTweetLabel, который использует RegexKitLite. В настоящее время доступна только поддержка, если URL начинается с http / https. Я хочу поддерживать ссылки без http, например: www.nytimes.com, и даже без "www", nytimes.com. (и куча других расширений).
Это префикс http/s reg exp:
@"([hH][tT][tT][pP][sS]?:\\/\\/[^ ,'\">\\]\\)]*[^\\. ,'\">\\]\\)])
Может кто-нибудь сказать мне другие регулярные выражения, которые мне нужны, чтобы ответить на другие мои требования.
Я пытался использовать этот, но добавление его в объективный код c порождает много проблем.
Спасибо
3 ответа
Это будет соответствовать как http://example.org/ и www.example.org.
@"(([hH][tT][tT][pP][sS]?:\\/\\/|www\\.)[^ ,'\">\\]\\)]*\\.[^\\. ,'\">\\]\\)]{2,6})
Хотя я добавил "группу совпадений", проверьте результат поиска / поиска, возвращаемый RegExp, чтобы правильные параметры были вставлены в нужное место.
Если бы вы могли опубликовать весь фрагмент кода, это было бы проще.
Объяснение RegExp:
(
(
[hH][tT][tT][pP][sS]?:\/\/ # Match HTTP/http (and hTtP :)
| # OR
www\. # www<literal DOT>
)
[^ ,'\">\]\)]* # Match at least 1 character that are not any of space, comma, apostrophe, quotation mark, "more than", "right square bracket", "right parenthese"
\. # Match <literal DOT>
[^\. ,'\">\]\)]{2,6} # Match 2-6 characters that are not any of dot, space, comma, apostrophe, quotation mark, "more than", "right square bracket", "right parenthese"
)
Ниже приведено регулярное выражение соответствия URL Джона Граберса:
(?i)\b(?:[a-z][\w-]+:(?:/{1,3}|[a-z0-9%])|www\d{0,3}[.]|[a-z0-9.\-]+[.][a-z]{2,4}/)(?:[^\s()<>]+|\(([^\s()<>]+|(\([^\s()<>]+\)))*\))+(?:\(([^\s()<>]+|(\([^\s()<>]+\)))*\)|[^\s`!()\[\]{};:'".,<>?«»“”‘’])
Ниже приведено регулярное выражение, которое я придумал, смешав несколько других регулярных выражений, которые у меня были, и хороший кусок регулярного выражения Grubers:
(?i)\b(?:(?:[a-z][\w\-]+://(?:\S+?(?::\S+?)?\@)?)|(?:(?:[a-z0-9\-]+\.)+[a-z]{2,4}))(?:[^\s()<>]+|\((?:[^\s()<>]+|(?:\([^\s()<>]*\)))*\))*(?<![\s`!()\[\]{};:'".,<>?«»“”‘’])
Ниже приведен пример программы, которая через RegexKitLite демонстрирует, что каждое регулярное выражение сопоставляется с примером текста:
Вы видели http://www.stackru.com/? Или http://www.stackru.com/?
И затем есть www.stackru.com/, наряду с www.stackru.com/index.
Может быть, что-то вроде stackru.com с дополнительным stackru.com? Или "stackru.com"?
Возможно jobs.stackru.com, или "http://twitter.com/#!/CHOCKENBERRY", ЧАСЫ!!
Файл @file:///Users/johne/rkl/rkl.html#RegexKitLiteCookbook?
Может быть, http://www.yahoo.com/index///i.html! http://www.yahoo.com/////xyz.html?!
Код:
#import <Foundation/Foundation.h>
#import "RegexKitLite.h"
int main(int argc, char *argv[]) {
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
NSString *urlRegex = @"(?i)\\b(?:(?:[a-z][\\w\\-]+://(?:\\S+?(?::\\S+?)?\\@)?)|(?:(?:[a-z0-9\\-]+\\.)+[a-z]{2,4}))(?:[^\\s()<>]+|\\((?:[^\\s()<>]+|(?:\\([^\\s()<>]*\\)))*\\))*(?<![\\s`!()\\[\\]{};:'\".,<>?«»“”‘’])";
// John Gruber's URL matching regex from http://daringfireball.net/2010/07/improved_regex_for_matching_urls
NSString *gruberURLRegex = @"(?i)\\b(?:[a-z][\\w-]+:(?:/{1,3}|[a-z0-9%])|www\\d{0,3}[.]|[a-z0-9.\\-]+[.][a-z]{2,4}/)(?:[^\\s()<>]+|\\(([^\\s()<>]+|(\\([^\\s()<>]+\\)))*\\))+(?:\\(([^\\s()<>]+|(\\([^\\s()<>]+\\)))*\\)|[^\\s`!()\\[\\]{};:'\".,<>?«»“”‘’])";
NSString *urlString = @"Did you see http://www.stackru.com? Or http://www.stackru.com/?\n\nAnd then there is www.stackru.com/, along with www.stackru.com/index.\n\nMaybe something like stackru.com with extra stackru.com? Or \"stackru.com\"?\n\nPerhaps jobs.stackru.com, or 'http://twitter.com/#!/CHOCKENBERRY', the CHOCKLOCK!!\n\nFile @file:///Users/johne/rkl/rkl.html#RegexKitLiteCookbook?\n\nMaybe http://www.yahoo.com/index///i.html! http://www.yahoo.com/////xyz.html?!";
NSLog(@"String :\n\n%@\n\n", urlString);
NSLog(@"Matches: %@\n", [urlString componentsMatchedByRegex:urlRegex]);
NSLog(@"Gruber URL Regex Matches: %@\n", [urlString componentsMatchedByRegex:gruberURLRegex]);
[pool release]; pool = NULL;
return(0);
}
Компилировать с:
shell% gcc -o url url.m RegexKitLite.m -framework Foundation -licucore
Когда запустить:
shell% ./url
2011-05-27 20:32:58.204 url[25520:903] String :
Did you see http://www.stackru.com? Or http://www.stackru.com/?
And then there is www.stackru.com/, along with www.stackru.com/index.
Maybe something like stackru.com with extra stackru.com? Or "stackru.com"?
Perhaps jobs.stackru.com, or 'http://twitter.com/#!/CHOCKENBERRY', the CHOCKLOCK!!
File @file:///Users/johne/rkl/rkl.html#RegexKitLiteCookbook?
Maybe http://www.yahoo.com/index///i.html! http://www.yahoo.com/////xyz.html?!
2011-05-27 20:32:58.211 url[25520:903] Matches: (
"http://www.stackru.com",
"http://www.stackru.com/",
"www.stackru.com/",
"www.stackru.com/index",
"stackru.com",
"stackru.com",
"stackru.com",
"jobs.stackru.com",
"http://twitter.com/#!/CHOCKENBERRY",
"file:///Users/johne/rkl/rkl.html#RegexKitLiteCookbook",
"http://www.yahoo.com/index///i.html",
"http://www.yahoo.com/////xyz.html"
)
2011-05-27 20:32:58.213 url[25520:903] Gruber URL Regex Matches: (
"http://www.stackru.com",
"http://www.stackru.com/",
"www.stackru.com/",
"www.stackru.com/index",
"http://twitter.com/#!/CHOCKENBERRY",
"file:///Users/johne/rkl/rkl.html#RegexKitLiteCookbook",
"http://www.yahoo.com/index///i.html",
"http://www.yahoo.com/////xyz.html"
)
РЕДАКТИРОВАТЬ 2011/05/27: Внесены незначительные изменения в регулярное выражение, чтобы исправить проблему, если она не соответствует (
)
в скобках правильно.
РЕДАКТИРОВАТЬ 2011/05/27: Найдены некоторые дополнительные угловые случаи, с которыми вышеприведенное регулярное выражение не справилось хорошо Обновленное регулярное выражение:
(?i)\b(?:[a-z][\w\-]+://(?:\S+?(?::\S+?)?\@)?)?(?:(?:(?<!:/|\.)(?:(?:[a-z0-9\-]+\.)+[a-z]{2,4}(?![a-z]))|(?<=://)/))(?:(?:[^\s()<>]+|\((?:[^\s()<>]+|(?:\([^\s()<>]*\)))*\))*)(?<![\s`!()\[\]{};:'".,<>?«»“”‘’])
... как строка Obj-C:
@"(?i)\\b(?:[a-z][\\w\\-]+://(?:\\S+?(?::\\S+?)?\\@)?)?(?:(?:(?<!:/|\\.)(?:(?:[a-z0-9\\-]+\\.)+[a-z]{2,4}(?![a-z]))|(?<=://)/))(?:(?:[^\\s()<>]+|\\((?:[^\\s()<>]+|(?:\\([^\\s()<>]*\\)))*\\))*)(?<![\\s`!()\\[\\]{};:'\".,<>?«»“”‘’])";
ОП также спросил, как сделать так, чтобы конечный TLD был "действительным". Вот то же регулярное выражение в строковой форме Obj-C со всеми действующими на данный момент TLD (по состоянию на 2011/05/27):
@"(?i)\\b(?:[a-z][\\w\\-]+://(?:\\S+?(?::\\S+?)?\\@)?)?(?:(?:(?<!:/|\\.)(?:(?:[a-z0-9\\-]+\\.)+(?:(ac|ad|ae|aero|af|ag|ai|al|am|an|ao|aq|ar|arpa|as|asia|at|au|aw|ax|az|ba|bb|bd|be|bf|bg|bh|bi|biz|bj|bm|bn|bo|br|bs|bt|bv|bw|by|bz|ca|cat|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|co|com|coop|cr|cu|cv|cx|cy|cz|de|dj|dk|dm|do|dz|ec|edu|ee|eg|er|es|et|eu|fi|fj|fk|fm|fo|fr|ga|gb|gd|ge|gf|gg|gh|gi|gl|gm|gn|gov|gp|gq|gr|gs|gt|gu|gw|gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|in|info|int|io|iq|ir|is|it|je|jm|jo|jobs|jp|ke|kg|kh|ki|km|kn|kp|kr|kw|ky|kz|la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|me|mg|mh|mil|mk|ml|mm|mn|mo|mobi|mp|mq|mr|ms|mt|mu|museum|mv|mw|mx|my|mz|na|name|nc|ne|net|nf|ng|ni|nl|no|np|nr|nu|nz|om|org|pa|pe|pf|pg|ph|pk|pl|pm|pn|pr|pro|ps|pt|pw|py|qa|re|ro|rs|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sj|sk|sl|sm|sn|so|sr|st|su|sv|sy|sz|tc|td|tel|tf|tg|th|tj|tk|tl|tm|tn|to|tp|tr|travel|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|xn--0zwm56d|xn--11b5bs3a9aj6g|xn--3e0b707e|xn--45brj9c|xn--80akhbyknj4f|xn--90a3ac|xn--9t4b11yi5a|xn--clchc0ea0b2g2a9gcd|xn--deba0ad|xn--fiqs8s|xn--fiqz9s|xn--fpcrj9c3d|xn--fzc2c9e2c|xn--g6w251d|xn--gecrj9c|xn--h2brj9c|xn--hgbk6aj7f53bba|xn--hlcj6aya9esc7a|xn--j6w193g|xn--jxalpdlp|xn--kgbechtv|xn--kprw13d|xn--kpry57d|xn--lgbbat1ad8j|xn--mgbaam7a8h|xn--mgbayh7gpa|xn--mgbbh1a71e|xn--mgbc0a9azcg|xn--mgberp4a5d4ar|xn--o3cw4h|xn--ogbpf8fl|xn--p1ai|xn--pgbs0dh|xn--s9brj9c|xn--wgbh1c|xn--wgbl6a|xn--xkc2al3hye2a|xn--xkc2dl3a5ee0h|xn--yfro4i67o|xn--ygbi2ammx|xn--zckzah|xxx|ye|yt|za|zm|zw))(?![a-z]))|(?<=://)/))(?:(?:[^\\s()<>]+|\\((?:[^\\s()<>]+|(?:\\([^\\s()<>]*\\)))*\\))*)(?<![\\s`!()\\[\\]{};:'\".,<>?«»“”‘’])";
Вы не хотите использовать регулярное выражение для этого.
Вы хотите NSDataDetector
и он найдет их всех для вас.