Out of memory when encoding file to base64

前端 未结 8 536
自闭症患者
自闭症患者 2020-11-30 03:46

Using Base64 from Apache commons

public byte[] encode(File file) throws FileNotFoundException, IOException {
        byte[] encoded;
        try (FileInputSt         


        
相关标签:
8条回答
  • 2020-11-30 03:55

    In Manifest in applcation tag write following android:largeHeap="true"

    It worked for me

    0 讨论(0)
  • 2020-11-30 03:57

    Java 8 added Base64 methods, so Apache Commons is no longer needed to encode large files.

    public static void encodeFileToBase64(String inputFile, String outputFile) {
        try (OutputStream out = Base64.getEncoder().wrap(new FileOutputStream(outputFile))) {
            Files.copy(Paths.get(inputFile), out);
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }
    
    0 讨论(0)
  • 2020-11-30 03:59

    Either the file is too big, or your heap is too small, or you've got a memory leak.

    • If this only happens with really big files, put something into your code to check the file size and reject files that are unreasonably big.

    • If this happens with small files, increase your heap size by using the -Xmx command line option when you launch the JVM. (If this is in a web container or some other framework, check the documentation on how to do it.)

    • If the file recurs, especially with small files, the chances are that you've got a memory leak.


    The other point that should be made is that your current approach entails holding two complete copies of the file in memory. You should be able to reduce the memory usage, though you'll typically need a stream-based Base64 encoder to do this. (It depends on which flavor of the base64 encoding you are using ...)

    This page describes a stream-based Base64 encoder / decoder library, and includes lnks to some alternatives.

    0 讨论(0)
  • 2020-11-30 04:00

    This is best code to upload image of more size

    bitmap=Bitmap.createScaledBitmap(bitmap, 100, 100, true);
    
    ByteArrayOutputStream stream = new ByteArrayOutputStream();
    bitmap.compress(Bitmap.CompressFormat.PNG, 100, stream); //compress to which format you want.
    byte [] byte_arr = stream.toByteArray();  
    String image_str = Base64.encodeBytes(byte_arr);
    
    0 讨论(0)
  • 2020-11-30 04:01

    Well, looks like your file is too large to keep the multiple copies necessary for an in-memory Base64 encoding in the available heap memory at the same time. Given that this is for a mobile device, it's probably not possible to increase the heap, so you have two options:

    • make the file smaller (much smaller)
    • Do it in a stram-based way so that you're reading from an InputStream one small part of the file at a time, encode it and write it to an OutputStream, without ever keeping the enitre file in memory.
    0 讨论(0)
  • 2020-11-30 04:08

    Well, do not do it for the whole file at once.

    Base64 works on 3 bytes at a time, so you can read your file in batches of "multiple of 3" bytes, encode them and repeat until you finish the file:

    // the base64 encoding - acceptable estimation of encoded size
    StringBuilder sb = new StringBuilder(file.length() / 3 * 4);
    
    FileInputStream fin = null;
    try {
        fin = new FileInputStream("some.file");
        // Max size of buffer
        int bSize = 3 * 512;
        // Buffer
        byte[] buf = new byte[bSize];
        // Actual size of buffer
        int len = 0;
    
        while((len = fin.read(buf)) != -1) {
            byte[] encoded = Base64.encodeBase64(buf);
    
            // Although you might want to write the encoded bytes to another 
            // stream, otherwise you'll run into the same problem again.
            sb.append(new String(buf, 0, len));
        }
    } catch(IOException e) {
        if(null != fin) {
            fin.close();
        }
    }
    
    String base64EncodedFile = sb.toString();
    
    0 讨论(0)
提交回复
热议问题