Фатальное исключение PICASSO: java.lang.OutOfMemoryError: не удалось выделить JNI Env

#java #android #out-of-memory #picasso

#java #Android #нехватка памяти #picasso

Вопрос:

У меня все время была эта большая проблема с picasso. У меня получилось так

 Fatal Exception: java.lang.OutOfMemoryError: Could not allocate JNI Env
   at java.lang.Thread.nativeCreate(Thread.java)
   at java.lang.Thread.start(Thread.java:729)
   at com.squareup.picasso.Picasso.(SourceFile:205)
   at com.squareup.picasso.Picasso$Builder.build(Picasso.java:864)
   at com.destina.travling.ui.main.fragment.home.restaurant.detail.FoodImageAdapter.onBindViewHolder(FoodImageAdapter.java:58)
   at com.destina.travling.ui.main.fragment.home.restaurant.detail.FoodImageAdapter.onBindViewHolder(FoodImageAdapter.java:27)
   at androidx.recyclerview.widget.RecyclerView$Adapter.onBindViewHolder(RecyclerView.java:6781)
   at androidx.recyclerview.widget.RecyclerView$Adapter.onBindViewHolder(RecyclerView.java:6781)
   at androidx.recyclerview.widget.RecyclerView$Adapter.getItemCount(RecyclerView.java:6781)
   at androidx.recyclerview.widget.RecyclerView$Adapter.onCreateViewHolder(RecyclerView.java:6823)
   at androidx.recyclerview.widget.RecyclerView$Adapter.bindViewHolder(RecyclerView.java:6823)
   at androidx.recyclerview.widget.RecyclerView$Recycler.tryBindViewHolderByDeadline(RecyclerView.java:5752)
   at androidx.recyclerview.widget.RecyclerView$Recycler.tryGetViewHolderForPositionByDeadline(RecyclerView.java:6019)
   at androidx.recyclerview.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:5858)
   at androidx.recyclerview.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:5854)
   at androidx.recyclerview.widget.LinearLayoutManager$LayoutState.next(LinearLayoutManager.java:2230)
   at androidx.recyclerview.widget.LinearLayoutManager.layoutChunk(LinearLayoutManager.java:1557)
   at androidx.recyclerview.widget.LinearLayoutManager.fill(LinearLayoutManager.java:1517)
   at androidx.recyclerview.widget.LinearLayoutManager.onLayoutChildren(LinearLayoutManager.java:612)
   at androidx.recyclerview.widget.RecyclerView.dispatchLayoutStep2(RecyclerView.java:3924)
   at androidx.recyclerview.widget.RecyclerView.onMeasure(RecyclerView.java:3336)
   at android.view.View.measure(View.java:19886)
   at androidx.constraintlayout.widget.ConstraintLayout.internalMeasureChildren(ConstraintLayout.java:1248)
   at androidx.constraintlayout.widget.ConstraintLayout.onMeasure(ConstraintLayout.java:1593)
   at android.view.View.measure(View.java:19886)
   at androidx.core.widget.NestedScrollView.measureChildWithMargins(NestedScrollView.java:1502)
   at android.widget.FrameLayout.onMeasure(FrameLayout.java:185)
   at androidx.core.widget.NestedScrollView.onMeasure(NestedScrollView.java:556)
   at android.view.View.measure(View.java:19886)
   at androidx.constraintlayout.widget.ConstraintLayout.internalMeasureChildren(ConstraintLayout.java:1248)
   at androidx.constraintlayout.widget.ConstraintLayout.onMeasure(ConstraintLayout.java:1593)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6085)
   at androidx.coordinatorlayout.widget.CoordinatorLayout.onMeasureChild(CoordinatorLayout.java:733)
   at androidx.coordinatorlayout.widget.CoordinatorLayout.onMeasure(CoordinatorLayout.java:805)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6085)
   at android.widget.FrameLayout.onMeasure(FrameLayout.java:185)
   at androidx.appcompat.widget.ContentFrameLayout.onMeasure(ContentFrameLayout.java:143)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6085)
   at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1464)
   at android.widget.LinearLayout.measureVertical(LinearLayout.java:758)
   at android.widget.LinearLayout.onMeasure(LinearLayout.java:640)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6085)
   at android.widget.FrameLayout.onMeasure(FrameLayout.java:185)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6085)
   at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1464)
   at android.widget.LinearLayout.measureVertical(LinearLayout.java:758)
   at android.widget.LinearLayout.onMeasure(LinearLayout.java:640)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:6085)
   at android.widget.FrameLayout.onMeasure(FrameLayout.java:185)
   at com.android.internal.policy.DecorView.onMeasure(DecorView.java:693)
   at android.view.View.measure(View.java:19886)
   at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:2323)
   at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1414)
   at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1667)
   at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1302)
   at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6441)
   at android.view.Choreographer$CallbackRecord.run(Choreographer.java:876)
   at android.view.Choreographer.doCallbacks(Choreographer.java:688)
   at android.view.Choreographer.doFrame(Choreographer.java:623)
   at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:862)
   at android.os.Handler.handleCallback(Handler.java:754)
   at android.os.Handler.dispatchMessage(Handler.java:95)
   at android.os.Looper.loop(Looper.java:163)
   at android.app.ActivityThread.main(ActivityThread.java:6238)
   at java.lang.reflect.Method.invoke(Method.java)
   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:904)
   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:794)
  

