如何使用正则表达式在字符串中查找所有 YouTube 视频 ID?

2022-08-30 07:47:04

我有一个文本字段,用户可以在其中编写任何内容。

例如:

Lorem Ipsum只是虚拟文本。印刷和排版行业的 http://www.youtube.com/watch?v=DUQi_R4SgWo。自16世纪以来,Lorem Ipsum一直是行业标准的虚拟文本,当时一位不知名的印刷商拿了一个打字机,并把它打乱成打字标本书。它不仅存活了五个世纪,而且还实现了向电子排版的飞跃,基本上保持不变。http://www.youtube.com/watch?v=A_6gNZCkajU&feature=relmfu它在20世纪60年代随着包含Lorem Ipsum段落的Letraset表格的发布而流行起来,最近又随着Aldus PageMaker等桌面出版软件(包括Lorem Ipsum的版本)而普及。

现在我想解析它并找到所有YouTube视频URL及其ID。

任何想法是如何工作的?


答案 1

可能会遇到多种格式的 YouTube 视频网址:

  • 最新短片格式:http://youtu.be/NLqAF9hrVbY
  • 内联框架: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
  • ytscreenroom: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
  • 无 cookie 域: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>');
}

笔记:

  • URL 的VIDEO_ID部分在以下唯一捕获组中捕获:。$1
  • 如果您知道您的文本不包含任何预先链接的 URL,则可以安全地删除测试此条件的否定前瞻断言(以注释开头的断言:“断言 URL 未预先链接。这将在一定程度上加快正则表达式的速度。
  • 可以修改替换字符串以适应。上面提供的那个只是创建一个指向通用样式URL的链接,并将链接文本设置为:。"http://www.youtube.com/watch?v=VIDEO_ID""YouTube link: VIDEO_ID"

编辑 2011-07-05:将连字符添加到 ID 字符类-

编辑 2011-07-17:修复了正则表达式,用于使用YouTube ID后面的URL的任何剩余部分(例如查询)。将函数重命名为驼峰大小写。改进了预链接的预链接前瞻测试。'i'

编辑 2011-07-27:添加了新的“用户”和“ytscreeningroom”格式的YouTube网址。

编辑 2011-08-02:简化/通用化以处理新的“任何/事物/去”YouTube网址。

编辑 2011-08-25:一些修改:

  • 添加了:函数的Javascript版本。linkifyYouTubeURLs()
  • 以前的版本将方案(HTTP协议)部分设置为可选,因此将匹配无效的URL。使方案成为必需的部分。
  • 以前的版本在VIDEO_ID周围使用了边界锚这个词。但是,如果VIDEO_ID以破折号开头或结尾,则此操作将不起作用。已修复,以便它处理此情况。\b-
  • 更改了VIDEO_ID表达式,使其长度必须正好为 11 个字符。
  • 如果预链接的 URL 后面的查询字符串跟在VIDEO_ID,则以前的版本无法排除这些 URL。改进了否定前置断言以解决此问题。
  • 已添加 和 到字符类匹配的查询字符串。+%
  • 将 PHP 版本正则表达式分隔符从: 更改为 a: 。%~
  • 添加了一个“注释”部分,其中包含一些方便的注释。

编辑 2011-10-12:YouTube 网址主机部分现在可能具有任何子域(不仅仅是 )。www.

编辑 2012-05-01:使用 URL 部分现在可能允许使用“-”。

编辑 2013-08-23:添加了@Mei提供的其他格式。(查询部分可能有一个点。.

编辑 2013-11-30:添加了由 @CRONUS 提供的其他格式:。youtube-nocookie.com

编辑 2016-01-25:修复了正则表达式来处理 CRONUS 提供的错误情况。


答案 2

这是我曾经为一个提取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. 找到一个将从文本中提取所有链接的正则表达式。谷歌会在那里帮助你。
  2. 循环所有链接并为每个链接调用 getHostInfo()

推荐