CoreAnimation初探(四) —— 深入理解动画时间

之前只是不断的提到时间是个很重要的东西,并没有过多深入探讨,因为我自己理解的也不是很深(→_→)。最近又详细学了下,有一些新的理解。


1.谈谈动画

先撇开技术,想想自己看到“动画”两个字的时候首先想到的是什么?

反正我想到的是死神、火影、海贼王,哈哈😅。当初火影漫画完结的时候真是又想看结局,又想等着看动画。谁知一等几年过去了,也不知道动画现在完结了没。。。
为什么比起漫画我更想看动画呢?很简单啊,漫画是静止的,哪会有动画看着爽快。其实动画就是干这件事的,让静止的画面“动起来”,英文animate的意思就是“赋予...以生命”。再想想“动”,动就是指位置发生了变化,初中物理老师告诉我们,位移等于速度乘以时间。要想动起来肯定要经历时间的(这不废话吗,博尔特跑的再快不给他时间能到终点吗?。。) 所以动画注定与时间难解难分。

早在两万五千年前的石器时代,人类就在洞穴上画了野牛奔跑的动作图,捕捉不同时间牛的动作。而真正推动动画产生的是1824年英国伦敦大学教授皮特·马克·罗葛特发现的视觉暂留现象。利用人的视觉暂留特性,播放一个连续动作的多个瞬间画面,就会造成流畅的视觉变化效果,这就是动画

咳..扯多了。。。回到我们程序员熟悉的画风,怎么实现一个动画呢?最简单的办法,隔个零点零几秒改变一次,不就动起来了吗?这就是基于定时器的动画。我们的目的是获取动画过程的不同时间点来做变化,其实还可以借助屏幕刷新。(记得我们玩游戏时通常会有一个选项“垂直同步”么?如果我们打开垂直同步,就会根据显示器的频率在每次刷新时更新画面。)又扯远了,CoreAnimation提供了一个实现同步刷新的类CADisplayLink,我们可以借助它获取屏幕刷新的信号:

// 便利构造方法生成一个CADisplayLink,并指定target和selector,屏幕刷新时回调
_displk = [CADisplayLink displayLinkWithTarget:self selector:@selector(onDisplay1:)];
// 需要它加入runloop中
[_displk addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];

简单的两行就可以同步屏幕刷新了。首先要记录动画开始时的时间:

_beginTime = CACurrentMediaTime();

这个CACurrentMediaTime()是获取CoreAnimation中的当前时间,后面会细说。然后在onDisplay1:得到时间差计算当前状态,比如我们想实现一个view在1s内从页面上方移动到底端,如果是匀速的:

- (void)onDisplay1:(CADisplayLink *)displk
{
    NSTimeInterval duration = 1;
    CGFloat fromY = 180;
    CGFloat toY = self.view.frame.size.height - 25;
    
    NSTimeInterval showedTime = CACurrentMediaTime() - _beginTime;
    
    CGFloat percent = showedTime/duration;
    
    if (percent > 1)
    {
        percent = 1;
        [displk invalidate];
    }
    
    CGFloat nowX = _testView.center.x;
    CGFloat nowY = fromY + percent*(toY - fromY);

    _testView.center = CGPointMake(nowX, nowY);
}

很简单,当前时间过去了多少,距离就移动多少。需要注意的时,当我们不使用DisplayLink时需要执行[displk invalidate]方法将其从runloop中移除,这里当时间到达1s后就执行invalidate。如果想实现重力的匀加速效果也是一样,只要回顾下物理课上学过的公式能计算出某时刻所处的状态即可,效果如下:



这里由于gif图片的原因显得有些卡顿,实际上是很流畅的。实现一个动效就这么简单,但这些还远远不够。站在开发者的角度,稳定性、性能等都是必须考虑的,我们先从最简单的方面看看CoreAnimation是怎么对时间建模封装的。

2.从概念到代码

刚才那段代码,最关键的就是CACurrentMediaTime()了,通过它我们记录动画开始的时间,在每次屏幕刷新时算出经过了多长时间从而改变view的位置,到达事先指定的持续时间1s时停止。这个函数定义在CABase.h中:

/* Returns the current CoreAnimation absolute time. This is the result of
 * calling mach_absolute_time () and converting the units to seconds. */
/* 返回当前CoreAnimation绝对时间。为mach_absolute_time()转换为秒的结果*/
CA_EXTERN CFTimeInterval CACurrentMediaTime (void)
    __OSX_AVAILABLE_STARTING (__MAC_10_5, __IPHONE_2_0);

绝对时间和相对时间也比较好理解,比如今天是2017年2月25日,在漫漫历史长河中指的就是今天;再比如我大学毕业1年了,这是一个相对时间。为什么要建立这种分层的时间系统,可以再回想下物理课上选择参照物的重要性。先看看官方的介绍:

CAMediaTiming协议由图层和动画来实现,它建立了一个分层的时间系统模型,每个对象描述从父对象时间到本地时间的映射。

从父时间到本地时间的转换有两个阶段:
1.到“本地活动时间(active local time,不知道该怎么翻译,自行理解吧。。)”的转换:包括对象在父时间轴中出现的点以及相对于父对象的速度。
2.到“基本本地时间(basic local time)”的转换:时间模型允许对象重复其基本持续时间多次,并且可选地在重复之前向后播放。

就像我们在做界面布局时,视图的位置都是基于父视图的,这样方便我们构建复杂的界面。建立分层的时间系统也是为了方便实现复杂的动画效果,比如一个在火车上一边行走一边扇扇子的人,人的父对象(或者说参照物)是火车、扇子的父对象是人。从火车开动起可以看成absolute time,人从座位上站起来开始行走起为active local time,而人每扇动一次扇子对应的就是basic local time。(当然我们开发app一般是不会有这么复杂的需求的,如果有干脆拍摄一段好了...)

