溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android中如何利用FTP實(shí)現(xiàn)多線程斷點(diǎn)續(xù)傳下載上傳功能

發(fā)布時(shí)間:2022-04-08 16:25:46 來源:億速云 閱讀:522 作者:iii 欄目:編程語言

這篇文章主要介紹了Android中如何利用FTP實(shí)現(xiàn)多線程斷點(diǎn)續(xù)傳下載上傳功能的相關(guān)知識(shí),內(nèi)容詳細(xì)易懂,操作簡單快捷,具有一定借鑒價(jià)值,相信大家閱讀完這篇Android中如何利用FTP實(shí)現(xiàn)多線程斷點(diǎn)續(xù)傳下載上傳功能文章都會(huì)有所收獲,下面我們一起來看看吧。

FTP下載原理

FTP單線程斷點(diǎn)續(xù)傳

FTP和傳統(tǒng)的HTTP協(xié)議有所不同,由于FTP沒有所謂的頭文件,因此我們不能像HTTP那樣通過設(shè)置header向服務(wù)器指定下載區(qū)間。

但是FTP協(xié)議提供了一個(gè)更好用的命令REST用于從指定位置恢復(fù)任務(wù),同時(shí)FTP協(xié)議也提供了一個(gè)命令SIZE用于獲取下載的文件大小,有了這兩個(gè)命令,F(xiàn)TP斷點(diǎn)續(xù)傳也就沒有什么問題。

FTP斷點(diǎn)續(xù)傳的原理和HTTP的斷點(diǎn)續(xù)傳原理差不多,在暫停時(shí)記錄文件的停止位置,再次下載時(shí),先讀取記錄的位置,如果位置存在,則通過REST命令告訴服務(wù)器從指定區(qū)間進(jìn)行下載。

FTP多線程斷點(diǎn)續(xù)傳

多線程下載的原理和HTTP多線程下載的原理差不多。先獲取文件大小,然后根據(jù)線程數(shù),對(duì)整個(gè)文件進(jìn)行分段下載,在任務(wù)停止時(shí),記錄每一條線程的暫停位置,重新開始下載,每一條線程讀取對(duì)應(yīng)的下載記錄,然后每一線程從指定位置開始下載。

Android中如何利用FTP實(shí)現(xiàn)多線程斷點(diǎn)續(xù)傳下載上傳功能

分段下載

和HTTP所不同的是,F(xiàn)TP并沒有提供文件區(qū)間的API,因此,F(xiàn)TP在分段下載中,只有起始位置而沒有結(jié)束位置。
因此,你需要在指定位置手動(dòng)停止線程。

功能實(shí)現(xiàn)

本文使用將采用apache commons-net實(shí)現(xiàn)FTP斷點(diǎn)續(xù)傳下載\上傳功能。<br>

通過下文的幾步操作,你就能很簡單的實(shí)現(xiàn)FTP斷點(diǎn)續(xù)傳。

登錄

FTP協(xié)議和HTTP協(xié)議有所不同,使用FTP進(jìn)行下載時(shí),你需要進(jìn)行登錄操作。

當(dāng)然,如果你服務(wù)器沒有登錄功能,你可以忽略登錄操作。

FTPClient client = new FTPClient();
client.connect(serverIp, port); //連接到FTP服務(wù)器
client.login(userName, passsword);

通過上面三行代碼,就可以很簡單的登錄到FTP服務(wù)器上。

在進(jìn)行登錄后,還需要驗(yàn)證是否登錄成功

int reply = client.getReplyCode();
if (!FTPReply.isPositiveCompletion(reply)) {
  client.disconnect();
  Log.d(TAG, "無法連接到ftp服務(wù)器,錯(cuò)誤碼為:" + reply);
  return;
 }

由于FTP協(xié)議中,連接成功的狀態(tài)有多個(gè),因此需要通過FTPReply.isPositiveCompletion(reply)用于驗(yàn)證是否成功連接到FTP服務(wù)器。

