Android Animation实现元素在屏幕上按照指定轨迹运动,以及出现NullPointerException的解决方案

在Android项目中,通过在DecorView添加FrameLayout并设置ImageView动画,实现了元素按指定轨迹移动。然而,在Android 4.0.x系统中,连续执行动画可能导致NullPointerException。解决方案是修改动画结束时的代码,避免在onAnimationEnd回调中直接移除View。问题可能源于多个动画执行时的回调时机冲突。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

因项目需要,在Android中实现了一个动画,当在Activity中点击特定按钮时,会在屏幕上添加一个ImageView,并按照指定的起点、终点,沿着特定的轨迹运动(例如直线)。

实现方法

实现思路是在Activity的DecorView中添加一个FrameLayout,然后在FrameLayout中添加ImageView,可通过Margin参数指定ImageView的起始位置。然后设置ImageView的动画,使其能运动到终点。当动画结束后,移除FrameLayout。

核心代码如下。项目GitHub链接 https://github.com/jzj1993/AnimationCrash

 
 
  1. public static boolean startAnim(Activity activity, int fromX, int toX, int fromY, int toY) {
  2. try {
  3. final ImageView img = new ImageView(activity);
  4. img.setImageResource(R.mipmap.ic_launcher);
  5. final FrameLayout tempLayout = new FrameLayout(activity);
  6. final FrameLayout.LayoutParams lp = new FrameLayout.LayoutParams(
  7. FrameLayout.LayoutParams.WRAP_CONTENT, FrameLayout.LayoutParams.WRAP_CONTENT);
  8. lp.setMargins(fromX, fromY, 0, 0);
  9. tempLayout.addView(img, lp);
  10. final ViewGroup container = (ViewGroup) activity.getWindow().getDecorView();
  11. container.addView(tempLayout, new ViewGroup.LayoutParams(
  12. ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));
  13. final Animation anim = new TranslateAnimation(0, toX - fromX, 0, toY - fromY);
  14. anim.setDuration(500);
  15. anim.setAnimationListener(new Animation.AnimationListener() {
  16. @Override
  17. public void onAnimationStart(Animation animation) {
  18. }
  19. @Override
  20. public void onAnimationEnd(Animation animation) {
  21. container.removeView(tempLayout);
  22. }
  23. @Override
  24. public void onAnimationRepeat(Animation animation) {
  25. }
  26. });
  27. img.startAnimation(anim);
  28. } catch (Exception e) {
  29. e.printStackTrace();
  30. }
  31. return true;
  32. }

错误描述

在大部分安卓手机上运行正常,但是在部分Android 4.0.x系统中(例如Nexus S Android 4.0.3模拟器),如果连续同时执行多个动画,可能会出现NullPointerException,如下:

 
 
  1. com.jzj1993.anim E/AndroidRuntime FATAL EXCEPTION: main
  2. java.lang.NullPointerException
  3. at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2488)
  4. at android.view.View.draw(View.java:10981)
  5. at android.widget.FrameLayout.draw(FrameLayout.java:450)
  6. at com.android.internal.policy.impl.PhoneWindow$DecorView.draw(PhoneWindow.java:2126)
  7.             at android.view.ViewRootImpl.draw(ViewRootImpl.java:2026)
  8.             at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1634)
  9.             at android.view.ViewRootImpl.handleMessage(ViewRootImpl.java:2442)
  10.             at android.os.Handler.dispatchMessage(Handler.java:99)
  11.             at android.os.Looper.loop(Looper.java:137)
  12.             at android.app.ActivityThread.main(ActivityThread.java:4424)
  13.             at java.lang.reflect.Method.invokeNative(Native Method)
  14.             at java.lang.reflect.Method.invoke(Method.java:511)
  15.             at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
  16. at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
  17. at dalvik.system.NativeStart.main(Native Method)

解决方法

将以下代码

 
 
  1. @Override
  2. public void onAnimationEnd(Animation animation) {
  3. container.removeView(tempLayout);
  4. }

改为:

 
 
  1. @Override
  2. public void onAnimationEnd(Animation animation) {
  3. container.post(new Runnable() {
  4. @Override
  5. public void run() {
  6. container.removeView(tempLayout);
  7. }
  8. });
  9. }

原因分析

以下是官方给出的ViewGroup.removeView方法的JavaDoc注释:

 
 
  1. public void removeView (View view)
  2. Added in API level 1
  3. Note: do not invoke this method from draw(android.graphics.Canvas), onDraw(android.graphics.Canvas), dispatchDraw(android.graphics.Canvas) or any related method.

猜测在同时执行多个动画时,前一个动画的onAnimationEnd被回调时,后一个动画正在执行,因此DecorView的onDraw方法在执行,可能会导致DecorView.onDraw执行时,onAnimationEnd回调中的DecorView.removeView方法被调用,从而导致错误。没有具体深入的去分析源码,但是经过验证问题确实解决了。

`NullPointerException` (空指针异常)在Android Studio中通常是由于尝试访问或操作null对象而产生的。这种错误通常发生在以下几个场景: 1. **初始化变量**: 如果你在创建对象之前没有给它赋予初始值,然后试图立即调用其方法或属性,可能会抛出NullPointerException。 ```java MyObject obj = null; // 没有赋值 obj.someMethod(); // 这里会报错,因为obj是null ``` 2. **回调函数**: 网络请求、数据库查询等操作完成后返回的数据可能是null,你需要检查结果是否为空再处理。 3. **获取组件**: 在试图操作UI元素前,需要确保该元素已经被正确地添加到布局中。 解决步骤一般包括: 1. **检查变量**: 确保你在使用对象之前已经进行了正确的初始化,如`obj = new MyObject()`。 2. **条件判断**: 使用if语句检查对象是否为null,再进行后续操作。 ```java if (obj != null) { obj.someMethod(); } ``` 3. **使用Optional类**: Java 8引入了Optional类,可以避免null pointer异常,提供了一个优雅的方式来表示可能存在也可能不存在的值。 4. **使用空安全的方法**: 当API支持时,尽量使用空安全的版本,比如`findViewById(int id)`代替`findViewById()`。 5. **异常处理**: 可以使用try-catch块捕获并处理`NullPointerException`,以便提供更好的错误反馈。 如果你能提供具体的异常堆栈信息,定位到引发异常的具体代码行将更有帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值