Я просто изменил настройку picasso следующим образом :

 OkHttpClient client = HttpsClient.getHttpsClient(context);
        Picasso picasso = new Picasso.Builder(context)
                .downloader(new OkHttp3Downloader(client))
                .memoryCache(new LruCache(getBytesForMem()))
                .build();
        picasso.load(link)
                .fit()
                .memoryPolicy(MemoryPolicy.NO_CACHE, MemoryPolicy.NO_STORE)
                .networkPolicy(NetworkPolicy.NO_CACHE, NetworkPolicy.NO_STORE)
                .placeholder(R.drawable.ic_broken_image)
                .error(R.drawable.ic_broken_image)
                .into(holder.ivFoodImage);

private int getBytesForMem() {
        ActivityManager.MemoryInfo memoryInfo = new ActivityManager.MemoryInfo();
        ActivityManager activityManager = (ActivityManager)
                context.getSystemService(ACTIVITY_SERVICE);
        if (activityManager != null) {
            activityManager.getMemoryInfo(memoryInfo);
        }

        double availableMemory = memoryInfo.availMem;
        return (int) (12 * availableMemory / 100);
    }
  

И я использую версию библиотеки piccasso com.squareup.picasso:picasso:2.71828

Раньше я помещал кэш памяти, тогда я часто получал эту ошибку, но сейчас кошмар снова приходит.

И мой вопрос

  1. Как заставить Picasso отлично работать без OutOfMemoryError ?
  2. Может Picasso.singleton прояснить эту проблему?

Я знаю, что некоторые пользователи, возможно, рекомендуют попробовать другую библиотеку, но я все еще использую library, потому что мой API использует самозаверяющий сертификат для получения API. И по этой причине вы можете видеть, как OkHttp объединяется с Picasso.

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

Я сдамся -голосуйте за любой вопрос, спасибо :).

Комментарии:

1. ваш код здесь неполон, но в соответствии с вашей ошибкой я рекомендую один раз создать объект Picasso в вашем конструкторе адаптера и останавливать воссоздание этого объекта в каждом цикле привязки RecyclerView.

2. можете ли вы привести мне пример кода, который вы имеете в виду?

3. I just using standard code of picasso . НЕПРАВИЛЬНО. Это не стандартный код. Вы изменили поведение кэширования и клиентской сети.

4. Хорошо, я был отредактирован. Можете ли вы удалить downvote 🙂

Ответ №1:

