技術(shù)提升美好事物發(fā)生的概率。Technologically, for greater probability to be happy
只要在 HTML5 中使用過視頻播放的同學(xué)對 video 標(biāo)簽一定不會(huì)陌生,不過很多同學(xué)只使用了 video 的基礎(chǔ)功能,實(shí)際上 video 擁有強(qiáng)大潛能的,只要姿勢正確就能讓其擁有超能力。不妨從下面幾個(gè)場景來逐漸了解下video 未曾被發(fā)掘的神秘空間:
點(diǎn)播領(lǐng)域里 mp4 是最普遍、兼容性最好的視頻容器,不過 mp4 也有它的局限性,比如常見的清晰度切換,我們是無法像youtube那樣做到無縫切換的。我們可以看下普通的mp4播放的網(wǎng)絡(luò)請求和youtube視頻播放的網(wǎng)絡(luò)請求的區(qū)別。
圖1.1 普通mp4的下載請求過程
圖1.2 Youtube視頻下載請求過程
這兩張圖不難看出,在默認(rèn)情況下 mp4 使用一次 http 請求所有的視頻數(shù)據(jù),Youtube 則分次請求。當(dāng)然這個(gè)描述很不專業(yè),但確實(shí)形象。造成這種差異的是 video 不支持流式的視頻數(shù)據(jù),Youtube 采用的是流式的視頻容器 webm,而 mp4 是非流式的。那如何解釋清楚流式的視頻數(shù)據(jù)呢,從專業(yè)的角度三言兩語很難說清楚,但用大白話翻譯過來就是流式的視頻數(shù)據(jù)支持分段獨(dú)立播放,非流式的不可以。換句話說一個(gè)10M的視頻文件,流式的視頻可以把0~1M的數(shù)據(jù)請求回來單獨(dú)播放,但是非流式的不可以。
上面我們描述了視頻格式的不同,接下來我們要說的是第一張圖中的視頻加載是瀏覽器來控制的,通過給 video 的 src 屬性配置視頻地址,觸發(fā)播放之后瀏覽器就會(huì)開始下載了,JS干涉不了。而 Youtube 的視頻加載是通過JS來控制的,各位可以再次看下第二張圖的網(wǎng)絡(luò)請求類型:xhr,足以證明這一點(diǎn)。
上面兩點(diǎn)搞清楚之后我們就該說下清晰度切換的事情了。這個(gè)需求大家都不陌生,但是直接使用 mp4 格式做無縫清晰度切換,難度還挺大的。先解釋下“無縫清晰度切換”的概念:從播放一個(gè)分辨率的視頻到另一個(gè)分辨率且保證畫面、聲音不停頓的平滑切換過程。了解了這個(gè)概念,大家應(yīng)該知道了用 video 無縫切換 mp4 有多難。一方面,video 是不支持流式的視頻格式的,一方面,video 的加載是不受JS控制的。通過切換 video 的 src 屬性,必然會(huì)導(dǎo)致畫面中斷、重新請求視頻數(shù)據(jù)等。有的同學(xué)想到說利用兩個(gè) video 再結(jié)合 z-index 來搞,但是當(dāng)你生成另一個(gè)video去加載視頻的時(shí)候,無法保證兩個(gè)畫面是嚴(yán)格一致的,即使將原來的畫面暫停到一個(gè)時(shí)刻,用另一個(gè)視頻通過 currentTime 屬性與之同步,切換仍然看到畫面閃爍,基本無法和 Youtube 無縫切換的體驗(yàn)匹敵。而且還會(huì)造成更多流量的浪費(fèi),背后的原因大家可以研究下 mp4 容器和 webm 容器的異同,也可以看下視頻解碼相關(guān)的文章。
還有一種方法就是將 mp4 格式統(tǒng)統(tǒng)轉(zhuǎn)碼到流式的視頻格式比如 hls、webm 等。不過這種看上去可行的方式實(shí)際上會(huì)帶來很大的成本開銷,如將大量視頻做轉(zhuǎn)碼會(huì)消耗高昂的機(jī)器資源、雙倍存儲(chǔ)的費(fèi)用、CDN的雙倍費(fèi)用等等。其實(shí)我們也是在這種背景下研究出來新的技術(shù)問題解決清晰度無縫切換的。
首先,我們改變對 mp4 視頻的播放流程,不再直接使用 video 的 src 來播放,因?yàn)槲覀儧]有任何可以操作的空間。video不僅支持 src 屬性還支持 Blob 對象,我們就是利用后者。播放的流程如下:
圖1.3 mp4 視頻新播放流程
然后在做清晰度切換的時(shí)候流程如下:
圖1.4 mp4視頻清晰度切換原理示意圖
圖1.5 mp4視頻清晰度切換流程示意圖
這個(gè)過程看上去比較繁瑣,但是所有的操作都是在瀏覽器端完成,也就是說都是JS來實(shí)現(xiàn)的。這樣之前說的所有成本問題都不存在,還能做到y(tǒng)outube相同體驗(yàn)的無縫切換。如果大家也想使用這個(gè)功能不需要自己再去實(shí)現(xiàn)一遍上述流程,可以使用如下代碼:
如果對這段代碼有什么疑惑,或者想深入了解下它背后是如何實(shí)現(xiàn)的,可以參考 Github:https://github.com/bytedance/xgplayer 或 閱讀原文:https://techblog.toutiao.com/
使用 video 的同學(xué)基本上都是這樣用的,如下:
2.利用source標(biāo)簽
這樣就可以播放視頻了,不過前面我們講過這樣使用 video ,視頻的加載是受瀏覽器控制的,可以看下瀏覽器在視頻剛開始播放的時(shí)候下載了多少數(shù)據(jù):
圖2.1 video默認(rèn)下載截圖
我隨便找了個(gè)視頻,大家看下視頻總長度是 02:08,在播放到 00:05 的時(shí)候,瀏覽器已經(jīng)下載到 01:30 了,如果用戶終止觀看,下載的視頻就這樣被浪費(fèi)掉了。當(dāng)然,如果不斷的 seek 也會(huì)造成較多的流量浪費(fèi)。按照我們之前的統(tǒng)計(jì)在短視頻領(lǐng)域,用戶 seek 的頻率在 80%,所以這部分流量是可以節(jié)省掉的。具體原理如下:
圖2.2 播放器加載視頻原理
具體實(shí)現(xiàn)代碼如下:
這樣就實(shí)現(xiàn)了視頻在播放過程中永遠(yuǎn)只預(yù)加載10秒的數(shù)據(jù),進(jìn)而保證節(jié)省流量。
擴(kuò)展鏈接,了解超能力西瓜視頻是怎樣煉成的。
多站點(diǎn)都會(huì)使用到視頻. HTML5 提供了展示視頻的標(biāo)準(zhǔn)。
檢測您的瀏覽器是否支持 HTML5 視頻:
檢測
Web站點(diǎn)上的視頻
直到現(xiàn)在,仍然不存在一項(xiàng)旨在網(wǎng)頁上顯示視頻的標(biāo)準(zhǔn)。
今天,大多數(shù)視頻是通過插件(比如 Flash)來顯示的。然而,并非所有瀏覽器都擁有同樣的插件。
HTML5 規(guī)定了一種通過 video 元素來包含視頻的標(biāo)準(zhǔn)方法。
瀏覽器支持
Internet Explorer 9+, Firefox, Opera, Chrome, 和 Safari 支持 <video> 元素.
注意: Internet Explorer 8 或者更早的IE版本不支持 <video> 元素。
HTML5 (視頻)- 如何工作
如需在 HTML5 中顯示視頻,您所有需要的是:
實(shí)例
<video width="320" height="240" controls>
<source src="movie.mp4" type="video/mp4">
<source src="movie.ogg" type="video/ogg">
您的瀏覽器不支持Video標(biāo)簽。
</video>
<video> 元素提供了 播放、暫停和音量控件來控制視頻。
同時(shí)<video> 元素元素也提供了 width 和 height 屬性控制視頻的尺寸.如果設(shè)置的高度和寬度,所需的視頻空間會(huì)在頁面加載時(shí)保留。。如果沒有設(shè)置這些屬性,瀏覽器不知道大小的視頻,瀏覽器就不能再加載時(shí)保留特定的空間,頁面就會(huì)根據(jù)原始視頻的大小而改變。
<video> 與</video> 標(biāo)簽之間插入的內(nèi)容是提供給不支持 video 元素的瀏覽器顯示的。
<video> 元素支持多個(gè) <source> 元素. <source> 元素可以鏈接不同的視頻文件。瀏覽器將使用第一個(gè)可識(shí)別的格式:
視頻格式與瀏覽器的支持
當(dāng)前, <video> 元素支持三種視頻格式: MP4, WebM, 和 Ogg:
瀏覽器 | MP4 | WebM | Ogg |
---|---|---|---|
Internet Explorer | YES | NO | NO |
Chrome | YES | YES | YES |
Firefox | YES | YES | YES |
Safari | YES | NO | NO |
Opera | YES (從 Opera 25 起) | YES | YES |
MP4 = 帶有 H.264 視頻編碼和 AAC 音頻編碼的 MPEG 4 文件
WebM = 帶有 VP8 視頻編碼和 Vorbis 音頻編碼的 WebM 文件
Ogg = 帶有 Theora 視頻編碼和 Vorbis 音頻編碼的 Ogg 文件
視頻格式
格式 | MIME-type |
---|---|
MP4 | video/mp4 |
WebM | video/webm |
Ogg | video/ogg |
HTML5 <video> - 使用 DOM 進(jìn)行控制
HTML5 <video> 和 <audio> 元素同樣擁有方法、屬性和事件。
<video> 和 <audio>元素的方法、屬性和事件可以使用JavaScript進(jìn)行控制.
其中的方法有用于播放、暫停以及加載等。其中的屬性(比如時(shí)長、音量等)可以被讀取或設(shè)置。其中的 DOM 事件能夠通知您,比方說,<video> 元素開始播放、已暫停,已停止,等等。
例中簡單的方法,向我們演示了如何使用 <video> 元素,讀取并設(shè)置屬性,以及如何調(diào)用方法。
實(shí)例 1
為視頻創(chuàng)建簡單的播放/暫停以及調(diào)整尺寸控件:
播放/暫停 放大 縮小 普通
上面的例子調(diào)用了兩個(gè)方法:play() 和 pause()。它同時(shí)使用了兩個(gè)屬性:paused 和 width。
HTML5 Video 標(biāo)簽
標(biāo)簽 | 描述 |
---|---|
<video> | 定義一個(gè)視頻 |
<source> | 定義多種媒體資源,比如 <video> 和<audio> |
<track> | 定義在媒體播放器文本軌跡 |
meta主要用于設(shè)置網(wǎng)頁中的一些元數(shù)據(jù),元數(shù)據(jù)不是給用戶看 charset 指定網(wǎng)頁的字符集 name 指定的數(shù)據(jù)的名稱 content 指定的數(shù)據(jù)的內(nèi)容
keywords 表示網(wǎng)站的關(guān)鍵字,可以同時(shí)指定多個(gè)關(guān)鍵字,關(guān)鍵字間使用,隔開
<meta name="Keywords" content="網(wǎng)上購物,網(wǎng)上商城,手機(jī),筆記本,電腦,MP3,CD,VCD,DV,相機(jī),數(shù)碼,配件,手表,存儲(chǔ)卡,京東"/>
description 用于指定網(wǎng)站的描述
<meta name="description" content="京東JD.COM-專業(yè)的綜合網(wǎng)上購物商城,銷售家電、數(shù)碼通訊、電腦、家居百貨、服裝服飾、母嬰、圖書、食品等數(shù)萬個(gè)品牌優(yōu)質(zhì)商品.便捷、誠信的服務(wù),為您提供愉悅的網(wǎng)上購物體驗(yàn)!"/>
<meta http-equiv="refresh" content="3;url=https://www.mozilla.org">
將頁面重定向到另一個(gè)網(wǎng)站
title標(biāo)簽的內(nèi)容會(huì)作為搜索結(jié)果的超鏈接上的文字顯示
<title>Document</title>
header 表示網(wǎng)頁的頭部 main 表示網(wǎng)頁的主體部分(一個(gè)頁面中只會(huì)有一個(gè)main) footer 表示網(wǎng)頁的底部 nav 表示網(wǎng)頁中的導(dǎo)航 aside 和主體相關(guān)的其他內(nèi)容(側(cè)邊欄) article 表示一個(gè)獨(dú)立的文章 section 表示一個(gè)獨(dú)立的區(qū)塊,上邊的標(biāo)簽都不能表示時(shí)使用section
div 沒有語義,就用來表示一個(gè)區(qū)塊,目前來講div還是我們主要的布局元素
span 行內(nèi)元素,沒有任何的語義,一般用于在網(wǎng)頁中選中文字
<header></header>
<main></main>
<footer></footer>
<nav></nav>
<aside></aside>
<article></article>
<section></section>
<div></div>
<span></span>
使用ol標(biāo)簽來創(chuàng)建無序列表,使用li表示列表項(xiàng)
<ul>
<li>結(jié)構(gòu)</li>
<li>表現(xiàn)</li>
<li>行為</li>
</ul>
使用ul標(biāo)簽來創(chuàng)建無序列表,使用li表示列表項(xiàng)
<ol>
<li>結(jié)構(gòu)</li>
<li>表現(xiàn)</li>
<li>行為</li>
</ol>
使用dl標(biāo)簽來創(chuàng)建一個(gè)定義列表, 使用dt來表示定義的內(nèi)容,使用dd來對內(nèi)容進(jìn)行解釋說明
<dl>
<dt>結(jié)構(gòu)</dt>
<dd>結(jié)構(gòu)表示網(wǎng)頁的結(jié)構(gòu),結(jié)構(gòu)用來規(guī)定網(wǎng)頁中哪里是標(biāo)題,哪里是段落</dd>
<dd>結(jié)構(gòu)表示網(wǎng)頁的結(jié)構(gòu),結(jié)構(gòu)用來規(guī)定網(wǎng)頁中哪里是標(biāo)題,哪里是段落</dd>
<dd>結(jié)構(gòu)表示網(wǎng)頁的結(jié)構(gòu),結(jié)構(gòu)用來規(guī)定網(wǎng)頁中哪里是標(biāo)題,哪里是段落</dd>
</dl>
<ul>
<li>
aa
<ul>
<li>aa-1</li>
<li>aa-2
<ul>
<li>aa-1</li>
<li>aa-2</li>
</ul>
</li>
</ul>
</li>
</ul>
超鏈接可以讓我們從一個(gè)頁面跳轉(zhuǎn)到其他頁面, 或者是當(dāng)前頁面的其他的位置
使用 a 標(biāo)簽來定義超鏈接
<a href="https://www.baidu.com">超鏈接</a>
指定跳轉(zhuǎn)的目標(biāo)路徑
值可以是一個(gè)外部網(wǎng)站的地址
也可以寫一個(gè)內(nèi)部頁面的地址
超鏈接是也是一個(gè)行內(nèi)元素,在a標(biāo)簽中可以嵌套除它自身外的任何元素
用來指定超鏈接打開的位置
_self 默認(rèn)值 在當(dāng)前頁面中打開超鏈接
_blank 在一個(gè)新的要么中打開超鏈接
<a href="07.列表.html" target="_blank">超鏈接</a>
將#作為超鏈接的路徑的展位符使用
javascript:; 來作為href的屬性,此時(shí)點(diǎn)擊這個(gè)超鏈接什么也不會(huì)發(fā)生
將超鏈接的href屬性設(shè)置為#,這樣點(diǎn)擊超鏈接以后 頁面不會(huì)發(fā)生跳轉(zhuǎn),而是轉(zhuǎn)到當(dāng)前頁面的頂部的位置
跳轉(zhuǎn)到頁面的指定位置,只需將href屬性設(shè)置 #目標(biāo)元素的id屬性值
<a href="#bottom">去底部</a>
<br><br>
<a href="#p3">去第三個(gè)自然段</a>
<br><br>
<p>
內(nèi)容多一點(diǎn)
</p>
<a href="#">這是一個(gè)新的超鏈接</a>
<br><br>
<a href="javascript:;">這是一個(gè)新的超鏈接</a>
<br><br>
<a id="bottom" href="#">回到頂部</a>
img標(biāo)簽來引入外部圖片,img標(biāo)簽是一個(gè)自結(jié)束標(biāo)簽
屬性:src 屬性指定的是外部圖片的路徑(路徑規(guī)則和超鏈接是一樣的)
alt 圖片的描述,這個(gè)描述默認(rèn)情況下不會(huì)顯示,有些瀏覽器會(huì)圖片無法加載時(shí)顯示
<img src="./img/1.gif" alt="松鼠">
用于向當(dāng)前頁面中引入一個(gè)其他頁面
<iframe src="https://www.qq.com" width="800" height="600" frameborder="0"></iframe>
audio 標(biāo)簽用來向頁面中引入一個(gè)外部的音頻文件的
<audio src="./source/audio.mp3" controls autoplay loop></audio>
除了通過src來指定外部文件的路徑以外,還可以通過source來指定文件的路徑
<audio controls>
<!-- 對不起,您的瀏覽器不支持播放音頻!請升級(jí)瀏覽器!-->
<source src="./source/audio.mp3">
<source src="./source/audio.ogg">
<embed src="./source/audio.mp3" type="audio/mp3" width="300" height="100">
</audio>
與 audio 相似
<video controls>
<source src="./source/flower.webm">
<source src="./source/flower.mp4">
<embed src="./source/flower.mp4" type="video/mp4">
</video>
<table border="1" width='50%' align="center">
<tr>
<td>A1</td>
<td>B1</td>
<td>C1</td>
<td>D1</td>
</tr>
<tr>
<td>A2</td>
<td>B2</td>
<td>C2</td>
<td rowspan="2">D2</td>
</tr>
<tr>
<td>A3</td>
<td>B3</td>
<td>C3</td>
</tr>
<tr>
<td>A4</td>
<td>B4</td>
<td colspan="2">C4</td>
</tr>
</table>
<table border="1" width='50%' align="center">
<thead>
<tr>
<th>日期</th>
<th>收入</th>
<th>支出</th>
<th>合計(jì)</th>
</tr>
</thead>
<tbody>
<tr>
<td>2000.1.1</td>
<td>500</td>
<td>200</td>
<td>300</td>
</tr>
<tr>
<td>2000.1.1</td>
<td>500</td>
<td>200</td>
<td>300</td>
</tr>
<tr>
<td>2000.1.1</td>
<td>500</td>
<td>200</td>
<td>300</td>
</tr>
<tr>
<td>2000.1.1</td>
<td>500</td>
<td>200</td>
<td>300</td>
</tr>
</tbody>
<tfoot>
<tr>
<td></td>
<td></td>
<td>合計(jì)</td>
<td>300</td>
</tr>
</tfoot>
</table>
border-spacing: 0px;
border-collapse: collapse;
<input type="text" name="username">
<input type="radio" name="hello" value="a">
<input type="radio" name="hello" value="b" checked>
<input type="checkbox" name="test" value="1">
<input type="checkbox" name="test" value="2">
<input type="checkbox" name="test" value="3" checked>
<select name="haha">
<option value="i">選項(xiàng)一</option>
<option selected value="ii">選項(xiàng)二</option>
<option value="iii">選項(xiàng)三</option>
</select>
<input type="submit" value="注冊">
<form action="target.html">
<input type="text" name="username" value="hello" readonly>
<br><br>
<input type="text" name="username" autofocus>
<br><br>
<input type="text" name="b">
<br><br>
<!-- <input type="color"> -->
<br><br>
<!-- <input type="email"> -->
<br><br>
<input type="submit">
<!-- 重置按鈕 -->
<input type="reset">
<!-- 普通的按鈕 -->
<input type="button" value="按鈕">
<br><br>
<button type="submit">提交</button>
<button type="reset">重置</button>
<button type="button">按鈕</button>
</form>
<!--
我是注釋中的注釋 注釋不能嵌套
-->
<!doctype html>
*請認(rèn)真填寫需求信息,我們會(huì)在24小時(shí)內(nèi)與您取得聯(lián)系。