• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

ios - 应用程序分配 650MB 的 RAM 到 CG 光栅数据加载 ~250 UIImages

[复制链接]
菜鸟教程小白 发表于 2022-12-12 09:14:54 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题

我目前正在开发一个 iPad 应用程序,它将 ~250 个 UIImages 加载到 UIButtons 中(然后更改颜色),创建一个世界地图 - 每个国家有自己的按钮和相应的图像 - 加载游戏时。我遇到的问题是,在视网膜 iPad 上,应用程序在加载图像时使用了大约 650MB 的 RAM,这太疯狂了。

当应用最初加载游戏时,它使用以下代码将图像设置为按钮(TerritoryUIButton 的子类)。

//Initialize the arrays of each territory and add an action to the territory
    for (int i = (int)territoryArray.count - 1; i >= 0; i--) {

        @autoreleasepool {

            //Cast the object into a territory object
            Territory *ter = (Territory *)[territoryArray objectAtIndex:i];

            //Set the territory's image
            [ter setImage:[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:[NSString stringWithFormat"%@T%i.png", [defaults objectForKey"Current_Map"], i + 1] ofType:nil]] forState:UIControlStateNormal];
        }
    }

这会导致以下内存使用情况。峰值在循环运行时出现,但即使 block 被 @autoreleasepool 包围也不会消失。注意 ImageIO 正在使用内存:

Without coloring

该屏幕截图是在领土的图像文件中具有原始颜色时拍摄的。我正在使用 MGImageUtilities 中的 UIImage+Tint 库为图像着色。使用该库时,使用以下代码:

[ter setImage:[[ter imageForState:UIControlStateNormal] imageTintedWithColor:[UIColor colorWithRed:[[colors objectAtIndex:0] floatValue]/255.0 green:[[colors objectAtIndex:1] floatValue]/255.0 blue:[[colors objectAtIndex:2] floatValue]/255.0 alpha:1.0]] forState:UIControlStateNormal];

在加载所有图像后,在另一个循环中使用此代码(在另一个函数中用 @autoreleasepool 括起来)时,会发生以下内存使用情况。请注意,CG raster data 正在使用内存。

With coloring

不知道为什么使用内存的东西不一样。

posted a thread在 Apple Developer Forums 上,这也是该问题的前奏。

我还联系了 Apple Developer TSI,他们建议使用“延迟图像加载”。我对此进行了调查,但我还没有找到在非基于页面的 UIScrollView 上执行此操作的方法。

在这一点上,我几乎不知道如何解决这个问题。经过数小时的努力,我已经尝试了我能想到的一切,但我似乎无法提出任何可行的解决方案。如果有人可以帮助我,我将不胜感激。如果您想查看更多信息或代码,请告诉我,我很乐意发布。提前感谢您的帮助。

编辑:

我一直在试验,我正在使用 scrollViewDidScroll: 中的以下代码。它似乎正在工作,但内存没有被释放,它继续上升。想法?

CGRect currentF = [scrollView convertRect:scrollView.bounds toView:scrollView.contentView];

    //Loop through all territories and determine which need to be rendered
    for (int i = (int)territoryArray.count - 1; i >= 0; i--) {

        @autoreleasepool {

            //See if the territory is on-screen
            if (CGRectIntersectsRect(currentF, [[territoryArray objectAtIndex:i] frame])) {

                //See if the territory needs an image
                if ([[territoryArray objectAtIndex:i] image] == nil) {

                    //Set the territory's image
                    [[territoryArray objectAtIndex:i] setImage:[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:[NSString stringWithFormat"%@T%i.png", [defaults objectForKey"Current_Map"], i + 1] ofType:nil]] forState:UIControlStateNormal];
                }
            } else {
                //Remove the territory's image, it is off-screen
                [[territoryArray objectAtIndex:i] setImage:nil forState:UIControlStateNormal];
            }
        }
    }



Best Answer-推荐答案


我发现,当我必须将图像加载到 ScrollView 中时,为了避免内存问题和应用程序的一般响应能力,您需要逐步加载图像。这意味着,尽管您在 ScrollView 中可能有 250 张图像,但您无法同时看到它们。所以加载他可见的瓷砖,以及它下面的几行。您可以在滚动时加载其余部分。

编辑:这是一个例子。

-(void)scrollViewDidScrollUIScrollView *)myScrollView {

int currentPage = (1 + myScrollView.contentOffset.x / kXItemSpacingIphone);
for (ItemView* itemView in [self.itemRow subviews]){
    if (itemView.tag >= currentPage-1 && itemView.tag <= currentPage+1)
    {
        //keep it visible
        if (!itemView.isLoaded) {
            [itemView layoutWithData:[self.items objectAtIndex:itemView.tag-1]];
        }
    }
    else
    {
        //hide it
        if (itemView.isLoaded) {
            [itemView unloadData];
        }

    }
}

}

上面的代码将加载可见页面上方页面和下方页面的图像。我发现页面滚动有点不稳定。但是我正在从网络加载图像,因此您可能会发现这很好用。

祝你好运!

编辑 2:

如果您的 ScrollView 没有设置为分页,试试这个。

  1. 让我的 View Controller 成为 ScrollView 的代表(如果你在代码中这样做,你必须修改你的 View Controller 的 .h 来说明它符合 UIScrollViewDelegate)。

  2. 定义一个scrollViewDidScroll方法,(a)确定 ScrollView 可见部分的框架; (b) 确定哪些 subview 与该可见部分相交; (c) 加载可见的项目,卸载不可见的项目。

它最终会看起来像这样。

    - (void)scrollViewDidScrollUIScrollView *)scrollView
{
    // Determine the frame of the visible portion of the scrollview.

    CGRect visibleScrollViewFrame = scrollView.bounds;
    visibleScrollViewFrame.origin = scrollView.contentOffset;

    // Now iterate through the various items, remove the ones that are not visible,
    // and show the ones that are.

    for (Item *itemObject in self.itemCollection)
    {
        // Determine the frame within the scrollview that the object does (or 
        // should) occupy.

        CGRect itemObjectFrame = [self getItemObjectFrame:itemObject];

        // see if those two frames intersect

        if (CGRectIntersectsRect(visibleScrollViewFrame, itemObjectFrame))
        {
            // If it's visible, then load it (if it's not already).
            // Personally, I have my object have a boolean property that
            // tells me whether it's loaded or not. You can do this any
            // way you want.

            if (!itemObject.loaded)
                [itemObject loadItem];
        }
        else
        {
            // If not, go ahead and unload it (if it's loaded) to conserve memory.

            if (itemObject.loaded)
                [itemObject unloadItem];
        }
    }
}

Reference

关于ios - 应用程序分配 650MB 的 RAM 到 CG 光栅数据加载 ~250 UIImages,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22952027/

回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关注0

粉丝2

帖子830918

发布主题
阅读排行 更多
广告位

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap