Android SurfaceView运行机制剖析--处理切换到后台再重新进入程序时的异常

有不少朋友都遇到过这种问题,程序执行时切换到后台,然后再重新进入会报异常,本文就这种问题全面讲解下SurfaceView的运行机制,了解了这些原理你就能自己解决这些问题了。

我们通常会通过单击HOME按键或返回按键等操作切换到后台,之后可能会再次进入程序,这个时候就有可能报异常。这里SurfaceView可能报的异常主要有两点,如下:

一、提交画布异常。如下图(模拟器错误提示,以及Logcat Detail)

Java代码

public void draw() {
  try {
    canvas = sfh.lockCanvas();
    if (canvas != null) {
      canvas.drawColor(Color.WHITE);
      canvas.drawBitmap(bmp, bmp_x, bmp_y, paint);
    }
  } catch (Exception e) {
    Log.v("Himi", "draw is Error!");
  } finally {//备注1
    if (canvas != null)//备注2
      sfh.unlockCanvasAndPost(canvas);
  }
} 

先看备注1这里,之前的文章中我给大家解释过为什么要把 sfh.unlockCanvasAndPost(canvas); 写在finally中,主要是为了保证能正常的提交画布。

今天主要说说备注2,这里一定要判定下canvas是否为空,因为当程序切入后台的时候,canvas是获取不到的!那么canvas一旦为空,提交画布这里就会出现参数异常的错误!

   二、线程启动异常。如下图(模拟器错误提示,以及Logcat Detail)

这种异常只是在当你程序运行期间点击Home按钮后再次进入程序的时候报的异常,异常说咱们的线程已经启动!为什么返回按钮就没事?

OK,下面我们就要来先详细讲解一下Android中Back和Home按键的机制!然后分析问题,并且解决问题!

先看下面MySurfaceViewAnimation.java的类中的代码:

Java代码

public class MySurfaceViewAnimation extends SurfaceView implements Callback, Runnable {
  private Thread th;
  private SurfaceHolder sfh;
  private Canvas canvas;
  private Paint paint;
  private Bitmap bmp;
  private int bmp_x, bmp_y;
  public MySurfaceViewAnimation(Context context) {
    super(context);
    this.setKeepScreenOn(true);
    bmp = BitmapFactory.decodeResource(getResources(), R.drawable.himi_dream);
    sfh = this.getHolder();
    sfh.addCallback(this);
    paint = new Paint();
    paint.setAntiAlias(true);
    this.setLongClickable(true);
    th = new Thread(this, "himi_Thread_one");
    Log.e("Himi", "MySurfaceViewAnimation");
  }
  public void surfaceCreated(SurfaceHolder holder) {
    th.start();
    Log.e("Himi", "surfaceCreated");
  }
  public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    Log.e("Himi", "surfaceChanged");
  }
  public void surfaceDestroyed(SurfaceHolder holder) {
    Log.e("Himi", "surfaceDestroyed");
  }
  public void draw() {
    try {
      canvas = sfh.lockCanvas();
      if (canvas != null) {
        canvas.drawColor(Color.WHITE);
        canvas.drawBitmap(bmp, bmp_x, bmp_y, paint);
      }
    } catch (Exception e) {
      Log.v("Himi", "draw is Error!");
    } finally {//备注1
      if (canvas != null)//备注2
        sfh.unlockCanvasAndPost(canvas);
    }
  }
  public void run() {
    while (true) {
      draw();
      try {
        Thread.sleep(100);
      } catch (Exception ex) {
      }
    }
  }
} 

以上是我们常用的自定义SurfaceView,并且使用Runnable接口老框架了不多说了,其中我在本类的构造、创建、状态改变、消亡函数都加上打印!

OK,下面看第一张图:(刚运行程序)

  上图的左边部分是Dubug。这里显示我们有一条线程在运行,名字叫”himi_Thread_one”。

上图的右边部分是LogCat日志。大家很清晰的看到,当第一次进入程序的时候,会先进入view构造函数、然后是创建view,然后是view状态改变,OK,这个大家都知道!

下面是我来点击Home(手机上的小房子)按键,这时程序处于后台,然后重新进入程序的过程!

上图可以看出我们的线程还是一条,这里主要观察从点击home到再次进入程序的过程,如下所述:

点击home 调用了view销毁,然后进入程序会先进入view创建,最后是view状态改变。

上面的过程很容易理解,重要的角色上场了~Back 按钮!点我点击Back按钮看看发生了什么!

先看左边的Debug一栏,多了一条线程! 看LogCat发现比点击Home按键多调用了一次构造函数!

好了,从我们测试的程序来看,无疑,点击Home 和 点击 Back按钮再次进入程序的时候,步骤是不一样的,线程数量也变了!

那么这里就能解释为什么我们点击Back按钮不异常,点击Home会异常了!

原因:因为点击Back按钮再次进入程序的时候先进入的是view构造函数里,那么就是说这里又new了一个线程出来,并启动!那么而我们点击Home却不一样了,因为点击home之后再次进入程序不会进入构造函数,而是直接进入了view创建这个函数,而在view创建这个函数中我们有个启动线程的操作,其实第一次启动程序的线程还在运行,so~这里就一定异常了,说线程已经启动!

有些童鞋会问,我们为何不把th = new Thread(this, “himi_Thread_one”);放在view创建函数中不就好了?!

没错,可以!但是当你反复几次之后你发现你的程序中会多出很多条进程!(如下图)

虽然可以避免出现线程已经启动的异常,很明显这不是我们想要的结果!

那么下面给大家介绍最合适的解决方案:

修改MySurfaceViewAnimation.java:

Java代码

public class MySurfaceViewAnimation extends SurfaceView implements Callback, Runnable {
  private Thread th;
  private SurfaceHolder sfh;
  private Canvas canvas;
  private Paint paint;
  private Bitmap bmp;
  private int bmp_x, bmp_y;
  private boolean himi; //备注1
  public MySurfaceViewAnimation(Context context) {
    super(context);
    this.setKeepScreenOn(true);
    bmp = BitmapFactory.decodeResource(getResources(), R.drawable.himi_dream);
    sfh = this.getHolder();
    sfh.addCallback(this);
    paint = new Paint();
    paint.setAntiAlias(true);
    this.setLongClickable(true);
    Log.e("Himi", "MySurfaceViewAnimation");
  }
  public void surfaceCreated(SurfaceHolder holder) {
    himi = true;
    th = new Thread(this, "himi_Thread_one");//备注2
    th.start();
    Log.e("Himi", "surfaceCreated");
  }
  public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    Log.e("Himi", "surfaceChanged");
  }
  public void surfaceDestroyed(SurfaceHolder holder) {
    himi = false;//备注3
    Log.e("Himi", "surfaceDestroyed");
  }
  public void draw() {
    try {
      canvas = sfh.lockCanvas();
      if (canvas != null) {
        canvas.drawColor(Color.WHITE);
        canvas.drawBitmap(bmp, bmp_x, bmp_y, paint);
      }
    } catch (Exception e) {
      Log.v("Himi", "draw is Error!");
    } finally {
      if (canvas != null)
        sfh.unlockCanvasAndPost(canvas);
    }
  }
  public void run() {
    while (himi) {//备注4
      draw();
      try {
        Thread.sleep(100);
      } catch (Exception ex) {
      }
    }
  }
} 

        这里修改的地方有以下几点:

1、我们都知道一个线程启动后,只要run方法执行结束,线程就销毁了,所以我增加了一个布尔值的成员变量 himi(备注1),这里可以控制我们的线程消亡的一个开关!(备注4)

2、在启动线程之前,设置这个布尔值为ture,让线程一直运行。

3、在view销毁时,设置这个布尔值为false,销毁当前线程!(备注3)

OK,这里图和解释够详细了,希望大家以后真正开发一款游戏的时候,一定要严谨代码,不要留有后患哈~

以上就对Android SurfaceView运行机制详细介绍,后续继续补充相关知识,谢谢大家对本站的支持!

(0)