干说概念可能比较晦涩,我们通过一个简单的例子深入理解一下。下面两张图是网上随处可见的关于timeOffset的例子。



第一张图里通过拖动滑块,改变layer的timeOffset将动画的进度可视化,主要代码如下:

//...
    [_slider addTarget:self action:@selector(onSliderChanged:) forControlEvents:UIControlEventValueChanged];
//...

- (void)onSliderChanged:(id)sender
{
    _testLayer.timeOffset = _slider.value;
}

第二张图是通过timeOffset实现动画的暂停和继续,主要代码如下:

- (void)pauseAnimation
{
    _testLayer.timeOffset = [_testLayer convertTime:CACurrentMediaTime() fromLayer:nil];
    _testLayer.speed = 0;
}

- (void)continueAnimation
{
    _testLayer.beginTime = CACurrentMediaTime() - _testLayer.timeOffset;
    _testLayer.timeOffset = 0;
    _testLayer.speed = 1;
}

看起来很简单的样子,但要搞明白其中的原理却也不容易。可以试着把这两个效果合在一起,既可以滑动滑块控制进程,又可以点击按钮让它继续播放:



如果不明白原理,只是将两个demo写到一起,并不能达到想要的效果。(如果到这里你脑中已经清晰的浮现出如何实现的代码,那就可以不用往下看了~下面这些看法不保证是正确的,您最好只将它当做一个参考)

这里一共涉及到了时间的三个属性:beginTime、speed、timeOffset。我们先看一下官方的解释:

/* The begin time of the object, in relation to its parent object, if
 * applicable. Defaults to 0. 
 * 对象的开始时间(动画开始之前的延迟时间),相对于父对象而言。 */
@property CFTimeInterval beginTime;

/* The rate of the layer. Used to scale parent time to local time, e.g.
 * if rate is 2, local time progresses twice as fast as parent time.
 * Defaults to 1. 
 * 控制动画的执行速度,也是相对于父对象 */
@property float speed;

/* 时间偏移量,下文详细讨论 */
@property CFTimeInterval timeOffset;

父时间tp和本地活动时间(active local time)t之间的关系为:t = (tp - begin) * speed + offset。这个公式暂且放在这里,先来看看第一个滑块的例子,当滑块valueChange的时候,取滑块的值(0到1之间)赋给了timeOffset,显然是将timeOffset作为了一个duration的相对值;而在第二个暂停的例子中,timeOffset用的是绝对时间转换到layer上的值,如果在运行中查看这个值,可以发现它并不是0到duration的相对值(而是一个很大的值)。

这里就有疑问了,timeOffset到底是怎么定义的?上面两个例子的代码(从官方定义上讲)是否都是正确的?

时间和空间最大的区别在于,时间不能被复用 -- 弗斯特梅里克

仔细想想,这个timeOffset不好理解的原因可能就是因为它试图表示已经逝去的时间。使用timeOffSet实现暂停、继续的原理大概如下图所示(凑合看吧。。)



继续时相当于把暂停经历的时间挪到前面让layer接着暂停时的状态继续活动。这么看来第二个例子暂停、继续是没毛病的,timeOffSet在这里是相当于取的layer层面的“绝对时间”。那第一个例子又该怎么解释?

我尝试在滑动滑块时加入动画并把layer的speed设置为0,使用滑块的value按相对值改变timeOffSet,然后让动画继续,惊奇的发现它仍然会从当前位置继续活动!这里确实困扰了我很久,怎么也闹不明白这个timeOffSet在CA里面是怎么运作的。后来想想一开始把滑块改变timeOffSet当做相对值就错了,因为在滑块那个例子中,动画扔到layer上,layer的速度设置为0,动画压根就没有执行,这时改变timeOffSet对动画来讲刚好相当于一个相对值,因此按继续的逻辑它仍会正常往前走。如果这个动画已经执行了一段时间再按照第一个例子的方法改变timeOffSet就不会有效了。

所以,对timeOffSet的理解和使用请参考第二个例子(暂停、继续)和上面那张图,第一个滑块的例子只是刚好表现正常而已,正确的使用滑块可视化动画进程的方法可以参考demo。(这里说一下,第一个例子本意是为了描述animation的timeOffSet,CAAnimation和CALayer都实现了CAMediaTiming协议,都有时间相关的属性,参考前文提到的active local time和basic local time)

3.从设计到实现

上面说的那么多算是我陷入理解误区钻牛角尖的一个过程吧,我觉得比起直接说结论,这个过程更能帮助理解关于动画时间的一些概念。毕竟CA只是人家封装好的一个框架,理解其中面向对象建模的原理才能更好的掌握这门技术,在接触别的动画库甚至自己写相关代码时也能有理论基础,举一反三。

其实任何技术都是这样,再花哨的东西最终都是一堆0和1。就拿动画来讲,首先是对真实世界现象的研究,物理学家和数学家告诉我们它是什么,遵循什么规律;计算机科学家提出的面向对象思想,我们可以用编程语言来对它进行建模;设计师设计出的酷炫的动画效果,从根本上讲也就是这些自然现象的组合,所以我们做的不过是把设计师的思想映射到这些基本原理上。

总结

参考上面那张关于timeOffSet的手图。。。

Demo

参考资料

性能和时间
iOS版动画 - 从不会到熟练应用
iOS开发之动画中的时间
iOS-Core-Animation-Advanced-Techniques.pdf
iOS-Core-Animation-Advanced-Techniques中文翻译

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容