文档章节

surfaceView和View的区别

丁佳辉
 丁佳辉
发布于 02/18 17:12
字数 937
阅读 6
收藏 0

概念:view在UI线程去更新自己;而SurfaceView则在一个子线程中去更新自己

  surfaceView是在一个新起的单独线程中可以重新绘制画面,而View必须在UI的主线程中更新画面

  在UI的主线程中更新动画,时间一旦太长就会出现问题

  surfaceView 在新的线程中更新画面所以不会阻塞你的UI主线程,但是涉及到线程同步,需要surfaceView中 thread处理,一般就需要有一个event queue的设计来保存touch event

 

  触摸产生的动画用view,比如打消消乐

  一直在动的动画用surfaceView,比如有一款跑步的app里面的效果

 

1.创建SurfaceView,需要创建一个新的扩展了SurfaceView的类,并实现SurfaceHolder.Callback

2.需要重写的方法

(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);
// 结束锁定画图,并提交改变。
4、实例

这里的例子实现了一个矩形和一个计时器

package xl.test; 
import android.app.Activity;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.os.Bundle;
import android.view.SurfaceHolder;import android.view.SurfaceView;
 public class ViewTest extends Activity {/** Called when the activity is first created. */@Override
      public void onCreate(Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);
         setContentView(new MyView(this));
      }
     //视图内部类
      class MyView extends SurfaceView implements SurfaceHolder.Callback
     {
          private SurfaceHolder holder;
         private MyThread myThread; 
         public MyView(Context context) {
             super(context);
             // TODO Auto-generated constructor stub
             holder = this.getHolder();
             holder.addCallback(this);
             myThread = new MyThread(holder);//创建一个绘图线程
         }
  
          @Override
         public void surfaceChanged(SurfaceHolder holder, int format, int width,
                  int height) {
             // TODO Auto-generated method stub
               
         }
  
         @Override
         public void surfaceCreated(SurfaceHolder holder) {
              // TODO Auto-generated method stub
             myThread.isRun = true;
             myThread.start();
        }
  
        @Override
         public void surfaceDestroyed(SurfaceHolder holder) {
            // TODO Auto-generated method stub
            myThread.isRun = false;
        }
         
      }
      //线程内部类
    class MyThread extends Thread
     {
          private SurfaceHolder holder;
          public boolean isRun ;
         public  MyThread(SurfaceHolder holder)
          {
              this.holder =holder; 
              isRun = true;
          }
          @Override
          public void run()
          {
              int count = 0;
              while(isRun)
              {
                  Canvas c = null;
                  try
                  {
                      synchronized (holder){
                      c = holder.lockCanvas();//锁定画布,一般在锁定后就可以通过其返回的画布对象Canvas,在其上面画图等操作了。
            c.drawColor(Color.BLACK);//设置画布背景颜色
            Paint p = new Paint(); //创建画笔
            p.setColor(Color.WHITE);
            Rect r = new Rect(100, 50, 300, 250);
            c.drawRect(r, p);
            c.drawText("这是第"+(count++)+"秒", 100, 310, p);
            Thread.sleep(1000);//睡眠时间为1秒
            }
        }
        catch (Exception e) {
            // TODO: handle exception
            e.printStackTrace();
        }
        finally
        {
             if(c!= null)
        {
         holder.unlockCanvasAndPost(c);//结束锁定画图,并提交改变。
  

本文转载自:https://www.cnblogs.com/dubo-/p/6638094.html

共有 人打赏支持
丁佳辉
粉丝 20
博文 434
码字总数 198765
作品 0
浦东
程序员
私信 提问
View 和 SurfaceView的区别

这里说的是在绘图中两者的区别: 1View在绘图中,重写onDraw(Canvas canvas)方法,通过invaldate()和pastInvalidate()两个方法进行重新绘制画布; invalidate()不能再自己创建的线程中循环调用...

宇宙执政
2014/07/25
0
0
Android之SurfaceView(六)

关于surfaceView相关知识: View和SurfaceView主要区别: View只能在UI线程中刷新,而SurfaceView可以在子线程中刷新 SurfaceView可以控制刷新频率 SurfaceView几个重要的方法 继承SurfaceVi...

徐荣
2016/01/27
136
0
SurfaceView和View区别

surfaceView是在一个新起的单独线程中可以重新绘制画面,而View必须在UI的主线程中更新画面。那么在UI的主线程中更新画面 可能会引发问题,比如你更新画面的时间过长,那么你的主UI线程会被你...

xyh12344
2015/11/27
35
0
surfaceView和View最本质的区别

surfaceView和View最本质的区别在于: surfaceView是在一个新起的单独线程中可以重新绘制画面,而View必须在UI的主线程中更新画面。那么在UI的主线程中更新画面 可能会引发问题,比如你更新画...

lightUp
2015/05/26
0
0
surfaceView和View最本质的区别

surfaceView和View最本质的区别在于:(转自:http://www.cnblogs.com/lipeil/archive/2012/08/31/2666187.html) surfaceView是在一个新起的单独线程中可以重新绘制画面,而View必须在UI的主...

kangyunqiang
2013/12/13
0
0

没有更多内容

加载失败,请刷新页面

加载更多

第11章 多线程

程序、进程、线程 程序(program)是为完成特定任务、用某种语言编写的一组指令的集合。即指一段静态的代码,静态对象。 **进程(process)**是程序的一次执行过程或是正在运行的一个程序。动...

流小文
12分钟前
1
0
SpringBoot引入第三方jar包或本地jar包的处理方式

在开发过程中有时会用到maven仓库里没有的jar包或者本地的jar包,这时没办法通过pom直接引入,那么该怎么解决呢 一般有两种方法 - 第一种是将本地jar包安装在本地maven库 - 第二种是将本地j...

独钓渔
今天
2
0
五、MyBatis缓存

一、MyBatis缓存介绍 缓存的使用可以明显的加快访问数据速度,提升程序处理性能,生活和工作中,使用缓存的地方很多。在开发过程中,从前端-->后端-->数据库等都涉及到缓存。MyBatis作为数据...

yangjianzhou
今天
2
0
最近研究如何加速UI界面开发,有点感觉了

最近在开发JFinal学院的JBolt开发平台,后端没啥说的,做各种极简使用的封装,开发者上手直接使用。 JBolt开发平台包含常用的用户、角色、权限、字典、全局配置、缓存、增删改查完整模块、电...

山东-小木
今天
3
0
《月亮与六便士》的读后感作文3000字

《月亮与六便士》的读后感作文3000字: 看完英国作家威廉.萨默塞特.毛姆所著《月亮与六便士》(李继宏译),第一疑问就是全书即没提到“月亮”,也没提到“六便士”。那这书名又与内容有什么...

原创小博客
昨天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部