相关推荐

  • Android中贝塞尔曲线的绘制方法示例代码

    贝塞尔曲线,很多人可能不太了解,什么叫做贝塞尔曲线呢?这里先做一下简单介绍:贝塞尔曲线也可以叫做贝济埃曲线或者贝兹曲线,它由线段与节点组成,节点是可拖动的支点,线段像可伸缩的皮筋.一般的矢量图形软件常利用贝塞尔曲线来精确画出曲线. 上面的介绍中,"线段像可伸缩的皮筋"这句话非常关键,但也特别好理解.至于贝塞尔曲线的详细内容大家可以查阅相关资料.        Android提供的贝塞尔曲线绘制接口 在Android开发中,要实现贝塞尔曲线其实还是很简单的,因为Android已经给我们提

  • Android 模拟信号示波器示例代码

    上次简单地介绍了AudioRecord和AudioTrack的使用,这次就结合SurfaceView实现一个Android版的手机模拟信号示波器.最近物联网炒得很火,作为手机软件开发者,如何在不修改手机硬件电路的前提下实现与第三方传感器结合呢?麦克风就是一个很好的ADC接口,通过麦克风与第三方传感器结合,再在软件里对模拟信号做相应的处理,就可以提供更丰富的传感化应用. 先来看看本文程序运行的效果图(屏幕录像速度较慢,真机实际运行起来会更加流畅): 本文程序使用8000hz的采样率,对X轴方向绘图

  • Android View进行手势识别详解

    我们在进行Android游戏开发时会用到很多种控制,包括前面讲到的按键和轨迹球控制方式,除此之外还有手势操作.重力感应等多种控制方式需要了解掌握.本节主要为大家讲解在View中如何进行手势识别. 很多网友发现Android中手势识别提供了两个类,由于Android 1.6以下的版本比如cupcake中无法使用android.view.GestureDetector,而android.gesture.Gesture是Android 1.6开始支持的,考虑到仍然有使用Android 1.5固件的网友

  • Android 物理游戏之重力系统开发示例代码

    本节为大家提供有关物理游戏的知识,讲解了一个简单的圆形自由落体Demo的编写.本文要介绍的重力系统实际上是类似的. 在重力传感器中,虽然我也实现了一个圆形会根据手机反转的角度而拥有不同的速度,但是其内置加速度算法都是Android os封装好的,而今天我们要讲的重力系统就是去模拟这个加速度,从而让一个自由落体的圆形,感觉跟现实中的皮球一样有质有量!下落的时候速度加快,反弹起来以后速度慢慢减下来. 先贴上两张效果截图,让大家有一个直观的了解,之后再详加讲解: 圆形自由落体Demo简介 当你点击模拟

  • Android Service判断设备联网状态详解

    首先,要想获得当前android设备是否处于联网状态,那么android本身给我们提供了一个服务. private ConnectivityManager connectivityManager;//用于判断是否有网络 connectivityManager = (ConnectivityManager) getActivity().getSystemService(Context.CONNECTIVITY_SERVICE);//获取当前网络的连接服务 NetworkInfo info = co

  • Android中图片的三级缓存机制

    我们不能每次加载图片的时候都让用户从网络上下载,这样不仅浪费流量又会影响用户体验,所以Android中引入了图片的缓存这一操作机制. 原理: 首先根据图片的网络地址在网络上下载图片,将图片先缓存到内存缓存中,缓存到强引用中 也就是LruCache中.如果强引用中空间不足,就会将较早存储的图片对象驱逐到软引用(softReference)中存储,然后将图片缓存到文件(内部存储外部存储)中:读取图片的时候,先读取内存缓存,判断强引用中是否存在图片,如果强引用中存在,则直接读取,如果强引用中不存在,则

  • Android 屏幕双击事件的捕获简单示例

    在Android游戏开发中,我们可能经常要像PC操作一样在屏幕上双击.对于屏幕双击操作,Android 1.6版本以前并没有提供完善的手势识别类,Android 1.5的SDK中提供了android.view.GestureDetector.OnDoubleTapListener,但经测试无法正常工作,不知是何原因.最终我们的解决方案如下面的代码: Java代码 public class TouchLayout extends RelativeLayout { public Handler do

  • Android开发之ScrollView的滑动监听

    我们需要监听ScroView的滑动情况,比如滑动了多少距离,是否滑到布局的顶部或者底部.可惜的是SDK并没有相应的方法,不过倒是提供了一个 protected void onScrollChanged(int l, int t, int oldl, int oldt) 显然这个方法是不能被外界调用的,因此就需要把它暴露出去,解决方式就是写一个接口 /** * Created by 刘楠 on 2016/8/21 0021.17:24 */ public interface ScrollViewL

  • Android 重力传感器在游戏开发中的应用

    手势操作可以说是智能手机的一种魅力所在,前两节给大家讲解了两种有趣的手势操作,将它们置于游戏当中,大大提升了游戏的可玩性和趣味性.本节将继续介绍智能手机的另一种神奇之处:传感器.    一.何为传感器 所谓传感器就是能够探测如光.热.温度.重力.方向等等的装置.    二.Android提供了哪些传感器 1.加速度传感器(重力传感器) 2.陀螺仪传感器 3.光传感器 4.恒定磁场传感器 5.方向传感器 6.恒定的压力传感器 7.接近传感器 8.温度传感器 今天我们给大家介绍的是游戏开发中最最常见

  • Android开发实例之多点触控程序

    智能终端设备的多点触控操作为我们带来了种种炫酷体验,这也使得很多Android开发者都对多点触控程序的开发感兴趣.实际上多点触控程序的实现并不是那么遥不可及,而是比较容易.本文就主要通过一个实例具体讲解多点触控程序的实现.        首先来了解一下Android中多点触控的原理. Android多点触控在本质上需要LCD驱动和程序本身设计上支持,目前市面上HTC.Motorola和Samsung等知名厂商只要使用电容屏触控原理的手机均可以支持多点触控Multitouch技术,对于网页缩放.手

  • Android重力传感器实现滚动的弹球

    熟知: 什么是传感器: 所谓传感器能够探测如光.热.温度.重力.方向 等等的功能! Android中提供传感器有哪些: 1.  加速度传感器(重力传感器)      2.  陀螺仪传感器      3.  光传感器      5.  恒定磁场传感器      6.  方向传感器      7.  恒定的压力传感器      8.  接近传感器      9.  温度传感器 一. 问题描述 Android中有多达11种传感器,不同的手机设备支持的传感器类型也不尽相同 1. 重力传感器 GV-sen

随机推荐