это упрощенный ответ на ваш запрос в комментариях, но я сомневаюсь, что он работает или нет для вас:

 public class TestAdapter extends RecyclerView.Adapter {

List<Object> items;
Picasso picasso;
Context context;
public TestAdapter(List<Object> inputs, Context in_context) {
    items = inputs;
    context = in_context;
    OkHttpClient client = HttpsClient.getHttpsClient(context);
    picasso = new Picasso.Builder(context)
            .downloader(new OkHttp3Downloader(client))
            .memoryCache(new LruCache(getBytesForMem()))
            .build();

}

@NonNull
@Override
public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
    //return your ViewHolder;
}

@Override
public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) {
    // some codes here

    picasso.load(link)
            .fit()
            .memoryPolicy(MemoryPolicy.NO_CACHE, MemoryPolicy.NO_STORE)
            .networkPolicy(NetworkPolicy.NO_CACHE, NetworkPolicy.NO_STORE)
            .placeholder(R.drawable.ic_broken_image)
            .error(R.drawable.ic_broken_image)
            .into(holder.ivFoodImage);
}

@Override
public int getItemCount() {
    return items.size();
}

private int getBytesForMem() {
    ActivityManager.MemoryInfo memoryInfo = new ActivityManager.MemoryInfo();
    ActivityManager activityManager = (ActivityManager)
            context.getSystemService(ACTIVITY_SERVICE);
    if (activityManager != null) {
        activityManager.getMemoryInfo(memoryInfo);
    }

    double availableMemory = memoryInfo.availMem;
    return (int) (12 * availableMemory / 100);
}
}
  

в вашей деятельности создайте свой адаптер RecyclerView, используя новый конструктор.

 //get your recyclerView from your layout as mRecyclerView
LinearLayoutManager manager = new LinearLayoutManager(this);
mRecyclerView.setLayoutManager(manager);
TestAdapter adapter = new TestAdapter(your_data,this);
mRecyclerView.setAdapter(adapter);
  

Комментарии:

1. можете ли вы назвать мне причину, по которой мы должны поместить picasso на общедоступный адаптер?

2. потому что это эффективно для памяти, и это сработало для меня с аналогичной проблемой.

3. и что я делаю в onDestroy?

4. для чего? этот объект является локальным в вашем адаптере (сделайте его приватным). очистка мусора сделает свою работу.

Ответ №2:

Вы можете использовать Picasso, но решить знаменитую ошибку нехватки памяти,
дело в том, что у вас заканчивается память, и это может произойти и с любой другой библиотекой

Иногда изображения большого размера вызывают такого рода проблемы
, иногда установка большой кучи в манифесте может решить проблему , …
Существует огромное количество данных в stackoverflow при нехватке памяти и Picasso

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

Не забывайте также контролировать кучу и использовать функции кучи

Комментарии:

1. Я установил largeheap, но это не работает, тогда каждое изображение фактически загружает более 100 КБ для миниатюр и обложки. Это проблема?

2. Обычно 100 кб не считается большим изображением, и мега-изображения обычно могут вызывать такую проблему, но вам также следует учитывать выделенную кучу памяти на вашем устройстве, кстати, я помню, что у меня когда-то тоже была проблема с Builder, есть ли какая-либо логическая причина, по которой вы просто не используете URL-адрес изображения в Picasso?

3. Хм, возможно, изменение размера — отличный выбор, в моем случае JSON, если нет изображения, значение равно «-» и значение, когда есть ссылка на изображение без https, поэтому я помещаю строку с http: // link

4. Я имею в виду, разве вы не можете забыть Picasso picasso = new Picasso.Builder(контекст) и okhttp и просто использовать Picasso.with(c).load(url). заполнитель (R.drawable.ic_launcher).ошибка(R.drawable.error).into( … , о URL, иногда нет необходимости вводить дополнительные данные, такие как http или даже название сайта в JSON, и вы можете определить их как постоянные или переменные в приложении Android

5. Как я уже сказал, моему JSON нужен сертификат для доступа к API, без которого мы не можем использовать API так же, как изображение