Недостаточно памяти, когда UrlEncodedFormEntity с изображением - PullRequest
2 голосов
/ 20 декабря 2011

Я пытаюсь отправить данные на сервер.То есть некоторые строковые поля и изображение, которое закодировано как base64 и передано в виде строки следующим образом:

ByteArrayOutputStream stream = new ByteArrayOutputStream();
        mBitmap.compress(Bitmap.CompressFormat.JPEG, 100, stream);
        byte[] byteArray = stream.toByteArray();
        //Cleaning memory
        try {
            stream.close();
            stream = null;
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        //Eecode base64
        nameValuePairs.add(new BasicNameValuePair(DatosDB.KEY_IMG, Base64.encodeToString(byteArray, Base64.DEFAULT)));
        byteArray = null;

И затем, при подготовке запроса HTTP PUT:

 try{
        HttpClient httpclient = new DefaultHttpClient();
        //PUT
        HttpPut httpput = new HttpPut(KEY_121 + ruta);
        //The exception is thrown when executing next instruction
        httpput.setEntity(new UrlEncodedFormEntity(nameValuePairs));
        HttpResponse response = httpclient.execute(httpput);

        HttpEntity entity = response.getEntity();
        is = entity.getContent();

        }catch(Exception e){
        Log.e(TAG, "Error in http connection "+e.toString());
        }   

Ошибка выглядит следующим образом:

12-20 00:24:11.622: ERROR/AndroidRuntime(7499): FATAL EXCEPTION: main
12-20 00:24:11.622: ERROR/AndroidRuntime(7499): java.lang.OutOfMemoryError
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at java.lang.AbstractStringBuilder.enlargeBuffer(AbstractStringBuilder.java:95)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at java.lang.AbstractStringBuilder.append0(AbstractStringBuilder.java:140)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at java.lang.StringBuilder.append(StringBuilder.java:125)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at java.net.URLEncoder.encode(URLEncoder.java:109)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at org.apache.http.client.utils.URLEncodedUtils.encode(URLEncodedUtils.java:184)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at org.apache.http.client.utils.URLEncodedUtils.format(URLEncodedUtils.java:163)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at org.apache.http.client.entity.UrlEncodedFormEntity.<init>(UrlEncodedFormEntity.java:71)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.upvar.Http_Request.put(Http_Request.java:171)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.upvar.Http_Request.poi(Http_Request.java:108)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.upvar.NewPOI.putPOI(NewPOI.java:360)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.upvar.NewPOI.access$6(NewPOI.java:325)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.upvar.NewPOI$5.onClick(NewPOI.java:157)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at android.view.View.performClick(View.java:2538)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at android.view.View$PerformClick.run(View.java:9152)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at android.os.Handler.handleCallback(Handler.java:587)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at android.os.Handler.dispatchMessage(Handler.java:92)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at android.os.Looper.loop(Looper.java:130)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at android.app.ActivityThread.main(ActivityThread.java:3687)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at java.lang.reflect.Method.invokeNative(Native Method)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at java.lang.reflect.Method.invoke(Method.java:507)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:842)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:600)
12-20 00:24:11.622: ERROR/AndroidRuntime(7499):     at dalvik.system.NativeStart.main(Native Method)
12-20 00:24:11.632: WARN/ActivityManager(3923):   Force finishing activity com.android.upvar/.NewPOI
12-20 00:24:11.636: ERROR/(3923): Dumpstate > /data/log/dumpstate_app_error

Я не понимаю, почему я получаю это исключение.Я просто отправляю обычное изображение, которое было снято с камеры Samsung Galaxay (наверняка менее 16 МБ).Любая идея?

РЕДАКТИРОВАТЬ: я должен добавить, что первый кусок кода и второй находятся на разных классах, поэтому данные на первом фрагменте кода передается объекту другого класса, который передаетэто к другому объекту другого класса, у которого есть запрос HTTP.

Ответы [ 3 ]

1 голос
/ 20 декабря 2011

Используйте потоковую передачу для вашей сущности, чтобы она использовала только небольшой размер буфера, который запрашивают потоки.ByteArrayOutputStream содержит все данные в памяти.

HTTPClient - Руководство по производительности

Попробуйте реализацию, подобную приведенной ниже, которая не создает ненужных объектов памяти.Следующее может не быть полной реализацией.Веселого кодирования!

class BitMapRequestEntity extends AbstractHttpEntity {

    private Bitmap bitmap;

    public BitMapRequestEntity(Bitmap mBitmap) {
        super();
        this.bitmap = mBitmap;
    }

    @Override
    public InputStream getContent() throws IOException,
            IllegalStateException {
        // TODO Auto-generated method stub
        return null;
    }

    @Override
    public long getContentLength() {
        // TODO Auto-generated method stub
        return 0;
    }

    @Override
    public boolean isRepeatable() {
        // TODO Auto-generated method stub
        return false;
    }

    @Override
    public boolean isStreaming() {
        return true;
    }

    @Override
    public void writeTo(OutputStream outstream) throws IOException {
        bitmap.compress(Bitmap.CompressFormat.JPEG, 100, outstream);
    }

}
try {
    org.apache.http.client.HttpClient httpclient = new DefaultHttpClient();
    // PUT
    HttpPut httpput = new HttpPut(KEY_121 + ruta);
    // The exception is thrown when executing next instruction
    httpput.setEntity(new BitMapRequestEntity(mBitmap));
    HttpResponse response = httpclient.execute(httpput);

    HttpEntity entity = response.getEntity();
    is = entity.getContent();

} catch (Exception e) {
    Log.e(TAG, "Error in http connection " + e.toString());
}
1 голос
/ 02 января 2012

В итоге, решение было сжать изображение до 50% следующим образом:

mBitmap.compress(Bitmap.CompressFormat.JPEG, 100, stream);

Это уменьшает размер изображения и делает его достаточно маленьким, чтобы отправить его в виде строки.

0 голосов
/ 20 декабря 2011

После того, как вы сделаете

mBitmap.compress(Bitmap.CompressFormat.JPEG, 100, stream);
byte[] byteArray = stream.toByteArray();

у вас есть две версии изображения в памяти, одна в виде растрового изображения и одна в байтовом массиве. Очистка первого с помощью Bitmap.recycle() может помочь.

Не обратился за советом: не служат ли «составные данные формы» для передачи изображений вашей цели? Это очень эффективно для памяти.

...