在某个项目中,我需要一个动画。代码如下
ObjectAnimator animator = new ObjectAnimator();
float begin = layoutShare.getTranslationY();
animator.setFloatValues(begin, 0);
animator.setDuration(300);
animator.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationEnd(Animator animation) {
layoutMask.setVisibility(View.VISIBLE);
}
});
animator.addUpdateListener(animation -> {
layoutShare.setTranslationY((Float) animation.getAnimatedValue());
});
animator.start();
在我的手机中运行完美。但是客户反映三星A7 会发生崩溃现象。日志如下
06-26 18:28:55.184: E/AndroidRuntime(12220): java.lang.NullPointerException
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.animation.PropertyValuesHolder.setupSetterAndGetter(PropertyValuesHolder.java:505)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.animation.ObjectAnimator.initAnimation(ObjectAnimator.java:487)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.animation.ValueAnimator.setCurrentPlayTime(ValueAnimator.java:517)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.animation.ValueAnimator.start(ValueAnimator.java:936)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.animation.ValueAnimator.start(ValueAnimator.java:946)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.animation.ObjectAnimator.start(ObjectAnimator.java:465)
06-26 18:28:55.184: E/AndroidRuntime(12220): at com.trs.myrb.activity.CustomShareActivity.lambda$beginAnimator$3$CustomShareActivity(CustomShareActivity.java:104)
06-26 18:28:55.184: E/AndroidRuntime(12220): at com.trs.myrb.activity.CustomShareActivity$$Lambda$2.run(Unknown Source)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.os.Handler.handleCallback(Handler.java:733)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.os.Handler.dispatchMessage(Handler.java:95)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.os.Looper.loop(Looper.java:146)
06-26 18:28:55.184: E/AndroidRuntime(12220): at android.app.ActivityThread.main(ActivityThread.java:5756)
06-26 18:28:55.184: E/AndroidRuntime(12220): at java.lang.reflect.Method.invokeNative(Native Method)
06-26 18:28:55.184: E/AndroidRuntime(12220): at java.lang.reflect.Method.invoke(Method.java:515)
06-26 18:28:55.184: E/AndroidRuntime(12220): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1291)
06-26 18:28:55.184: E/AndroidRuntime(12220): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1107)
06-26 18:28:55.184: E/AndroidRuntime(12220): at dalvik.system.NativeStart.main(Native Method)
通过源码定位到Android 4.4.4 中PropertyValuesHolder的 505行
问题就出现在这个target ,这是ObjectAnimator 特有的。ObjectAnimator是ValueAnimator的子类,他本身就已经包含了时间引擎和值计算,所以它拥有为对象的某个属性设置动画的功能。这使得为任何对象设置动画更加的容易。你不再需要实现 ValueAnimator.AnimatorUpdateListener接口,因为ObjectAnimator动画自己会自动更新相应的属性值。
ObjectAnimator的实例和ValueAnimator是类似的,但是你需要描叙该对象,需要设置动画的属性的名字(一个字符串),以及动画属性值的变化范围:
ObjectAnimator anim = ObjectAnimator.ofFloat(foo, “alpha”, 0f, 1f);
anim.setDuration(1000);
anim.start();
而在我的代码中,我是通过AnimatorUpdateListener自己更新的属性。所以没有设置target。也不需要设置target。于是我把ObjectAnimator 换成 ValueAnimator 就不能在Android4.4.4 中崩溃了。
至于为什么在我的手机上没有崩溃(Android 8.0) 看源码就知道了。添加了对空的检查。
通过这件事我明白,写代码脑子一定要清晰。对每一个类的认识要深刻。否则的话。也许效果出来了,但那不是你写的好。而是别人代码比较健壮。最后的坑还是得自己填。