当前位置: 首页 > 知识库问答 >
问题:

Google Appengine JAVA-Zip大量图片保存在Blobstore中

司徒炎彬
2023-03-14

开发了一个简单的媒体库,你可以在其中选择一组图像并下载它们。当客户机请求下载时,servlet会接收用于创建zip文件的blob密钥,然后为该过程启动一个任务

该任务遍历接收到的blob密钥,并将图像压缩到存档中。任务完成后,将向用户发送带有下载链接的邮件。

这就是我的问题:

FileWriteChannel writeChannel = fileService.openWriteChannel(file, lock);
OutputStream blobOutputStream = Channels.newOutputStream(writeChannel);
ZipOutputStream zip = new ZipOutputStream(blobOutputStream);   

单个通道只能处理这个字节数

BlobstoreService。MAX_BLOB_FETCH_尺寸

正因为如此,我必须打开和关闭通道每1mb的数据,我必须写(同样的问题,读,但我使用了这个代码,它的工作原理).

打开和关闭通道与一个正常的输出Stream,不提出问题,像这个代码

但是处理一个Zip文件我也必须管理

ZipOutputStream zip = new ZipOutputStream(blobOutputStream);   
ZipEntry zipEntry = new ZipEntry(image_file_name);
zipOut.putNextEntry(zipEntry);
// while the image has bytes to write
   zipOut.write(bytesToWrite);

在我写了1MB的数据在ZipEntry我必须关闭通道,并再次打开它。

所以这里的问题是:如果我打开一个新频道,我就无法访问我之前写的zipEntry,然后我就无法继续写下我正在处理的下一个1MB的图像。

并且,在打开一个新的通道后,如果我试图在zipEntry对象上写入(w/o重新初始化它),我得到一个ClosedChannel异常

下面是我编写的示例代码,我知道它不起作用,但解释了我正在尝试做的事情。

那么我的问题是:如何(如果可能的话,当然)我可以创建一个每次写入1MB的zip文件?

我也可以使用其他方法,我需要的是将一些图像压缩到一个压缩包中,并将其保存到blobstore中,如果您有其他想法,请告诉我

共有1个答案

慕容雅珺
2023-03-14

您应该创建自己的可以操作通道的流。当达到Blob大小限制时,您的流将关闭当前通道并打开一个新通道。

本地文件的示例:

public class ZipChannels {
public static void main(String[] args) throws IOException {
    File dirToZip = new File("target\\dependency");

    //create zip-files
    ChannelOutput out = new ChannelOutput();
    ZipOutputStream zip = new ZipOutputStream(out);
    int b = 0;
    for(File file: dirToZip.listFiles()) {
        ZipEntry zipEntry = new ZipEntry(file.getName());
        zip.putNextEntry(zipEntry);
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream(file));
        while((b = bis.read()) != -1) {
            zip.write(b);
        }
        bis.close();
        zip.closeEntry();
    }
    zip.close();

    //merge all into one file for check it
    BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("package_all.zip"));
    for (int i = 0; i < out.getChannelCount(); i++) {
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("package_" + i + ".zip"));
        while((b = bis.read()) != -1) {
            bos.write(b);
        }
        bis.close();
    }
    bos.close();
}

public static class ChannelOutput extends OutputStream {
    private OutputStream channel;
    private int count = 0;
    final private int MAX = 1000000;

    @Override
    public void write(int b) throws IOException {
        if(count++ % MAX == 0) {
            openNewChannel();
        }
        channel.write(b);
    }

    protected void openNewChannel() throws IOException {
        if(channel != null) {
            channel.close();
        }
        channel = new BufferedOutputStream(new FileOutputStream("package_" + (count / MAX) + ".zip")); 
    }
    public int getChannelCount() {
        return count / MAX + 1;
    }

    @Override
    public void close() throws IOException {
        channel.close();
    }
    @Override
    public void flush() throws IOException {
        channel.flush();
    }
}
}

如果您有任何问题,请随时提问。

 类似资料:
  • 图片保存 将需要的内容保存在独立的文件里便于之后的导出。(存储于 Web 所用格式 Alt + Shift + Ctrl + S) 如需保存独立图层则要把需要的图层拖到新建的透明背景的图层,或在图层上右键复制(Duplicate)图层选择地址为新文件即可。 图片与背景合并的切图方法如下 保存格式的选择 保存类型一:色彩丰富切无透明要求时保存为 JPG 格式并选用时候的品质(通常使用品质 80 )。

  • 如何保存flat16(https://en.wikipedia.org/wiki/Half-precision_floating-point_format)最大数量在flat32(https://en.wikipedia.org/wiki/Single-precision_floating-point_format)格式? 我想有一个可以将0x7bff转换为65504的函数。0x7bff是可以用浮

  • 函数名称:保存图片 函数功能: 将图片对象转成图片 函数方法 image.save(img,path,quality) 参数 类型 必填 说明 img userdata 是 图片对象 path string 是 要保存的图片路径和文件名,保存的路径必须已存在,不存在不会自动创建 quality userdata 否 图片质量,仅对 jpg 格式有效,参数范围:0.1 - 1 函数用例 path =

  • 图片下载和存取的简单demo。实现同步和异步两种下载方式。并且将下载好的图片保存在Document,然后进行读取显示。 [Code4App.com]

  • 问题内容: 我的应用程序使用了很多图片,这些图片是从Internet下载并在Android手机上本地缓存的。我想知道,保存这些图片的正确方法是什么。我认为有几种方法不能完全令人满意。 将它们保存在公共文件夹中的SD卡上 占用了无法在卸载时释放的空间 用户可以在图库中查看照片 在sdcard根目录上需要一个文件夹(您实际上可以在浏览手机时看到它) 将它们保存在非公共文件夹中的SD卡上 占用了无法在卸

  • 当我运行progressdialog出现在屏幕上时,它突然停止运行,并发出通知“应用程序停止了,再试一次”,然后我在android manitor上重复了这个错误,当然我首先要做的是: