ฉันจะค้นหารหัสวิดีโอ YouTube ทั้งหมดในสตริงโดยใช้นิพจน์ทั่วไปได้อย่างไร


91

ฉันมีช่องข้อความที่ผู้ใช้สามารถเขียนอะไรก็ได้

ตัวอย่างเช่น:

Lorem Ipsum เป็นเพียงข้อความจำลอง http://www.youtube.com/watch?v=DUQi_R4Sg อุตสาหกรรมการพิมพ์และการเรียงพิมพ์ 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
  • ใด ๆ / subdomain / เกินไป: 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 ที่มี regex ที่แสดงความคิดเห็นซึ่งตรงกับแต่ละรูปแบบ 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 ที่มี regex เหมือนกันทุกประการ (โดยลบความคิดเห็นออก):

// 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 ที่ลิงก์ไว้ล่วงหน้าคุณสามารถลบการยืนยันผู้มองเชิงลบซึ่งทดสอบเงื่อนไขนี้ได้อย่างปลอดภัย (คำยืนยันที่ขึ้นต้นด้วยความคิดเห็น: "Assert URL is not pre-linked" ) ซึ่งจะช่วยเร่งความเร็ว regex ขึ้นบ้าง
  • สตริงการแทนที่สามารถแก้ไขให้เหมาะสมได้ หนึ่งที่ให้ไว้ข้างต้นก็สร้างลิงก์ไปยังทั่วไป"http://www.youtube.com/watch?v=VIDEO_ID"URL "YouTube link: VIDEO_ID"สไตล์และการตั้งค่าการเชื่อมโยงข้อความไปที่:

แก้ไข 2011-07-05:เพิ่ม-ยัติภังค์ในคลาส ID char

แก้ไข 2011-07-17:แก้ไข regex เพื่อใช้ส่วนที่เหลือ (เช่นการสืบค้น ) ของ URL ตาม YouTube ID เพิ่มตัวปรับ'i' กรณีละเว้น เปลี่ยนชื่อฟังก์ชันเป็น camelCase ปรับปรุงการทดสอบ Lookahead ที่เชื่อมโยงล่วงหน้า

แก้ไข 2011-07-27:เพิ่มรูปแบบ "ผู้ใช้" และ "ytscreeningroom" ใหม่ของ URL ของ YouTube

แก้ไข 2011-08-02:ประยุกต์ / ใช้งานทั่วไปเพื่อจัดการ URL YouTube "any / thing / go" ใหม่

แก้ไข 2011-08-25:การปรับเปลี่ยนหลายอย่าง:

  • เพิ่มเวอร์ชัน Javascript ของ: linkifyYouTubeURLs()function
  • เวอร์ชันก่อนหน้ามีส่วนแบบแผน (โปรโตคอล HTTP) เป็นทางเลือกดังนั้นจะตรงกับ URL ที่ไม่ถูกต้อง กำหนดส่วนโครงการที่จำเป็น
  • เวอร์ชันก่อนหน้าใช้\bคำว่าจุดยึดขอบเขตรอบ VIDEO_ID อย่างไรก็ตามจะใช้ไม่ได้หาก VIDEO_ID เริ่มต้นหรือลงท้ายด้วย-เครื่องหมายขีดกลาง แก้ไขเพื่อให้สามารถจัดการกับเงื่อนไขนี้ได้
  • เปลี่ยนนิพจน์ VIDEO_ID เพื่อให้มีความยาว 11 อักขระ
  • เวอร์ชันก่อนหน้าไม่สามารถยกเว้น URL ที่เชื่อมโยงล่วงหน้าหากมีสตริงการสืบค้นตาม VIDEO_ID ปรับปรุงการยืนยันผู้มองเชิงลบเพื่อแก้ไขปัญหานี้
  • เพิ่ม+และ%ไปยังสตริงการสืบค้นที่ตรงกับคลาสอักขระ
  • เปลี่ยน PHP รุ่น regex คั่นจาก: ไปที่:%~
  • เพิ่มส่วน "หมายเหตุ" พร้อมบันทึกย่อที่มีประโยชน์

แก้ไข 2011-10-12:ตอนนี้ส่วนโฮสต์ URL ของ YouTube อาจมีโดเมนย่อย (ไม่ใช่แค่www.)

แก้ไข 2012-05-01:ตอนนี้ส่วนการใช้ URL อาจอนุญาตสำหรับ "-"

แก้ไข 2013-08-23:เพิ่มรูปแบบเพิ่มเติมโดย @Mei (ส่วนของแบบสอบถามอาจมี.จุด

แก้ไข 2013/11/30:เพิ่มรูปแบบเพิ่มเติมโดย youtube-nocookie.com@CRONUS:

แก้ไข 2016-01-25:แก้ไข regex เพื่อจัดการกรณีข้อผิดพลาดที่จัดทำโดย 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นอกเหนือจาก URL ทั่วไป
Chamilyan

ยินดีต้อนรับ ขอบคุณสำหรับการอัปเดตฉันแก้ไขในการเปลี่ยนแปลง หมายเหตุด้านข้าง regex ของ ridgerunner ดูเหมือนจะเป็นเรื่องจริงและฉันขอแนะนำให้ใช้มันกับสิ่งง่ายๆของฉัน ไชโย
Christof

สิ่งที่ฉันกำลังมองหา จุดคู่! +1
blackpla9ue

8

แม้ว่าคำตอบของ ridgerunner จะเป็นพื้นฐานสำหรับคำตอบของฉัน แต่ก็ไม่ได้แก้ปัญหาสำหรับ URL ทั้งหมดและฉันไม่เชื่อว่าจะสามารถทำได้เนื่องจากการจับคู่ที่เป็นไปได้หลายรายการVIDEO_IDใน URL ของ YouTube regex ของฉันรวมถึงวิธีการที่ก้าวร้าวของเขาเป็นทางเลือกสุดท้าย แต่จะพยายามจับคู่ทั่วไปทั้งหมดก่อนซึ่งจะช่วยลดความเป็นไปได้ที่จะจับคู่ผิดใน URL ในภายหลัง

regex นี้:

/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_

ฉันตรวจสอบออนไลน์ที่นี่บนรูปูลาร์กับสตริงตัวอย่างของคุณ


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 ทำไมคุณถึงต้องการสร้างรายการที่มีลิงก์ทั้งหมดในตอนแรก และฉันคิดว่าไม่จำเป็นต้องใช้ regexes 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 อื่น ๆ จากเนื้อหา
Rahil Wazir

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 ทั้งหมดในปัจจุบัน

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.