0
点赞
收藏
分享

微信扫一扫

Flutter获取大文件MD5值的方法以及大文件实现分块上传和断点续传

Flutter获取大文件MD5值的方法

最近一直在搞flutter,有一个需求是将一个不到1G的大文件从App端上传到服务器,为了做文件校验所以要获取到文件的MD5。 1.第一步首先获取到文件,并计算出文件大小和分快的数目

File file = File(path);
int fileSize = file.lengthSync();
int totalPart = (fileSize * 1.0 / partSize).ceil();

static const int partSize = 1024 * 1024 * 3; //默认3m每块

2.第二步导入flutter 官方的convert库

import 'package:convert/convert.dart' as convert;/起一个别名

int start; //开始读文件的位置
int length; //读取文件的长度
var output = convert1.AccumulatorSink<Digest>();
var input = md5.startChunkedConversion(output);//这是一个sink 会把所有块的md5拼接到一起

3.第三部循环所有分快

int currentPart = 0;
while (currentPart < totalPart) {
      //文件开始读取的位置
      start = currentPart * partSize;
      //读取的长度 最后一块直接到文件长度
      length = (start + partSize > fileSize) ? (fileSize - start ) : partSize;
      //每块读取的位置
      RandomAccessFile raf = file.openSync(mode: FileMode.read);
      raf.setPositionSync(start);
      Uint8List data = raf.readSync(length);
    
      input.add(data);
      currentPart++;
}

4.获取最终md5,关闭sink

input.close();
    var digest = output.events.single.toString();
    return digest;

5.完整方法调用

String _getMd5(String path) {
    File file = File(path);

    int fileSize = file.lengthSync();
    int totalPart = (fileSize * 1.0 / partSize).ceil();

    int start; //开始读文件的位置
    int length; //读取文件的长度
    var output = convert.AccumulatorSink<Digest>();
    var input = md5.startChunkedConversion(output);

    int currentPart = 0;
    while (currentPart < totalPart) {
      start = currentPart * partSize;
      length = (start + partSize > fileSize) ? (fileSize - start ) : partSize;

      RandomAccessFile raf = file.openSync(mode: FileMode.read);
      raf.setPositionSync(start);
      Uint8List data = raf.readSync(length);
      input.add(data);
      currentPart++;
    }

    input.close();
    var digest = output.events.single.toString();
    return digest;
  }

大文件实现分块上传和断点续传

1.将文件进行分块,我们和后台约定的是3M每块

static const int partSize = 1024 * 1024 * 3; //3m每块

///上传文件 后台要记录从第几块开始传,也就是currentPart
Future<bool> _uploadZip(String path) async {
    File file = File(path);
    String fileName = FileUtil.getBaseName(file.path);
    int fileSize = file.lengthSync(); //文件总大小
    int totalPart = (fileSize * 1.0 / partSize).ceil(); //分块总数 向上取整
    int currentPart = currentPart; //当前要传第几块 由后台记录返回的
   RandomAccessFile raf = file.openSync(mode: FileMode.read);
   //开启循环上传每一块
    while (currentPart < totalPart) {
      bool isOK = await _uploadPart(file, currentPart, totalPart, fileSize, fileName,raf);
      if (!isOK) {
        invokeStateListener(ProgressState.error, null);
        raf.closeSync();
        return false;
      }
      //更新进度
      invokeProgressListeners((currentPart + 1) * 1.0 / totalPart);
      currentPart++;
    }
     raf.closeSync();
    //所有分快上传成功
    invokeStateListener(ProgressState.complete, null);
    return true;
  }

2.将每一个分块上传到后台,并记录上传是否成功

Future<bool> _uploadPart(File file, int currentPart, int totalPart, int fileSize, String fileName,RandomAccessFile raf) async {
    //开始读文件的位置     
    int start = currentPart * partSize; 
    //读取文件的长度 注意不要让长度超过总长度
    var length = (start + partSize > fileSize) ? (fileSize - start) : partSize; 
    raf.setPositionSync(start);
    Uint8List data = raf.readSync(length);//从文件总长度中截取出要上传的长度

    Map<String, dynamic> map = {};
    map['total'] = totalPart; //总块数
    map['currentPart'] = currentPart; //当前上传到第几块
    map['part'] = MultipartFile.fromBytes(data, filename:'随便设置文件名');//当前快的数据 

    Map<String, dynamic>? response = await 真正的上传函数
    //自定义逻辑 失败的话返回false
 
    return true;
  }

举报

相关推荐

0 条评论