android之SurfaceView

SurfaceView是视图(View)的继承类,这个视图里内嵌了一个专门用于绘制的Surface。你可以控制这个Surface的格式和尺寸。Surfaceview控制这个Surface的绘制位置。

 

1,它的特性是:可以在主线程之外的线程中向屏幕绘图上。由于它拥有独立的绘图表面,因此SurfaceView的UI就可以在一个独立的线程中进行绘制。又由于不会占用主线程资源,SurfaceView一方面可以实现复杂而高效的UI,另一方面又不会导致用户输入得不到及时响应。普通的Android控件,例如TextView、Button和CheckBox等,它们都是将自己的UI绘制在宿主窗口的绘图表面之上,这意味着它们 的UI是在应用程序的主线程中进行绘制的。由于应用程序的主线程除了要绘制UI之外,还需要及时地响应用户输入,否则的话,系统就会认为应用程序没有响应 了,因此就会弹出一个ANR对话框出来。对于一些游戏画面,或者摄像头预览、视频播放来说,它们的UI都比较复杂,而且要求能够进行高效的绘制,因此,它 们的UI就不适合在应用程序的主线程中进行绘制。这时候就必须要给那些需要复杂而高效UI的视图生成一个独立的绘图表面,以及使用一个独立的线程来绘制这 些视图的UI。

 

(注意:Surface对应了一块屏幕缓冲区,每个window对应一个Surface,任何View都要画在Surface的Canvas上(后面有原因解释)。传统的view共享一块屏幕缓冲区,所有的绘制必须在UI线程中进行。)

 

2、实现原理:

我们假设在一个Activity窗口的视图结构中,除了有一个DecorView顶层视图之外,还有两个TextView控件,以及一个 SurfaceView视图,这样该Activity窗口在SurfaceFlinger服务中就对应有两个Layer或者一个Layer的一个 LayerBuffer,如图1所示:

 

 在图1中,Activity窗口的顶层视图DecorView及其两个TextView控件的UI都是绘制在SurfaceFlinger服务中的同一 个Layer上面的,而SurfaceViewUI是绘制在SurfaceFlinger服务中的另外一个Layer或者LayerBuffer上的。

 (注意:用来描述SurfaceViewLayer或者LayerBufferZ轴位置是小于用来其宿主Activity窗口的LayerZ轴位置 的,但是前者会在后者的上面挖一个出来,以便它的UI可以对用户可见。实际上,SurfaceView在其宿主Activity窗口上所挖的只不过是在其宿主Activity窗口上设置了一块透明区域。

 

需要重写的方法

 (1)public void surfaceChanged(SurfaceHolder holder,int format,int width,int height){}

     //在surface的大小发生改变时激发

 (2)public void surfaceCreated(SurfaceHolder holder){}

     //在创建时激发,一般在这里调用画图的线程。

 (3)public void surfaceDestroyed(SurfaceHolder holder) {}

     //销毁时激发,一般在这里将画图的线程停止、释放。

整 个过程:继承SurfaceView并实现SurfaceHolder.Callback接口 ----> SurfaceView.getHolder()获得SurfaceHolder对象 ---->SurfaceHolder.addCallback(callback) 添加回调函数---->SurfaceHolder.lockCanvas()获得Canvas对象并锁定画布----> Canvas绘画 ---->SurfaceHolder.unlockCanvasAndPost(Canvas canvas)结束锁定画图,并提交改变,将图形显示。

 

3、SurfaceHolder
这里用到了一个类SurfaceHolder,可以把它当成surface的控制器,用来操纵surface。处理它的Canvas上画的效果和动画,控制表面,大小,像素等。
几个需要注意的方法:
(1)、abstract void addCallback(SurfaceHolder.Callback callback);
// 给SurfaceView当前的持有者一个回调对象。
(2)、abstract Canvas lockCanvas();
// 锁定画布,一般在锁定后就可以通过其返回的画布对象Canvas,在其上面画图等操作了。
(3)、abstract Canvas lockCanvas(Rect dirty);
// 锁定画布的某个区域进行画图等..因为画完图后,会调用下面的unlockCanvasAndPost来改变显示内容。
// 相对部分内存要求比较高的游戏来说,可以不用重画dirty外的其它区域的像素,可以提高速度。
(4)、abstract void unlockCanvasAndPost(Canvas canvas);
// 结束锁定画图,并提交改变。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在 AndroidSurfaceView 中实现倒计时功能,您需要创建一个线程,该线程在 SurfaceView 上绘制计时器并更新计时器的值。以下是一个简单的示例,您可以参考: ```java public class CountdownSurfaceView extends SurfaceView implements SurfaceHolder.Callback { private SurfaceHolder mHolder; private CountdownThread mThread; private int mCountdownValue = 10; // 倒计时的初始值 public CountdownSurfaceView(Context context, AttributeSet attrs) { super(context, attrs); mHolder = getHolder(); mHolder.addCallback(this); } @Override public void surfaceCreated(SurfaceHolder holder) { mThread = new CountdownThread(mHolder, mCountdownValue); mThread.start(); } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { } @Override public void surfaceDestroyed(SurfaceHolder holder) { boolean retry = true; mThread.setRunning(false); while (retry) { try { mThread.join(); retry = false; } catch (InterruptedException e) { // 重试 } } } private class CountdownThread extends Thread { private SurfaceHolder mThreadHolder; private int mCountdownValue; private boolean mIsRunning = true; public CountdownThread(SurfaceHolder holder, int countdownValue) { mThreadHolder = holder; mCountdownValue = countdownValue; } public void setRunning(boolean isRunning) { mIsRunning = isRunning; } @Override public void run() { while (mIsRunning) { Canvas canvas = null; try { canvas = mThreadHolder.lockCanvas(); synchronized (mThreadHolder) { drawCountdown(canvas); Thread.sleep(1000); // 每秒钟更新一次计时器 mCountdownValue--; if (mCountdownValue == 0) { break; // 计时器结束 } } } catch (InterruptedException e) { // 重试 } finally { if (canvas != null) { mThreadHolder.unlockCanvasAndPost(canvas); } } } } private void drawCountdown(Canvas canvas) { Paint paint = new Paint(); paint.setColor(Color.WHITE); paint.setTextSize(100); canvas.drawColor(Color.BLACK); canvas.drawText(String.valueOf(mCountdownValue), getWidth() / 2, getHeight() / 2, paint); } } } ``` 在这个例子中,我们创建了一个 CountdownSurfaceView 类,它继承自 SurfaceView,并实现了 SurfaceHolder.Callback 接口。在 surfaceCreated() 方法中,我们创建了一个 CountdownThread 线程,并启动该线程。在 CountdownThread 线程中,我们使用 Canvas 绘制计时器,并使用 Thread.sleep() 方法使线程休眠一秒钟。每次线程醒来时,我们更新计时器的值,并检查计时器是否结束。如果计时器结束,我们退出线程。最后,在 surfaceDestroyed() 方法中,我们停止线程并等待线程结束。 请注意,在这个例子中,我们将计时器的值绘制在 SurfaceView 的中心。您可以根据自己的需要更改这个位置。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值