我的应用程序在我的 iPad 上运行。但它的表现非常糟糕——我的速度低于 15fps。谁能帮我优化一下?
它基本上是一个轮子(派生自 UIView),包含 12 个按钮(派生自 UIControl)。
当用户旋转它时,按钮会动态展开和收缩(例如,12 点钟位置的按钮应该始终是最大的)
所以我的轮子包含:
- (void) displayLinkIsCallingBack: (CADisplayLink *) dispLink
{
:
// using CATransaction like this goes from 14fps to 19fps
[CATransaction begin];
[CATransaction setDisableActions: YES];
// NEG, as coord system is flipped/fucked
self.transform = CGAffineTransformMakeRotation(-thetaWheel);
[CATransaction commit];
if (BLA)
[self rotateNotch: direction];
}
…根据最近的触摸输入计算轮子的新旋转。这里已经有一个性能问题,我正在一个单独的线程中解决:iOS Core-Animation: Performance issues with CATransaction / Interpolating transform matrices
此例程还检查滚轮是否完成了另一个 1/12 圈,如果完成则指示所有 12 个按钮调整大小:
// Wheel.m
- (void) rotateNotch: (int) direction
{
for (int i=0; i < [self buttonCount] ; i++)
{
CustomButton * b = (CustomButton *) [self.buttons objectAtIndex: i];
// Note that b.btnSize is a dynamic property which will calculate
// the desired button size based on the button index and the wheels rotation.
[b resize: b.btnSize];
}
}
现在是实际调整大小的代码,在 button.m 中:
// Button.m
- (void) scaleFrom: (float) s_old
to: (float) s_new
time: (float) t
{
CABasicAnimation * scaleAnimation = [CABasicAnimation animationWithKeyPath: @"transform.scale"];
[scaleAnimation setDuration: t ];
[scaleAnimation setFromValue: (id) [NSNumber numberWithDouble: s_old] ];
[scaleAnimation setToValue: (id) [NSNumber numberWithDouble: s_new] ];
[scaleAnimation setTimingFunction: [CAMediaTimingFunction functionWithName: kCAMediaTimingFunctionEaseOut] ];
[scaleAnimation setFillMode: kCAFillModeForwards];
scaleAnimation.removedOnCompletion = NO;
[self.contentsLayer addAnimation: scaleAnimation
forKey: @"transform.scale"];
if (self.displayShadow && self.shadowLayer)
[self.shadowLayer addAnimation: scaleAnimation
forKey: @"transform.scale"];
size = s_new;
}
// - - -
- (void) resize: (float) newSize
{
[self scaleFrom: size
to: newSize
time: 1.];
}
我想知道问题是否与排队的多个 transform.scale 操作的开销有关——每个按钮调整大小需要一整秒才能完成,如果我快速旋转轮子,我可能每秒旋转几圈;这意味着每个按钮每秒调整大小 24 次。
** 创建按钮层 **
我想拼图的最后一 block 是查看按钮的 contentsLayer。但是我试过了
contentsLayer.setRasterize = YES;
应该有效地将其存储为位图。因此通过设置,代码实际上动态调整了 12 个位图的大小。
我无法相信这会超出设备的极限。然而,core animation instrument 告诉我的不是这样;当我旋转方向盘时(通过在圆圈中拖动我的手指),它报告 ~15fps。
这不好:我最终需要在每个按钮内放置一个文本层,这将进一步降低性能(...除非我使用上面的 .setRasterize 设置,在这种情况下它应该是相同的).
一定是我做错了什么!但是什么?
编辑:这里是负责生成按钮内容层(即带阴影的形状)的代码:
- (CALayer *) makeContentsLayer
{
CAShapeLayer * shapeOutline = [CAShapeLayer layer];
shapeOutline.path = self.pOutline;
CALayer * contents = [CALayer layer];
// get the smallest rectangle centred on (0,0) that completely contains the button
CGRect R = CGRectIntegral(CGPathGetPathBoundingBox(self.pOutline));
float xMax = MAX(abs(R.origin.x), abs(R.origin.x+R.size.width));
float yMax = MAX(abs(R.origin.y), abs(R.origin.y+R.size.height));
CGRect S = CGRectMake(-xMax, -yMax, 2*xMax, 2*yMax);
contents.bounds = S;
contents.shouldRasterize = YES; // try NO also
switch (technique)
{
case kMethodMask:
// clip contents layer by outline (outline centered on (0, 0))
contents.backgroundColor = self.clr;
contents.mask = shapeOutline;
break;
case kMethodComposite:
shapeOutline.fillColor = self.clr;
[contents addSublayer: shapeOutline];
self.shapeLayer = shapeOutline;
break;
default:
break;
}
if (NO)
[self markPosition: CGPointZero
onLayer: contents ];
//[self refreshTextLayer];
//[contents addSublayer: self.shapeLayerForText];
return contents;
}
如您所见,我正在尝试所有可能的方法,我正在尝试两种制作形状的方法,并且分别切换 .shouldRasterize
** 妥协 UI 设计以获得可接受的帧率 **
编辑:现在我尝试禁用动态调整大小行为,直到轮子进入新位置,并设置 wheel.setRasterize = YES。所以它有效地在手指下方旋转一个预渲染的 UIView(它确实占据了大部分屏幕)(它很乐意这样做 @~60fps),直到轮子停止,此时它执行这个滞后的调整大小动画( @<>
虽然这给出了一个可以接受的结果,但我不得不以这种方式牺牲我的 UI 设计似乎很疯狂。我确信我一定做错了什么。
编辑:我刚刚尝试手动调整按钮的大小;即在每个按钮中放置一个显示链接回调,并动态计算此给定帧的预期大小,使用 CATransaction 明确禁用动画,就像我对滚轮所做的一样,设置新的转换矩阵(从预期大小生成比例转换)。除此之外,我还设置了按钮内容层 shouldRasterize = YES。所以它应该简单地将每帧 12 个位图缩放到一个本身旋转的 UIView 上。令人惊讶的是,这太慢了,它甚至使模拟器停止运行。这绝对比使用核心动画的动画功能自动执行慢 10 倍。
最佳答案
我没有开发 iPad 应用程序的经验,但我有一些优化视频游戏的经验。所以,我不能给出确切的答案,但我想给出一些优化方面的提示。
Do not guess. Profile it.
您似乎试图在不分析代码的情况下进行更改。更改一些可疑代码并祈祷并没有真正起作用。您应该通过检查每个任务花费的时间 时间以及它们需要运行的频率 来分析您的代码。尝试分解您的任务并放置分析代码以测量时间 和频率。如果您可以测量每个任务使用了多少内存,或者多少其他系统资源,那就更好了。根据证据而不是你的感觉找到你的瓶颈。
对于您的具体问题,您认为当您开始调整大小时程序变得非常慢。但是,您确定吗?它可能是别的东西。在我们拥有实际的分析数据之前,我们无法确定。
Minimize problematic area and measure real performance before making changes.
分析后,您就有了瓶颈候选者。如果您仍然可以将原因分解为几个小任务,那就去做并分析它们,直到您不能再分解为止。然后,通过重复运行它们几千次来尝试测量它们的精确性能。这很重要,因为您需要在进行任何更改之前了解性能(速度和空间),以便您可以将其与 future 的更改进行比较。
对于您的具体问题,如果调整大小确实是问题所在,请尝试检查它的执行方式。执行一次调整大小调用需要多长时间?您需要多久调整一次大小才能完成您的工作?
Improve it. How? It depends.
现在,您有了有问题的代码块及其当前性能。你现在必须改进它。如何?好吧,这真的取决于问题是什么。你可以搜索更好的算法,如果你可以延迟计算直到你真正需要执行,你可以做lazy fetching,或者你可以做over-eager evaluation 如果您过于频繁地执行相同的操作,则通过缓存一些数据。您越了解原因,就越容易改善它。
对于您的具体问题,这可能只是操作系统 ui 功能的限制。通常,调整按钮大小不仅是调整按钮本身的大小,而且还会使整个区域或其父窗口小部件无效,以便每个 ui 内容都可以正确呈现。调整按钮的大小可能很昂贵,如果是这种情况,您可以通过简单地使用基于图像的方法而不是基于 OS ui 的系统来解决问题。如果来自 OS API 的图像操作还不够,您可以尝试使用 OpenGL。但是,同样,直到我们实际试用它们并分析这些替代方案时,我们才知道。祝你好运! :)
关于performance - iOS/核心动画 : Performance tuning,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5003616/
这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下
我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的
Unity自动旋转动画1.开门需要门把手先动,门再动2.关门需要门先动,门把手再动3.中途播放过程中不可以再次进行操作觉得太复杂?查看我的文章开关门简易进阶版效果:如果这个门可以直接打开的话,就不需要放置"门把手"如果门把手还有钥匙需要旋转,那就可以把钥匙放在门把手的"门把手",理论上是可以无限套娃的可调整参数有:角度,反向,轴向,速度运行时点击Test进行测试自己写的代码比较垃圾,命名与结构比较拉,高手轻点喷,新手有类似的需求可以拿去做参考上代码usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;u
1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里
print"Enteryourpassword:"pass=STDIN.noecho(&:gets)puts"Yourpasswordis#{pass}!"输出:Enteryourpassword:input.rb:2:in`':undefinedmethod`noecho'for#>(NoMethodError) 最佳答案 一开始require'io/console'后来的Ruby1.9.3 关于ruby-为什么不能使用类IO的实例方法noecho?,我们在StackOverflow上
我目前对后台队列不太满意。我正在尝试让Resque工作。我已经安装了redis和Resquegem。Redis正在运行。一个worker正在运行(rakeresque:workQUEUE=simple)。使用Web界面,我可以看到工作人员正在运行并等待工作。当我运行“rakeget_updates”时,作业已排队但失败了。我已经用defself.perform和defperform试过了。发条.raketask:get_updates=>:environmentdoResque.enqueue(GetUpdates)end类文件(app/workers/get_updates.rb)c
动画/*INITIALIZEANANIMATION 初始化一个动画*-----------------------*/lv_anim_ta;lv_anim_init(&a);/*MANDATORYSETTINGS 必选设置*------------------*//*Setthe"animator"function 设置“动画”功能*/lv_anim_set_exec_cb(&a,(lv_anim_exec_xcb_t)lv_obj_set_x);/*Setthe"animator"function*/lv_anim_set_var(&a,obj);/*Lengthoftheanim
当我将IO::popen与不存在的命令一起使用时,我在屏幕上打印了一条错误消息:irb>IO.popen"fakefake"#=>#irb>(irb):1:commandnotfound:fakefake有什么方法可以捕获此错误,以便我可以在脚本中进行检查? 最佳答案 是:升级到ruby1.9。如果您在1.9中运行它,则会引发Errno::ENOENT,您将能够拯救它。(编辑)这是在1.8中的一种hackish方式:error=IO.pipe$stderr.reopenerror[1]pipe=IO.popen'qwe'#
我是Ruby分析的新手,看起来像ruby-prof是一个受欢迎的选择。我刚刚安装了gem并调用了我的程序:ruby-prof./my-prog.rb但是,输出非常冗长,因为包含所有Ruby核心和标准库方法以及其他gem的分析数据。例如,前三行是:8.790.0110.0100.0000.0013343*String#%7.280.0780.0090.0000.0692068*Array#each4.930.0380.0060.0000.0321098*Array#map这对我来说不是什么有用的信息,因为我已经知道我的程序经常处理字符串和数组,并且大概已经对这些类进行了优化。我只关心我代
当我尝试使用“套接字”库中的方法“read_nonblock”时出现以下错误IO::EAGAINWaitReadable:Resourcetemporarilyunavailable-readwouldblock但是当我通过终端上的IRB尝试时它工作正常如何让它读取缓冲区? 最佳答案 IgetthefollowingerrorwhenItrytousethemethod"read_nonblock"fromthe"socket"library当缓冲区中的数据未准备好时,这是预期的行为。由于异常IO::EAGAINWaitReadab