即使在回收Bitmap

时间:2016-04-30 13:39:27

标签: java android out-of-memory picasso

我已经阅读了很多有关此主题的主题,并试图在我的代码中使用它们,但我仍然无法解决它。

我想做什么: 像Facebook或任何其他从服务器下载包含图像的Feed的应用程序,我也是这样做,并试图在图像视图上显示提要和图像。

发生了什么: 我能够下载包含图像URL的feed(用JSON),我可以在ImageView上显示它们。 在Emulator中,我的MAX堆大小为64MB。我在前10个Feed中消耗了大约30MB(不知道为什么,但这是我从Android Studio的Android监视器和甚至Android设备监视器中的Memory选项卡获得的)。 我的应用程序中有一个刷新按钮,在删除之前填充的所有Feed后重新加载相同的Feed。我预计我将消耗相同的内存或更多。但与此相反,我的内存使用量增加到42MB。因此,在点击刷新3到4次后,它会导致OutOFMemory Execption。即使我一次加载下一个10个Feed或50个Feed,我也会收到OutOfMemory Exception。

我知道facebook Instagram和更多此类应用程序做同样的事情,但不确定他们如何实现代码来覆盖这种情况。

以下是我填充Feed的代码

private void loadFeed(List<Feed> feedList)
{
    Log.v(Constant.TAG,"Loading Feed in social feed page");
    for(final Feed feed:feedList) {
        LinearLayout feedBox = new LinearLayout(this);
        feedBox.setOrientation(LinearLayout.VERTICAL);
        FrameLayout profileDetailContainer= new FrameLayout(this);
        LinearLayout profileDetailContainerParent=new LinearLayout(this);
        LinearLayout profileDetailContainerChild=new LinearLayout(this);
        profileDetailContainerChild.setOrientation(LinearLayout.VERTICAL);
        ImageView imgProfile= new ImageView(this);
        TextView txtDate= new TextView(this);
        TextView txtName= new TextView(this);
        ImageView imgProduct= new ImageView(this);
        txtName.setText(feed.getUserFullName());
        TextView txtDesciption= new TextView(this);
        txtName.setTypeface(null, Typeface.BOLD);
        if(feed.getDescription().length()>Constant.NUMBER_OF_DESCRIPTION_CHAR_SHOW_ON_RESULT)
        {
            txtDesciption.setText(feed.getDescription().substring(0,Constant.NUMBER_OF_DESCRIPTION_CHAR_SHOW_ON_RESULT)+"...");
        }
        else
            txtDesciption.setText(feed.getDescription());

        if(!IOUtil.fileExists(this,feed.getProductImageName())) {
            WebRequest request = new WebRequest();
            request.setUrl(Constant.ROOT_APPLICATION_URL_WITH_SEPARATOR + feed.getImgPath());
            request.setParam(feed.getId() + "");
            new ImageDownloadTask(this, true, feed.getProductImageName(), this).execute(request);
            Log.v(Constant.TAG,"URL:"+Constant.ROOT_APPLICATION_URL_WITH_SEPARATOR+feed.getImgPath());
            Picasso.with(getApplicationContext()).load(R.drawable.logo).into(imgProduct);
            feedBox.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    ScreenUtility.alertException(v.getContext(),"Please wait untill product image loads");
                }
            });
            PixyfiSession.save(feed.getId() + "", imgProduct);
            PixyfiSession.save(feed.getId() + "_feed", feed);

        }
        else
        {
            ImageUtil.recycleIfPossible(imgProduct);
            try {
                imgProduct.setImageBitmap(ImageUtil.getLocalImage(feed.getProductImageName(),this));
                FeedboxOnClickListener feedboxOnClickListener = new FeedboxOnClickListener(feed);
                feedBox.setOnClickListener(feedboxOnClickListener);
            } catch (FileNotFoundException e) {
                Log.v(Constant.TAG,e.getMessage(),e);
            }

        }

        if(FacebookUtil.localProfilePicExists(feed.getUserName(),this))
        {
            Bitmap profileImage= FacebookUtil.getLocalProfilePicture(feed.getUserName(),this);
            imgProfile.setImageBitmap(profileImage);
        }
        else {
            FacebookUtil.loadProfilePicture(feed.getUserName(),this,this);
            PixyfiSession.save(feed.getUserName(),imgProfile);
            imgProfile.setImageResource(R.drawable.profile);
        }
        try {

            if(feed.getDate()==null) {
                txtDate.setText(new SimpleDateFormat("dd MMM yyyy").format(new SimpleDateFormat("yyyy-MM-dd").parse(feed.getFeedDate())));
            }
            else {
                txtDate.setText(new SimpleDateFormat("dd MMM yyyy").format(feed.getDate()));
            }
        } catch (ParseException e) {
            Log.e(Constant.TAG,e.getMessage(),e);
        }
        LinearLayout.LayoutParams layoutParam= new LinearLayout.LayoutParams(140,140);
        layoutParam.setMargins(5,5,0,0);
        imgProfile.setLayoutParams(layoutParam);

        layoutParam= new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT,LinearLayout.LayoutParams.WRAP_CONTENT,Gravity.TOP|Gravity.LEFT);
        layoutParam.setMargins(20,5,0,0);
        txtName.setLayoutParams(layoutParam);
        layoutParam= new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT,LinearLayout.LayoutParams.WRAP_CONTENT,Gravity.LEFT);
        layoutParam.setMargins(20,5,0,0);
        txtDate.setLayoutParams(layoutParam);

        profileDetailContainerParent.addView(imgProfile);
        profileDetailContainerChild.addView(txtName);
        profileDetailContainerChild.addView(txtDate);
        profileDetailContainerParent.addView(profileDetailContainerChild);
        feedBox.addView(profileDetailContainerParent);
        LinearLayout.LayoutParams feedLayoutParam=new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.WRAP_CONTENT,Gravity.CENTER_VERTICAL);
        feedLayoutParam.setMargins(5,5,5,5);
        imgProduct.setLayoutParams(feedLayoutParam);
        txtDesciption.setLayoutParams(feedLayoutParam);
        feedBox.addView(txtDesciption);
        feedBox.addView(imgProduct);
        feedLayoutParam=new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.WRAP_CONTENT,Gravity.CENTER_VERTICAL);
        feedLayoutParam.setMargins(0,5,0,5);
        feedBox.setLayoutParams(feedLayoutParam);

        feedBox.setBackgroundColor(Color.parseColor("#cccccc"));
        PixyfiSession.save(feed.getId()+"_feedbox",feedBox);
        imgProfile.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                PixyfiSession.save(Constant.SELECTED_USER_ID,feed.getUserName());
                ScreenUtility.loadScreen(v.getContext(),UsersProfile.class,false);
            }
        });
        this.feedContainer.addView(feedBox);

    }
    if(feedList.size()==Constant.FEED_SIZE_IN_ONE_REQUEST)
    {
        Button moreFeed= new Button(this);
        moreFeed.setText("Load MOre Feed");
        moreFeed.setOnClickListener(new MoreFeedButtonListener(this));
        this.feedContainer.addView(moreFeed);
    }
    this.progressBar.setVisibility(View.INVISIBLE);
}

用于重新加载/刷新Feed

this.currentPage=1;
this.recycleImages();
this.feedContainer.removeAllViews();
PixyfiSession.save(Constant.CURRENT_FEED_PAGE,this.currentPage);
this.progressBar.setVisibility(View.VISIBLE);
loadFeed();

recycleImages方法:

private void recycleImages()
{
    for(int i=0;i<this.feedContainer.getChildCount();i++)
    {
        if(this.feedContainer.getChildAt(i) instanceof  ImageView)
        {
            ImageView view=(ImageView)this.feedContainer.getChildAt(i);
            ImageUtil.recycleIfPossible(view);
        }
    }
}

如果您需要有关代码的更多详细信息,请告知我们。 还有可能在Android设备监视器中看到其他应用程序如facebook的内存使用情况吗? 的更新 ImageUtil.getLocalImage方法

public static Bitmap getLocalImage(String imageName, Context context) throws FileNotFoundException
{
    Bitmap bitmap=null;
    InputStream is=null;
    if(IOUtil.fileExists(context,imageName)) {
        is = context.openFileInput(imageName);
        bitmap= BitmapFactory.decodeStream(is);
    }
    else {
        throw new FileNotFoundException("Image file doesn't exists");
    }
    return bitmap;
}

5 个答案:

答案 0 :(得分:0)

正如Saber Safavi所说的那样先做,然后添加

 defaultConfig {
    ....
    multiDexEnabled true
    ....
}

中的

  

应用程序/的build.gradle

文件..

答案 1 :(得分:0)

看起来您在加载图片时遇到问题。请检查您的ImageUtils.getLocalImages,解码并将缩放版本的图像加载到内存中,还是将原始图像加载到内存中?

关注this document from Android developer,我认为这是在Android中高效加载图片的最佳教程。

答案 2 :(得分:0)

Android中的图像处理非常棘手。更好地利用建立的图书馆来处理所涉及的情况和复杂性。 我更喜欢 UniversalImageLoader或Piccaso!

答案 3 :(得分:0)

这是因为您在运行时内存中保留了太多位图图像,尽量不要创建太多图像并尝试使它们无效。

移动设备通常具有受限制的系统资源。 Android设备可以为单个应用程序提供少至16MB的内存。虚拟机兼容性为各种屏幕尺寸和密度提供所需的最小应用程序内存。应优化应用程序以在此最小内存限制下执行。但是,请记住,许多设备配置了更高的限制。

我不会建议您使用android manifest中的heapSize标志。我希望你不是在开发游戏:),如果你愿意,可以去改造或任何其他图像处理库。我建议你看一下Google提供的BitmapFun示例。它可以在开发人员门户网站上找到。

http://developer.android.com/training/displaying-bitmaps/index.html

干杯!!

答案 4 :(得分:0)

我正在补充答案:

您应该始终更喜欢listview或更好的回收站视图,而不是自己管理视图组件。 因为它可以帮助您避免视图泄漏并更好地重用已创建的视图。