GridFSDownloadStream无法读取所有数据

问题描述 投票:0回答:3

我尝试从mongodb下载文件。上传文件的大小是2106字节和chunkSize=2048字节,因此GridFS将文件分为2个块​​。当我执行downloadStream.read(bytesToWriteTo)的代码时,只读取第一个块的数据,而不能读取第二个块的数据。我怎样才能读取所有数据?

public class OpenDownloadStreamDemo {

          public static void main(String[] args) {
            MongoClient mongoClient = new MongoClient("127.0.0.1", 27017);
            MongoDatabase mongoDatabase = mongoClient.getDatabase("demo");
            GridFSBucket gridFSBucket = GridFSBuckets[enter image description here][1].create(mongoDatabase);
            ObjectId fileId = new ObjectId("56f25a8b163b4598987b666b");
            GridFSDownloadStream downloadStream = gridFSBucket.openDownloadStream(fileId);
            int fileLength = (int) downloadStream.getGridFSFile().getLength();
            byte[] bytesToWriteTo = new byte[fileLength];
            downloadStream.read(bytesToWriteTo);
            downloadStream.close();
            System.out.println(new String(bytesToWriteTo, StandardCharsets.UTF_8));
        }
    }

The files Collection

The chunks Collection

java mongodb gridfs
3个回答
1
投票

我知道这是一个非常晚的回复,您可以尝试使用以下实现

public static void main(String[] args) {
        MongoClient mongoClient = new MongoClient("127.0.0.1", 27017);
        MongoDatabase mongoDatabase = mongoClient.getDatabase("demo");
        GridFSBucket gridFSBucket = GridFSBuckets.create(mongoDatabase);
        ObjectId fileId = new ObjectId("56f25a8b163b4598987b666b");
        GridFSDownloadStream downloadStream = gridFSBucket.openDownloadStream(fileId);
        ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
        int data = downloadStream.read();
        while (data >= 0) {
            outputStream.write((char) data);
            data = downloadStream.read();
        }
        byte[] bytesToWriteTo = outputStream.toByteArray();
        downloadStream.close();
        System.out.println(new String(bytesToWriteTo, StandardCharsets.UTF_8));
    }
}

0
投票

您可以直接使用输入流。打开的下载流足以获得所有块

GridFSDownloadStream downloadStream = mongo.getGridFSBucket().openDownloadStream(fileId);    
//int fileLength = (int) downloadStream.getGridFSFile().getLength();    
//byte[] bytesToWriteTo = new byte[(fileLength];    
//downloadStream.read(bytesToWriteTo);    
return downloadStream;   

0
投票

这让我难以忍受了一天,GridFS页面上的示例只调用了一次读取,但为了下载所有需要重复调​​用的块。每次调用都会返回从当前块读取的字节数,因此您必须将值相加以跟踪当前位置并将该值传递回read(...)以便将chuck复制到正确的位置在您的字节数组中的位置。这就是我实现它的方式。

public class OpenDownloadStreamDemo {

      public static void main(String[] args) {
        MongoClient mongoClient = new MongoClient("127.0.0.1", 27017);
        MongoDatabase mongoDatabase = mongoClient.getDatabase("demo");
        GridFSBucket gridFSBucket = GridFSBuckets[enter image description here][1].create(mongoDatabase);
        ObjectId fileId = new ObjectId("56f25a8b163b4598987b666b");
        GridFSDownloadStream downloadStream = gridFSBucket.openDownloadStream(fileId);
        int fileLength = (int) downloadStream.getGridFSFile().getLength();
        byte[] bytesToWriteTo = new byte[fileLength];
        int streamDownloadPosition = 0;
        while(streamDownloadPosition != -1) {
            streamDownloadPosition += downloadStream.read(bytesToWriteTo, streamDownloadPosition, fileLength);
        }
        downloadStream.close();
        System.out.println(new String(bytesToWriteTo, StandardCharsets.UTF_8));
    }
}
© www.soinside.com 2019 - 2024. All rights reserved.