首先來(lái)看一下多線(xiàn)程下載的原理。多線(xiàn)程下載就是將同一個(gè)網(wǎng)絡(luò)上的原始文件根據(jù)線(xiàn)程個(gè)數(shù)分成均等份,然后每個(gè)單獨(dú)的線(xiàn)程下載對(duì)應(yīng)的一部分,然后再將下載好的文件按照原始文件的順序“拼接”起來(lái)就構(gòu)
成了完整的文件了。這樣就大大提高了文件的下載效率。對(duì)于文件下載來(lái)說(shuō),多線(xiàn)程下載是必須要考慮的環(huán)節(jié)。
多線(xiàn)程下載大致可分為以下幾個(gè)步驟:
一.獲取服務(wù)器上的目標(biāo)文件的大小
顯然這一步是需要先訪(fǎng)問(wèn)一下網(wǎng)絡(luò),只需要獲取到目標(biāo)文件的總大小即可。目的是為了計(jì)算每個(gè)線(xiàn)程應(yīng)該分配的下載任務(wù)。
二. 在本地創(chuàng)建一個(gè)跟原始文件同樣大小的文件
在本地可以通過(guò)RandomAccessFile 創(chuàng)建一個(gè)跟目標(biāo)文件同樣大小的文件,該api 支持文件任意位置的讀寫(xiě)操作。這樣就給多線(xiàn)程下載提供了方便,每個(gè)線(xiàn)程只需在指定起始和結(jié)束腳標(biāo)范圍內(nèi)寫(xiě)數(shù)據(jù)即可。
三.計(jì)算每個(gè)線(xiàn)程下載的起始位置和結(jié)束位置
我們可以把原始文件當(dāng)成一個(gè)字節(jié)數(shù)組,每個(gè)線(xiàn)程只下載該“數(shù)組”的指定起始位置和指定結(jié)束位置之間的部分。在第一步中我們已經(jīng)知道了“數(shù)組”的總長(zhǎng)度。因此只要再知道總共開(kāi)啟的線(xiàn)程的個(gè)數(shù)就好計(jì)算每個(gè)線(xiàn)程要下載的范圍了。每個(gè)線(xiàn)程需要下載的字節(jié)個(gè)數(shù)(blockSize)=總字節(jié)數(shù)(totalSize)/線(xiàn)程數(shù)(threadCount)。 假設(shè)給線(xiàn)程按照0,1,2,3...n 的方式依次進(jìn)行編號(hào),那么第n 個(gè)線(xiàn)程下載文件的范圍為:
起始腳標(biāo)startIndex=n*blockSize。
結(jié)束腳標(biāo)endIndex=(n-1)*blockSize-1。
考慮到totalSize/threadCount 不一定能整除,因此對(duì)已最后一個(gè)線(xiàn)程應(yīng)該特殊處理,最后一個(gè)線(xiàn)程的起始腳標(biāo)計(jì)算公式不變,但是結(jié)束腳標(biāo)為endIndex=totalSize-1即可。
四.開(kāi)啟多個(gè)子線(xiàn)程開(kāi)始下載
在子線(xiàn)程中實(shí)現(xiàn)讀流操作,將conn.getInputStream()讀到RandomAccessFile中。