iOS - iPhone 4S 上的动画速度较慢(但在模拟器上很好)
我使用以下代码对 UILabel 进行了子类化,该代码工作正常 - 但涉及子类的任何动画的运行速度都比正常的 UILabel 慢得多。我假设 Quartz 是罪魁祸首,但是我能做些什么来加快速度吗?
- (void)drawTextInRect:(CGRect)rect
{
CGSize shadowOffset = self.shadowOffset;
UIColor *textColor = self.textColor;
// Establish the Quartz 2D drawing destination:
CGContextRef context = UIGraphicsGetCurrentContext();
CGContextSetLineWidth(context, 1);
CGContextSetLineJoin(context, kCGLineJoinRound);
// Draw the label’s outline:
CGContextSetTextDrawingMode(context, kCGTextStroke);
self.textColor = [UIColor whiteColor];
[super drawTextInRect:rect];
// Draw the label:
CGContextSetTextDrawingMode(context, kCGTextFill);
self.textColor = [UIColor textColor];
self.shadowOffset = CGSizeMake(0, 0);
[super drawTextInRect:rect];
self.shadowOffset = shadowOffset;
}
I have subclassed a UILabel with the following code, which works fine - but any animations involving the subclass runs a lot slower than normal UILabels. I'm assuming Quartz is the culprit, but is there anything I could do to speed things up a bit?
- (void)drawTextInRect:(CGRect)rect
{
CGSize shadowOffset = self.shadowOffset;
UIColor *textColor = self.textColor;
// Establish the Quartz 2D drawing destination:
CGContextRef context = UIGraphicsGetCurrentContext();
CGContextSetLineWidth(context, 1);
CGContextSetLineJoin(context, kCGLineJoinRound);
// Draw the label’s outline:
CGContextSetTextDrawingMode(context, kCGTextStroke);
self.textColor = [UIColor whiteColor];
[super drawTextInRect:rect];
// Draw the label:
CGContextSetTextDrawingMode(context, kCGTextFill);
self.textColor = [UIColor textColor];
self.shadowOffset = CGSizeMake(0, 0);
[super drawTextInRect:rect];
self.shadowOffset = shadowOffset;
}
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
@MobileOverlord所说的当然适用,尤其是关于分析的部分。
我要指出的是,设置 shouldRasterize=YES 并不是一个包罗万象的解决方案(如果是这种情况,为什么 Apple 不将其设置为默认值?)。是的,它可以提高滚动性能,但它会以内存使用为代价,因为最终可能会在缓存中留下一堆大图像。
我相信(但必须检查以确保),它还会在创建时产生开销,包括离屏渲染通道以实际创建光栅化副本。根据层的使用方式,这实际上可能会损害性能。
另一个需要考虑的因素是您的观点是否具有透明度。如果您可以向框架保证您的视图是不透明的(参见 setOpaque/isOpaque),他们可以通过不考虑与 alpha 通道等相关的所有复杂性来优化渲染。类似的考虑也适用于 CALayer。
最后,在您展示的代码块之外,您是否对支持层做了任何偷偷摸摸的事情(例如设置阴影或角半径)?这也是扼杀动画性能的一种快速方法。
What @MobileOverlord said is certainly applicable, especially the parts about about profiling.
I will note that setting shouldRasterize=YES is not a catch-all solution (why wouldn't Apple have set it on as default if that were the case?). Yes, it can improve scrolling performance, but it can do so at the expense of memory use since you can end up with a bunch of large images sitting around in cache.
It also incurs overhead at the time of creation, I believe (but would have to check to be sure) including an off-screen rendering pass to actually create the rasterized copy. Depending on how the layer is used, this could actually hurt performance.
An additional factor to consider is whether your view has any transparency. If you can guarantee to the frameworks that your view is opaque (cf. setOpaque/isOpaque), they can optimize rendering by not considering all the complexities associated with alpha channels, etc. Similar considerations apply to CALayer.
Finally, outside the block of code you showed, did you do anything sneaky to the backing layer (e.g. set a shadow or corner radius)? That's a quick way to kill performance on animation too.
完成标签绘制后,您可以在其图层上调用
shouldRasterize
,它应该会加快动画速度。来自 CALayer 类参考
模拟器总是会给您带来比设备更好的结果,因为它能够使用系统的全部处理能力和内存。通过这种方式,您通常会得到有缺陷的结果。每当您结合 CoreAnimation 进行 CoreGraphics 绘图时,在真实设备上测试结果非常重要。
为此,您可以尝试在 Instruments Core Animation Tool 中运行您的应用程序,以尝试找到罪魁祸首。查看我的教程。
Instruments – 优化核心动画
After you are finished drawing your label you can call
shouldRasterize
on it's layer and it should speed up your animation.From CALayer Class Reference
The simulator is always going to give you way better results than a device will because it is able to use the full processing power and memory of your system. You'll usually get flawed results this way. Whenever you are doing CoreGraphics drawing in conjunction with CoreAnimation it is important to test the results on a real device.
For this you can try to run your app in Instruments Core Animation Tool to try to find culprits. Check out my tutorial on it.
Instruments – Optimizing Core Animation