文件信息獲取

在連接到FTP服務(wù)器后,就需要開始獲取下載最重要的幾個(gè)參數(shù)(文件長度、文件名)。

客戶端可以通過client.listFiles(remotePath)獲取FTP服務(wù)器上該路徑的文件列表。

  • 如果路徑是文件,只會(huì)返回一個(gè)長度為1的數(shù)組。

  • 如果該路徑為文件夾,則會(huì)返回該文件夾下對(duì)應(yīng)的所有文件。

String remotePath = "/upload/qjnn.apk"; //FTP服務(wù)器上文件路徑
FTPFile[] files = client.listFiles(remotePath);
FTPFile file = files[0]; //文件信息
long size = file.getSize();
String fileaName = file.getName();

如果你的文件為英文名,并且路徑中沒有中文,那么通過上述代碼,便可以獲取到正確的文件信息。

但如果FTP上的服務(wù)器上的文件名有中文或路徑有中文,那么上述代碼,你將獲取不到正確的文件信息。

正確的寫法

由于FTP服務(wù)器默認(rèn)的編碼是ISO-8859-1,因此,客戶端在獲取文件信息時(shí)

  • 需要請(qǐng)求服務(wù)器使用UTF-8編碼(如果服務(wù)器支持的話),如果服務(wù)器不支持開啟UTF-8編碼,那么客戶端需要指定字符串編碼格式

  • 客戶端在請(qǐng)求remotePath路徑、獲取文件名時(shí),都需要對(duì)路徑進(jìn)行編碼轉(zhuǎn)換處理。

String remotePath = "/upload/qjnn.apk"; //FTP服務(wù)器上文件路徑

String charSet = "UTF-8";
if (!FTPReply.isPositiveCompletion(client.sendCommand("OPTS UTF8", "ON"))) {  //向服務(wù)器請(qǐng)求使用"UTF-8"編碼
  charSet = "GBK";
}
FTPFile[] files = client.listFiles(new String(remotePath.getBytes(charSet), "ISO-8859-1")); //對(duì)remotePath進(jìn)行編碼轉(zhuǎn)換
FTPFile file = files[0]; //文件信息
long size = file.getSize();
String fileaName = new String(fileName.getBytes(), Charset.forName(charSet));

通過以上代碼,便可以獲取到正確的文件信息。

文件下載

配置每條線程的下載區(qū)間

long fileLength = mEntity.getFileSize();
Properties pro = CommonUtil.loadConfig(mConfigFile);
int blockSize = (int) (fileLength / mThreadNum);
int[] recordL = new int[mThreadNum];
for (int i = 0; i < mThreadNum; i++) {
 recordL[i] = -1;
}
int rl = 0;
for (int i = 0; i < mThreadNum; i++) {
 long startL = i * blockSize, endL = (i + 1) * blockSize;
 Object state = pro.getProperty(mTempFile.getName() + "_state_" + i);
 if (state != null && Integer.parseInt(state + "") == 1) { //該線程已經(jīng)完成
  if (resumeRecordLocation(i, startL, endL)) return;
  continue;
 }
 //分配下載位置
 Object record = pro.getProperty(fileName + "_record_" + i);
 //如果有記錄,則恢復(fù)下載
 if (record != null && Long.parseLong(record + "") >= 0) {
  Long r = Long.parseLong(record + "");
  mConstance.CURRENT_LOCATION += r - startL;
  Log.d(TAG, "任務(wù)【" + mEntity.getFileName() + "】線程__" + i + "__恢復(fù)下載");
  startL = r;
  recordL[rl] = i;
  rl++;
 } else {
  recordL[rl] = i;
  rl++;
 }
 //最后一個(gè)線程的結(jié)束位置即為文件的總長度
 if (i == (mThreadNum - 1)) endL = fileLength;
 //創(chuàng)建分段線程
 AbsThreadTask task = createSingThreadTask(i, startL, endL, fileLength);
 if (task == null) return;
 mTask.put(i, task);
}
startSingleTask(recordL);

在上面的代碼中,主要做了兩步操作:

  • 在文件下載前,先從本地文件中讀取當(dāng)前下載的每一條線程的下載情況

  • 如果下載記錄存在,從記錄位置開始下載,如果記錄不存在,則重新開始下載

FTP 分段線程區(qū)間自動(dòng)停止

由于FTP協(xié)議沒有區(qū)間下載的原因,為了讓線程只下載特定區(qū)間的內(nèi)容,需要客戶端在單條線程累計(jì)讀的數(shù)據(jù)長度已經(jīng)超過了所分配的區(qū)間長度的時(shí)候,停止該條線程。

 client.enterLocalPassiveMode();  //設(shè)置被動(dòng)模式
 client.setFileType(FTP.BINARY_FILE_TYPE); //設(shè)置文件傳輸模式
 client.setRestartOffset(mConfig.START_LOCATION);  //設(shè)置恢復(fù)下載的位置
 client.allocate(mBufSize);
 is = client.retrieveFileStream(new String(remotePath.getBytes(charSet), SERVER_CHARSET));
 //發(fā)送第二次指令時(shí),還需要再做一次判斷
 reply = client.getReplyCode();
 if (!FTPReply.isPositivePreliminary(reply)) {
  client.disconnect();
  fail(mChildCurrentLocation, "獲取文件信息錯(cuò)誤,錯(cuò)誤碼為:" + reply, null);
  return;
 }
 file = new BufferedRandomAccessFile(mConfig.TEMP_FILE, "rwd", mBufSize);
 file.seek(mConfig.START_LOCATION);
 byte[] buffer = new byte[mBufSize];
 int len;
 while ((len = is.read(buffer)) != -1) { 
  //如果該條線程讀取的數(shù)據(jù)長度大于所分配的區(qū)間長度,則只能讀到區(qū)間的最大長度
  if (mChildCurrentLocation + len >= mConfig.END_LOCATION) {
    len = (int) (mConfig.END_LOCATION - mChildCurrentLocation);
    file.write(buffer, 0, len);
    progress(len);
    break;
  } else {
    file.write(buffer, 0, len);
    progress(len);
  }
 }

這里還有幾個(gè)坑需要處理一下:

  1. 對(duì)于FTP客戶端來說,一般需要設(shè)置被動(dòng)模式,被動(dòng)模式和主動(dòng)模式的區(qū)別

  2. 在獲取文件流后,還需要使用FTPReply.isPositivePreliminary(reply)進(jìn)行第二次命令判斷

關(guān)于FTP文件上傳

FTP 文件斷點(diǎn)續(xù)傳的方式原理和下載的都差不多:

  1. 都是在停止的時(shí)候記錄停止位置,重新開始下載的時(shí)候從指定位置通過REST命令恢復(fù)斷點(diǎn)。

  2. 都需要在任務(wù)執(zhí)行前獲取文件信息,比對(duì)服務(wù)器上的文件。

而和下載有區(qū)別的是:

  1. FTP上傳時(shí)需要指定工作目錄、在遠(yuǎn)程服務(wù)器上創(chuàng)建文件夾

  2. 需要服務(wù)器給用戶打開刪除和讀入IO的權(quán)限,否則會(huì)出現(xiàn)550權(quán)限錯(cuò)誤問題

關(guān)于“Android中如何利用FTP實(shí)現(xiàn)多線程斷點(diǎn)續(xù)傳下載上傳功能”這篇文章的內(nèi)容就介紹到這里,感謝各位的閱讀!相信大家對(duì)“Android中如何利用FTP實(shí)現(xiàn)多線程斷點(diǎn)續(xù)傳下載上傳功能”知識(shí)都有一定的了解,大家如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI