A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 小丫 中级黑马   /  2016-3-19 19:49  /  939 人查看  /  8 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

重用和延迟加载(lazy load) Views

更多的view意味着更多的渲染,也就是更多的CPU和内存消耗,对于那种嵌套了很多view在UIScrollView里边的App更是如此。

这里我们用到的技巧就是模仿`UITableView`和`UICollectionView`的操作: 不要一次创建所有的subview,而是当需要时才创建,当它们完成了使命,把他们放进一个可重用的队列中。

这样的话你就只需要在滚动发生时创建你的views,避免了不划算的内存分配。

创建views的能效问题也适用于你app的其它方面。想象一下一个用户点击一个按钮的时候需要呈现一个view的场景。有两种实现方法:

  • 创建并隐藏这个view当这个screen加载的时候,当需要时显示它;
  • 当需要时才创建并展示。


每个方案都有其优缺点。

用第一种方案的话因为你需要一开始就创建一个view并保持它直到不再使用,这就会更加消耗内存。然而这也会使你的app操作更敏感因为当用户点击按钮的时候它只需要改变一下这个view的可见性。

第二种方案则相反——消耗更少内存,但是会在点击按钮的时候比第一种稍显卡顿。

Cache, Cache, 还是Cache!

一个极好的原则就是,缓存所需要的,也就是那些不大可能改变但是需要经常读取的东西。

我们能缓存些什么呢?一些选项是,远端服务器的响应,图片,甚至计算结果,比如UITableView的行高。

NSURLConnection默认会缓存资源在内存或者存储中根据它所加载的HTTP Headers。你甚至可以手动创建一个NSURLRequest然后使它只加载缓存的值。

下面是一个可用的代码段,你可以可以用它去为一个基本不会改变的图片创建一个NSURLRequest并缓存它:

注意你可以通过 NSURLConnection 获取一个URL request, AFNetworking也一样的。这样你就不必为采用这条tip而改变所有的networking代码了。

如果想了解更多关于HTTP caching, NSURLCache, NSURLConnection的相关知识,可以读下这篇文章。

如果你需要缓存其它不是HTTP Request的东西,你可以用NSCache。

NSCache和NSDictionary类似,不同的是系统回收内存的时候它会自动删掉它的内容。

权衡渲染方法

在iOS中可以有很多方法做出漂亮的按钮。你可以用整幅的图片,可调大小的图片,uozhe可以用CALayer, CoreGraphics甚至OpenGL来画它们。

当然每个不同的解决方法都有不同的复杂程度和相应的性能。有一篇Apple UIKit team中的一员Andy Matuschak推荐过的很棒的关于graphic性能的帖子很值得一读。

简单来说,就是用事先渲染好的图片更快一些,因为如此一来iOS就免去了创建一个图片再画东西上去然后显示在屏幕上的程序。问题是你需要把所有你需要用到的图片放到app的bundle里面,这样就增加了体积 – 这就是使用可变大小的图片更好的地方了: 你可以省去一些不必要的空间,也不需要再为不同的元素(比如按钮)来做不同的图。

然而,使用图片也意味着你失去了使用代码调整图片的机动性,你需要一遍又一遍不断地重做他们,这样就很浪费时间了,而且你如果要做一个动画效果,虽然每幅图只是一些细节的变化你就需要很多的图片造成bundle大小的不断增大。

总得来说,你需要权衡一下利弊,到底是要性能能还是要bundle保持合适的大小。

处理内存警告

一旦系统内存过低,iOS会通知所有运行中app。在官方文档中是这样记述:

如果你的App收到了内存警告,它就需要尽可能释放更多的内存。最佳方式是移除对缓存,图片object和其他一些可以重创建的objects的strong references。

幸运的是,UIKit提供了几种收集低内存警告的方法:

在app delegate中,使用`applicationDidReceiveMemoryWarning:` 的方法,在你的自定义UIViewController的子类(subclass)中覆盖`didReceiveMemoryWarning`,注册并接收UIApplicationDidReceiveMemoryWarningNotification的通知。一旦收到这类通知,你就需要释放任何不必要的内存使用。

例如,UIViewController的默认行为是移除一些不可见的view, 它的一些子类则可以补充这个方法,删掉一些额外的数据结构。一个有图片缓存的app可以移除不在屏幕上显示的图片。

这样对内存警报的处理是很必要的,若不重视,你的app就可能被系统杀掉。

然而,当你一定要确认你所选择的object是可以被重现创建的来释放内存。一定要在开发中用模拟器中的内存提醒模拟去测试一下。

重用大开销对象

一些objects的初始化很慢,比如NSDateFormatter和NSCalendar。然而,你又不可避免地需要使用它们,比如从JSON或者XML中解析数据。

想要避免使用这个对象的瓶颈你就需要重用他们,可以通过添加属性到你的class里或者创建静态变量来实现。

注意如果你要选择第二种方法,对象会在你的app运行时一直存在于内存中,和单例(singleton)很相似。

下面的代码说明了使用一个属性来延迟加载一个date formatter. 第一次调用时它会创建一个新的实例,以后的调用则将返回已经创建的实例:

还需要注意的是,其实设置一个NSDateFormatter的速度差不多是和创建新的一样慢的!所以如果你的app需要经常进行日期格式处理的话,你会从这个方法中得到不小的性能提升。

使用Sprite Sheets

你是一个游戏开发者吗,那么Sprite sheets一定是一个你的最好的朋友了。Sprite sheet可以让渲染速度加快,甚至比标准的屏幕渲染方法节省内存。

我们有两个很好的关于Sprite的教程:

  • How To Use Animations and Sprite Sheets in Cocos2D
  • How to Create and Optimize Sprite Sheets in Cocos2D with Texture Packer and Pixel Formats


第二个教程涵盖了可能在很大程度上影响你游戏性能的pixel格式的细节。

如果你对于spirte sheet还不是很熟悉,可以看下这两个(youtube)视频SpriteSheets – The Movie, Part 1 和 Part 2。视频的作者是创建Sprite sheet很流行的工具之一Texture Packer的作者Andreas Low。

除了使用Sprite sheets,其它写在这里的建议当然也可以用于游戏开发中。比如你需要很多的Sprite sheets,像敌人,导弹之类的动作类必备元素,你可以重用这些sprites而不用每次都要重新创建。

避免反复处理数据

许多应用需要从服务器加载功能所需的常为JSON或者XML格式的数据。在服务器端和客户端使用相同的数据结构很重要。在内存中操作数据使它们满足你的数据结构是开销很大的。

比如你需要数据来展示一个table view,最好直接从服务器取array结构的数据以避免额外的中间数据结构改变。

类似的,如果需要从特定key中取数据,那么就使用键值对的dictionary。




评分

参与人数 5黑马币 +71 收起 理由
海棠 + 18 赞一个!
jcd1216 + 10
13453837658 + 11 很给力!
qqhao01 + 22 很给力!
黑客吴二少 + 10 神马都是浮云

查看全部评分

8 个回复

倒序浏览
顶一个!!!
回复 使用道具 举报
学习了...大神
回复 使用道具 举报
feiyiban 来自手机 中级黑马 2016-3-19 21:28:03
板凳
哟哟切克闹,煎饼果子来一套!!!
回复 使用道具 举报
feiyiban 来自手机 中级黑马 2016-3-19 21:29:14
报纸
好好学习天天向上
回复 使用道具 举报
不能送技术分么
回复 使用道具 举报
cxk 中级黑马 2016-3-19 22:40:28
7#
好厉害的样子哦,看不懂怎么办?
回复 使用道具 举报
赞一个
回复 使用道具 举报
NGUGG 中级黑马 2016-3-20 17:49:59
9#
膜拜大神。。。
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马