Как найти все идентификаторы видео YouTube в строке с помощью регулярных выражений?
У меня есть текстовое поле, где пользователи могут писать что угодно.
Например:
Lorem Ipsum - просто фиктивный текст. http://www.youtube.com/watch?v=DUQi_R4SgWo из индустрии печати и набора текста. Лорем Ипсум был стандартным манекенщиком в отрасли с 1500-х годов, когда неизвестный принтер взял набор шрифтов и скремблировал его, чтобы сделать книгу типовых образцов. Он пережил не только пять веков, но и скачок в электронном наборе текста, оставаясь практически неизменным. http://www.youtube.com/watch?v=A_6gNZCkajU&feature=relmfu Он был популяризирован в 1960-х годах с выпуском листов Letraset, содержащих отрывки из Lorem Ipsum, и совсем недавно с программным обеспечением для настольной публикации, таким как Aldus PageMaker, включая версии Lorem Ipsum.
Теперь я хотел бы проанализировать его и найти все URL-адреса видео YouTube и их идентификаторы.
Есть идеи, как это работает?
10 ответов
URL видео YouTube может встречаться в различных форматах:
- последний короткий формат:
http://youtu.be/NLqAF9hrVbY
- IFrame:
http://www.youtube.com/embed/NLqAF9hrVbY
- iframe (безопасный):
https://www.youtube.com/embed/NLqAF9hrVbY
- параметр объекта:
http://www.youtube.com/v/NLqAF9hrVbY?fs=1&hl=en_US
- вставка объекта:
http://www.youtube.com/v/NLqAF9hrVbY?fs=1&hl=en_US
- часы:
http://www.youtube.com/watch?v=NLqAF9hrVbY
- пользователей:
http://www.youtube.com/user/Scobleizer#p/u/1/1p3vcRhsYGo
- ytscreeningroom:
http://www.youtube.com/ytscreeningroom?v=NRHVzbJVx8I
- любая / вещь / идет!:
http://www.youtube.com/sandalsResorts#p/c/54B8C800269D7C1B/2/PPS-8DMrAn4
- любой / поддомен / тоже:
http://gdata.youtube.com/feeds/api/videos/NLqAF9hrVbY
- больше параметров:
http://www.youtube.com/watch?v=spDj54kf-vY&feature=g-vrec
- запрос может иметь точку:
http://www.youtube.com/watch?v=spDj54kf-vY&feature=youtu.be
- Домен nocookie:
http://www.youtube-nocookie.com
Вот функция PHP с закомментированным регулярным выражением, которое соответствует каждой из этих форм URL и преобразует их в ссылки (если они уже не являются ссылками):
// Linkify youtube URLs which are not already links.
function linkifyYouTubeURLs($text) {
$text = preg_replace('~(?#!js YouTubeId Rev:20160125_1800)
# Match non-linked youtube URL in the wild. (Rev:20130823)
https?:// # Required scheme. Either http or https.
(?:[0-9A-Z-]+\.)? # Optional subdomain.
(?: # Group host alternatives.
youtu\.be/ # Either youtu.be,
| youtube # or youtube.com or
(?:-nocookie)? # youtube-nocookie.com
\.com # followed by
\S*? # Allow anything up to VIDEO_ID,
[^\w\s-] # but char before ID is non-ID char.
) # End host alternatives.
([\w-]{11}) # $1: VIDEO_ID is exactly 11 chars.
(?=[^\w-]|$) # Assert next char is non-ID or EOS.
(?! # Assert URL is not pre-linked.
[?=&+%\w.-]* # Allow URL (query) remainder.
(?: # Group pre-linked alternatives.
[\'"][^<>]*> # Either inside a start tag,
| </a> # or inside <a> element text contents.
) # End recognized pre-linked alts.
) # End negative lookahead assertion.
[?=&+%\w.-]* # Consume any URL (query) remainder.
~ix', '<a href="http://www.youtube.com/watch?v=$1">YouTube link: $1</a>',
$text);
return $text;
}
; // Конец $YouTubeId.
И вот версия JavaScript с точно таким же регулярным выражением (с удаленными комментариями):
// Linkify youtube URLs which are not already links.
function linkifyYouTubeURLs(text) {
var re = /https?:\/\/(?:[0-9A-Z-]+\.)?(?:youtu\.be\/|youtube(?:-nocookie)?\.com\S*?[^\w\s-])([\w-]{11})(?=[^\w-]|$)(?![?=&+%\w.-]*(?:['"][^<>]*>|<\/a>))[?=&+%\w.-]*/ig;
return text.replace(re,
'<a href="http://www.youtube.com/watch?v=$1">YouTube link: $1</a>');
}
Заметки:
- Часть VIDEO_ID URL-адреса захвачена в одной-единственной группе захвата:
$1
, - Если вы знаете, что ваш текст не содержит предварительно связанных URL-адресов, вы можете безопасно удалить отрицательное предварительное утверждение, которое проверяет это условие (утверждение, начинающееся с комментария: "URL-адрес подтверждения предварительно не связан".) Это ускорит до регулярного выражения несколько.
- Заменяющая строка может быть изменена в соответствии с требованиями. Приведенный выше просто создает ссылку на общий
"http://www.youtube.com/watch?v=VIDEO_ID"
стиль URL и устанавливает текст ссылки на:"YouTube link: VIDEO_ID"
,
Изменить 2011-07-05: Добавлено -
дефис для идентификатора класса char
Изменить 2011-07-17: исправлено регулярное выражение для использования любой оставшейся части (например, запроса) URL-адреса после идентификатора YouTube. добавленной 'i'
модификатор игнорирования. Переименована функция в camelCase. Улучшен предварительный тест с предварительным связыванием.
Изменить 2011-07-27: добавлены новые форматы URL-адресов YouTube для пользователей и ytscreeningroom.
Изменить 2011-08-02: Упрощенный / обобщенный, чтобы обрабатывать новые "любые / вещь / идет" URL-адреса YouTube.
Редактировать 2011-08-25: несколько модификаций:
- Добавлена версия Javascript:
linkifyYouTubeURLs()
функция. - В предыдущей версии часть схемы (протокол HTTP) была необязательной и, следовательно, соответствовала бы недействительным URL-адресам. Сделана схема обязательной части.
- Предыдущая версия использовала
\b
якорь границы слова вокруг VIDEO_ID. Однако это не будет работать, если VIDEO_ID начинается или заканчивается-
тире. Исправлено так, что он обрабатывает это условие. - Изменено выражение VIDEO_ID, так что оно должно быть ровно 11 символов.
- Предыдущая версия не смогла исключить предварительно связанные URL, если у них была строка запроса, следующая за VIDEO_ID. Улучшено негативное предположение, чтобы исправить это.
- добавленной
+
а также%
к классу символов, соответствующему строке запроса. - Изменен разделитель регулярных выражений версии PHP с:
%
к:~
, - Добавлен раздел "Заметки" с некоторыми удобными заметками.
Изменить 2011-10-12: часть хоста URL YouTube теперь может иметь любой поддомен (не только www.
).
Изменить 2012-05-01: в разделе "Использовать URL" теперь можно указывать "-".
Редактировать 2013-08-23: Добавлен дополнительный формат, предоставляемый @Mei. (Часть запроса может иметь .
точка.
Изменить 2013-11-30: добавлен дополнительный формат, предоставляемый @CRONUS: youtube-nocookie.com
,
Редактировать 2016-01-25: исправлено регулярное выражение для обработки ошибки, предоставленной CRONUS.
Вот метод, который я однажды написал для проекта, который извлекает видео ключи YouTube и Vimeo:
/**
* strip important information out of any video link
*
* @param string link to a video on the hosters page
* @return mixed FALSE on failure, array on success
*/
function getHostInfo ($vid_link)
{
// YouTube get video id
if (strpos($vid_link, 'youtu'))
{
// Regular links
if (preg_match('/(?<=v\=)([\w\d-_]+)/', $vid_link, $matches))
return array('host_name' => 'youtube', 'original_key' => $matches[0]);
// Ajax hash tag links
else if (preg_match('§([\d\w-_]+)$§i', $vid_link, $matches))
return array('host_name' => 'youtube', 'original_key' => $matches[0]);
else
return FALSE;
}
// Vimeo get video id
elseif (strpos($vid_link, 'vimeo'))
{
if (preg_match('§(?<=/)([\d]+)§', $vid_link, $matches))
return array('host_name' => 'vimeo', 'original_key' => $matches[0]);
else
return FALSE;
}
else
return FALSE;
}
- Найти регулярное выражение, которое будет извлекать все ссылки из текста. Google поможет вам там.
- Зациклить все ссылки и вызвать getHostInfo() для каждого
В то время как ответ Ridgerunner является основой для моего ответа, он НЕ решает все URL, и я не верю, что он способен на это из-за множества возможных совпадений VIDEO_ID
в URL YouTube. Мое регулярное выражение включает в себя его агрессивный подход в качестве крайней меры, но сначала пытается все общие совпадения, что значительно снижает вероятность неправильного совпадения позже в URL.
Это регулярное выражение:
/https?:\/\/(?:[0-9A-Z-]+\.)?(?:youtu\.be\/|youtube\.com(?:\/embed\/|\/v\/|\/watch\?v=|\/ytscreeningroom\?v=|\/feeds\/api\/videos\/|\/user\S*[^\w\-\s]|\S*[^\w\-\s]))([\w\-]{11})[?=&+%\w-]*/ig;
Обрабатывает все случаи, которые изначально упоминались в примерах ridgerunners, плюс любой URL, который может иметь последовательность из 11 символов позже в URL. то есть:
http://www.youtube.com/watch?v=GUEZCxBcM78&feature=pyv&feature=pyv&ad=10059374899&kw=%2Bwingsuit
Вот рабочий пример, который тестирует все примеры URL-адресов YouTube:
На оригинальном плакате спрашивалось: "Я хотел бы проанализировать его и найти все URL-адреса видео YouTube и их идентификаторы". Я переключил самый популярный ответ выше на preg_match и вернул идентификатор видео и URL.
Получить URL и идентификатор YouTube из сообщения:
$match[0] = Full URL
$match[1] = video ID
function get_youtube_id($input) {
$input = preg_match('~https?://(?:[0-9A-Z-]+\.)?(?:youtu\.be/|youtube(?:-nocookie)?\.com\S*[^\w\s-])([\w-]{11})(?=[^\w-]|$)(?![?=&+%\w.-]*(?:[\'"][^<>]*>|</a>))[?=&+%\w.-]*~ix',
$input, $match);
return $match;
}
Пытаться
[^\s]*youtube\.com[^\s]*?v=([-\w]+)[^\s]*
Вы найдете идентификаторы видео в первой группе захвата. Что я не знаю, что такое действительный идентификатор видео? На данный момент я проверяю v=
и захватить все -A-Za-z0-9_
,
Я проверил это онлайн здесь на Rubular с вашим примером строки.
Хорошо, я сделал свою собственную функцию. Но я считаю, что это довольно неэффективно. Любые улучшения приветствуются:
function get_youtube_videos($string) {
$ids = array();
// Find all URLs
preg_match_all('/(http|https)\:\/\/[a-zA-Z0-9\-\.]+\.[a-zA-Z]{2,3}(\/\S*)?/', $string, $links);
foreach ($links[0] as $link) {
if (preg_match('~youtube\.com~', $link)) {
if (preg_match('/[^=]+=([^?]+)/', $link, $id)) {
$ids[] = $id[1];
}
}
}
return $ids;
}
Использование:
<?php
// The YouTube URL string
$youtube_url='http://www.youtube.com/watch?v=8VtUYvwktFQ';
// Use regex to get the video ID
$regex='#(?<=v=)[a-zA-Z0-9-]+(?=&)|(?<=[0-9]/)[^&\n]+|(?<=v=)[^&\n]+#';
preg_match($regex, $youtube_url, $id);
// Plug that into our HTML
?>
Я попробовал простое выражение, чтобы получить только видеоид:
[?&]v=([^&#]*)
Проверьте это работает онлайн здесь на phpliveregex.
Найти ссылку на YouTube легко из строки:
function my_url_search($se_action_data)
{
$regex = '/https?\:\/\/[^\" ]+/i';
preg_match_all($regex, $se_action_data, $matches);
$get_url=array_reverse($matches[0]);
return array_unique($get_url);
}
echo my_url_search($se_action_data)
String urlid="" ;
String url="http://www.youtube.com/watch?v=0zM4nApSvMg#t=0m10s";
Pattern pattern =Pattern.compile("(?:http|https|)(?::\\/\\/|)(?:www.|)(?:youtu\\.be\\/|youtube\\.com(?:\\/embed\\/|\\/v\\/|\\/watch\\?v=|\\/ytscreeningroom\\?v=|\\/feeds\\/api\\/videos\\/|\\/user\\\\S*[^\\w\\-\\s]|\\S*[^\\w\\-\\s]))([\\w\\-\\_]{11})[a-z0-9;:@#?&%=+\\/\\$_.-]*");
Matcher result = pattern.matcher(url);
if (result.find())
{
urlid=result.group(1);
}
Этот код в Java работает абсолютно хорошо для всех URL-адресов YouTube в настоящее время.