摘要:比如這個(gè)在線視頻我們可以正常播放,但是找不到下載按鈕。因此,這個(gè)在線視頻被拆分成了若干多的小片段,然后通過瀏覽器下載到本地進(jìn)行播放。使用自帶的多線程庫多線程下載這些片段。第行可以指定線程池里工作線程的個(gè)數(shù)。
比如這個(gè)在線視頻:
我們可以正常播放,但是找不到下載按鈕。
打開Chrome開發(fā)者工具,在Network標(biāo)簽頁里能看到很多網(wǎng)絡(luò)傳輸請(qǐng)求:
隨便看一個(gè)請(qǐng)求的響應(yīng),發(fā)現(xiàn)類型為video,大小為500多k。因此,這個(gè)在線視頻被拆分成了若干500多k的小片段,然后通過瀏覽器下載到本地進(jìn)行播放。
這個(gè)片段的url:
http://d2vvqvds83fsd.cloudfro...:event/18/36/06/3/rt/1/resources/180919_PID_Intelligent_Enterprise_Gruenewald_720p-5F92.smil/media_b433000_10.ts
那么這個(gè)片段一共有多少個(gè)片段呢?在所有片段開始下載之前,有這樣一個(gè)請(qǐng)求:chunklist即是視頻片段的清單。
通過這個(gè)清單我們知道這個(gè)視頻一共分為55個(gè)片段,序號(hào)從0開始。
了解了原理,我們就可以開始編程了。
1. 首先實(shí)現(xiàn)視頻片段的下載邏輯,新建一個(gè)類,實(shí)現(xiàn)Runnable接口。
2. 使用JDK自帶的多線程庫 ExecutorService多線程下載這些片段。ExecutorService實(shí)際是一個(gè)線程池。第15行可以指定線程池里工作線程(Working thread)的個(gè)數(shù)。
private void download(){ URL task = null; String path = DownloadLauncher.LOCALPATH + this.mIndex + DownloadLauncher.POSTFIX; String url = this.mTask; try { task = new URL(url); DataInputStream dataInputStream = new DataInputStream(task.openStream()); FileOutputStream fileOutputStream = new FileOutputStream(new File(path)); ByteArrayOutputStream output = new ByteArrayOutputStream(); byte[] buffer = new byte[1024]; int length; while ((length = dataInputStream.read(buffer)) > 0) { output.write(buffer, 0, length); } fileOutputStream.write(output.toByteArray()); dataInputStream.close(); fileOutputStream.close(); System.out.println("File: " + this.mIndex + " downloaded ok"); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } }
下載完成后,能在Eclipse的console控制臺(tái)看到這些輸出:
下載成功的視頻片段:
3. Merger負(fù)責(zé)把這些片段合并成一個(gè)大文件。
private static void run() throws IOException{ FileInputStream in = null; String destFile = DownloadLauncher.LOCALPATH + DownloadLauncher.MERGED; FileOutputStream out = new FileOutputStream(destFile,true); for( int i = 0; i <= DownloadLauncher.LAST; i++){ byte[] buf = new byte[1024]; int len = 0; String sourceFile = DownloadLauncher.LOCALPATH + i + DownloadLauncher.POSTFIX; in = new FileInputStream(sourceFile); while( (len = in.read(buf)) != -1 ){ out.write(buf,0,len); } } out.close(); } public static void main(String[] args) { try { run(); } catch (IOException e) { e.printStackTrace(); } System.out.println("Merged ok!"); }
完整的代碼在我的github上:
https://github.com/i042416/Ja...
要獲取更多Jerry的原創(chuàng)文章,請(qǐng)關(guān)注公眾號(hào)"汪子熙":
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/71920.html
摘要:比如這個(gè)在線視頻我們可以正常播放,但是找不到下載按鈕。因此,這個(gè)在線視頻被拆分成了若干多的小片段,然后通過瀏覽器下載到本地進(jìn)行播放。使用自帶的多線程庫多線程下載這些片段。第行可以指定線程池里工作線程的個(gè)數(shù)。 比如這個(gè)在線視頻: showImg(https://segmentfault.com/img/remote/1460000016829926); 我們可以正常播放,但是找不到下載按...
摘要:能理解線程模型多線程優(yōu)缺點(diǎn)以及如何避免。多線程的出現(xiàn)主要是為了提高的利用率任務(wù)的執(zhí)行效率。所以要考慮清楚是否真的需要多線程。這一塊的內(nèi)容可以然我們知道寫大牛處理并發(fā)的思路,對(duì)我們自己編寫高質(zhì)量的多線程程序也有很多幫助。 showImg(https://segmentfault.com/img/remote/1460000015980196?w=2048&h=1363); 前言 已經(jīng)記不...
閱讀 2094·2021-11-24 09:39
閱讀 1563·2021-10-11 10:59
閱讀 2507·2021-09-24 10:28
閱讀 3382·2021-09-08 09:45
閱讀 1275·2021-09-07 10:06
閱讀 1672·2019-08-30 15:53
閱讀 2067·2019-08-30 15:53
閱讀 1424·2019-08-30 15:53