Як знайти всі ідентифікатори відео YouTube у рядку за допомогою регулярного виразу?


91

У мене є текстове поле, де користувачі можуть писати що завгодно.

Наприклад:

Lorem Ipsum - це просто фіктивний текст. http://www.youtube.com/watch?v=DUQi_R4SgWo друкарської та набірної індустрії. Lorem Ipsum є стандартним фіктивним текстом у галузі ще з 1500-х років, коли невідомий друкар взяв камбуз типу та скрембував його, щоб зробити книгу зразків типу. Він пережив не лише п’ять століть, але й стрибок в електронну версію, залишившись по суті незмінним. http://www.youtube.com/watch?v=A_6gNZCkajU&feature=relmfu Він був популяризований у 1960-х роках з випуском аркушів Letraset, що містять уривки Lorem Ipsum, і нещодавно з використанням програмного забезпечення для настільних видань, таких як Aldus PageMaker, включаючи версії Lorem Ipsum.

Тепер я хотів би проаналізувати його та знайти всі URL-адреси відео YouTube та їх ідентифікатори.

Будь-яка ідея, як це працює?


Відповіді:


289

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".

Редагувати 05.07.2011: Додано -дефіс до класу ID char

Редагувати 17.07.2011: Виправлено регулярний вираз для споживання будь-якої залишкової частини (наприклад, запиту ) URL-адреси за ідентифікатором YouTube. Додано модифікатор 'i' ігнорування регістру . Перейменовано функцію на camelCase. Покращений попередньо зв’язаний тест lookahead.

Редагувати 27.07.2011: Додано нові формати URL-адрес YouTube "user" та "ytscreeningroom".

Редагувати 02.08.2011: спрощений / узагальнений для обробки нових URL-адрес YouTube "що-небудь / щось / йде".

Редагувати 25.08.2011: Кілька модифікацій:

  • Додано Javascript-версію linkifyYouTubeURLs()функції :.
  • У попередній версії частина схеми (протокол HTTP) була необов’язковою і, таким чином, відповідала б недійсним URL-адресам. Зробив необхідну частину схеми.
  • Попередня версія використовувала \bприв'язку межі слова навколо VIDEO_ID. Однак це не спрацює, якщо VIDEO_ID починається або закінчується -тире. Виправлено так, що він справляється з цією умовою.
  • Змінено вираз VIDEO_ID таким чином, що він повинен мати рівно 11 символів.
  • Попередня версія не змогла виключити попередньо зв’язані URL-адреси, якщо вони мали рядок запиту, що слідує за VIDEO_ID. Покращено негативне твердження про пошук, щоб це виправити.
  • Додано +та %до рядка запиту, що відповідає класу символів.
  • Змінений PHP версія регулярних виразів Обмежувач з: %до: ~.
  • Додано розділ "Нотатки" з деякими зручними нотатками.

Редагувати 12.10.2011: Частина хосту URL-адреси YouTube тепер може мати будь-який субдомен (не тільки www.).

Редагувати 01.05.2012: Розділ URL-адреси споживання тепер може дозволити '-'.

Редагувати 23.08.2013: Додано додатковий формат, наданий @Mei. (Частина запиту може мати .крапку.

Редагування 2013-11-30: долучення додатковий формат забезпечується @CRONUS: youtube-nocookie.com.

Редагувати 25.01.2016: Виправлено регулярний вираз для обробки випадків помилок, наданих CRONUS.


2
Я не бачив специфікації, хоча я її і шукав. Я щойно помітив тире в деяких посиланнях в нетрі Інтернету. Наприклад: youtube.com/watch?v=CLPk-6_xgiY
cottonBallPaws

1
@littleFluffyKitty: Дякую за увагу. Оновили відповідь, включивши дефіс як дійсний символ посвідчення особи.
ridgerunner

1
@ridgerunner: Якщо ви не впевнені в редагуванні, можете відмовитись. Крім того, як і у вікіпедії, вся історія зберігається з вашими кредитами. Я бачив, як ти з часом справді придумав відповідь, тож було б безглуздо загубити тебе тут.
hakre

1
Ось той, який не спрацював: youtube.com/watch?v=E1IPnnttL9k&feature=youtu.be
andrebola

1
Це чудово працює, але не працює з цим (новим?) Параметром рядка запитів: feature = youtu.be. Зміна [? = & +% \ W -] * на [? = & +% \ W - \.] * У вашому рядку "споживає залишок url" робить трюк. Дякую!
Mei Gwilym

10

Ось метод, який я колись писав для проекту, який витягує відео-ключі 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;
}
  1. Знайдіть регулярний вираз, який витягне всі посилання з тексту. Google вам там допоможе.
  2. Зациклюйте всі посилання та зателефонуйте getHostInfo () для кожного

1
щиро дякую! невеликий мод if(strpos($vid_link, 'youtu'))захоплює коротку URL-адресу youtu.beна додаток до загальної.
Chamilyan

Ласкаво просимо. дякую за оновлення, я відредагував зміну. на додаток, регулярний вираз ridgerunner здається справжньою справою, і я рекомендую використовувати його над моєю простою штукою. ура
Christof

саме те, що я шукав. місце на партнері! +1
blackpla9ue

8

Хоча відповідь 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:

http://jsfiddle.net/DJSwc/5/


2

Спробуйте

[^\s]*youtube\.com[^\s]*?v=([-\w]+)[^\s]*

Ви знайдете ідентифікатори відео в першій групі зйомки. Я не знаю, що таке дійсний ідентифікатор відео? На даний момент я перевіряю v=і фіксую все -A-Za-z0-9_.

Я перевірив це в Інтернеті тут на rubular за допомогою вашого зразка рядка.


2

Використання:

<?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
?>

2

Гаразд, я зробив власну функцію. Але я вважаю, що це досить неефективно. Будь-які вдосконалення вітаються:

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;
}

Якщо ви шукаєте лише посилання з youtube.com, чому ви хочете спочатку створити список із усіма посиланнями? І я думаю, що не потрібно використовувати 3 різні регулярні вирази.
stema


1

Оригінальний плакат запитував: "Я хотів би проаналізувати його та знайти всі 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;
}

0

Легко знайти посилання на 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)

Це не тільки для YoutTube, воно буде відповідати іншим URL-адресам із вмісту.
Рахіл Вазір,

0
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.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.