博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android -- View
阅读量:5944 次
发布时间:2019-06-19

本文共 13568 字,大约阅读时间需要 45 分钟。

setContentView                                                                       

只要你使用过Activity,那么你一定使用过setContentView这个方法。一般都是这样调用该方法:

setContentView(R.layout.main);

然后,在手机或者模拟器上就可以看见自己的布局。

如果,你留意的话,setContentView还有很多过载方法:

public void setContentView(int layoutResID) {      getWindow().setContentView(layoutResID);  }    public void setContentView(View view) {      getWindow().setContentView(view);  }    public void setContentView(View view, ViewGroup.LayoutParams params) {      getWindow().setContentView(view, params);  }

那么,getWindow()方法是做什么的呢?一探究竟:

public Window getWindow() {          return mWindow;  }

可以看出,该方法返回一个Window实例。但是Window是一个抽象类啊,怎么可以有实例对象???原来,Window类有一个子类PhoneWindow,那么如何得知getWindow返回的是PhoneWindow实例呢?来,看下面这张图:

至此,您应该明白setContentView()方法是调用PhoneWindow类的同名方法。源码如下:

@Override      public void setContentView(int layoutResID) {          if (mContentParent == null) {              installDecor();          } else {              mContentParent.removeAllViews();          }          mLayoutInflater.inflate(layoutResID, mContentParent);          final Callback cb = getCallback();          if (cb != null) {              cb.onContentChanged();          }      }        @Override      public void setContentView(View view) {          setContentView(view, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));      }        @Override      public void setContentView(View view, ViewGroup.LayoutParams params) {          if (mContentParent == null) {              installDecor();          } else {              mContentParent.removeAllViews();          }          mContentParent.addView(view, params);          final Callback cb = getCallback();          if (cb != null) {              cb.onContentChanged();          }      }

每个Activity都会实例化一个Window并且只有一个,而View就像是贴在Window上的装饰品。窗户(Window)只有一个,但是窗花(View)可以有很多。

LayoutInflater                                                                        

获得 LayoutInflater 实例

LayoutInflater inflater = getLayoutInflater();LayoutInflater localinflater =(LayoutInflater)context.getSystemServie(Context.LAYOUT_INFLATER_SERVICE); LayoutInflater inflater = LayoutInflater.from(context);

对于第一种,主要是调用 Activity 的 getLayoutInflater() 方法。

继续跟踪研究 android 源码,Activity 中的该方法是调用 PhoneWindow 的 getLayoutInflater()方法!

public PhoneWindow(Context context) {        super(context);        mLayoutInflater = LayoutInflater.from(context);}

可以看出它其实是调用 LayoutInflater.from(context), 那么该方法其实是调用第二种方法,看看源码,如下:

/**     * Obtains the LayoutInflater from the given context.     */    public static LayoutInflater from(Context context) {        LayoutInflater LayoutInflater =        (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);        if (LayoutInflater == null) {            throw new AssertionError("LayoutInflater not found.");        }        return LayoutInflater;    }

inflate 方法

inflate 原意是充气之类的,在这里主要意思就是,扩张、使之膨胀。换句话说就是将当前视图view补充完整、扩展该视图。

public View inflate (int resource, ViewGroup root)public View inflate (XmlPullParser parser, ViewGroup root)public View inflate (XmlPullParser parser, ViewGroup root, boolean attachToRoot)public View inflate (int resource, ViewGroup root, boolean attachToRoot)

示例代码:

LayoutInflater inflater = (LayoutInflater)getSystemService(LAYOUT_INFLATER_SERVICE);/* R.id.test 是 custom.xml 中根(root)布局 LinearLayout 的 id */View view = inflater.inflate(R.layout.custom,(ViewGroup)findViewById(R.id.test));/* 通过该 view 实例化 EditText对象, 否则报错,因为当前视图不是custom.xml.即没有 setContentView(R.layout.custom) 或者 addView() *///EditText editText = (EditText)findViewById(R.id.content);// errorEditText editText = (EditText)view.findViewById(R.id.content);

对于上面代码,指定了第二个参数 ViewGroup root,当然你也可以设置为 null 值。

注意:该方法与 findViewById 方法不同。

inflater 是用来找 layout 下 xml 布局文件,并且实例化!而 findViewById() 是找具体 xml 下的具体 widget 控件(如: Button,TextView 等)。

postInvalidate()   (参考)                                                      

在子线程中控制UI:

@Override      protected void onRestart() {          super.onRestart();           /*onRestart中开启新线程,更新UI*/          Thread thread = new Thread(new Runnable() {                            @Override              public void run() {                  System.out.println(Thread.currentThread().getName() + ": " + Thread.currentThread().getId());                  tv.postInvalidate();                  btn.postInvalidate();                  tv.setText("update UI is success!");                  btn.setText("update UI is success!");              }});          thread.start();      }

postInvalidate() 方法,源码:

public void postInvalidate() {          postInvalidateDelayed(0);      }  public void postInvalidateDelayed(long delayMilliseconds) {          // We try only with the AttachInfo because there's no point in invalidating          // if we are not attached to our window          if (mAttachInfo != null) {              Message msg = Message.obtain();              msg.what = AttachInfo.INVALIDATE_MSG;              msg.obj = this;              mAttachInfo.mHandler.sendMessageDelayed(msg, delayMilliseconds);          }      }

其实,是调用了 Handler 的处理消息的机制!该方法可以在子线程中直接用来更新UI。但是在 Button 的事件中开启线程,更新 UI就会报错报异常。

Handler 和 invalidate 方法结合多线程更新 UI                              

方法 invalidate 主要用在主线程中(即UI 线程中),不可以用于子线程如果在子线程中需要使用 postInvalidate 方法。

public void invalidate() {          if (ViewDebug.TRACE_HIERARCHY) {              ViewDebug.trace(this, ViewDebug.HierarchyTraceType.INVALIDATE);          }          if ((mPrivateFlags & (DRAWN | HAS_BOUNDS)) == (DRAWN | HAS_BOUNDS)) {              mPrivateFlags &= ~DRAWN & ~DRAWING_CACHE_VALID;              final ViewParent p = mParent;              final AttachInfo ai = mAttachInfo;              if (p != null && ai != null) {                  final Rect r = ai.mTmpInvalRect;                  r.set(0, 0, mRight - mLeft, mBottom - mTop);                  // Don't call invalidate -- we don't want to internally scroll                  // our own bounds                  p.invalidateChild(this, r);              }          }      }

invalidate 方法如果你直接在主线程中调用,是看不到任何更新的。需要与Handler结合!

Android 在 onDraw 事件处理绘图,而 invalidate() 函数可以再一次触发 onDraw 事件,然后再一次进行绘图动作。

public class MasterActivity extends Activity {      static int times = 1;         /** Called when the activity is first created. */      @Override      public void onCreate(Bundle savedInstanceState) {          super.onCreate(savedInstanceState);             setContentView( new View(null){                 Paint vPaint = new Paint();  //绘制样式物件              private int i = 0;           //弧形角度                 @Override              protected void onDraw (Canvas canvas) {                  super.onDraw(canvas);                  System.out.println("this run " + (times++) +" times!");                     // 设定绘图样式                  vPaint.setColor( 0xff00ffff ); //画笔颜色                  vPaint.setAntiAlias( true );   //反锯齿                  vPaint.setStyle( Paint.Style.STROKE );                     // 绘制一个弧形                  canvas.drawArc(new RectF(60, 120, 260, 320), 0, i, true, vPaint );                     // 弧形角度                  if( (i+=10) > 360 )                      i = 0;                     // 重绘, 再一次执行onDraw 程序                  invalidate();              }          });      }  }

经过测试,发现 times 一直在++,说明 onDraw 被多次调用,并且一直在画图!

注释掉的话:

// 重绘, 再一次执行onDraw 程序              //invalidate();

可以看出,图像只画了一条线,说明onDraw()方法被调用一次。从log上也可以看出来:

D/mark    (  221): this run onDraw() 1 times!

那么,是什么力量促使onDraw()方法被调用呢?

setContentView()View view方法,其实是调用PhoneWindow的setContentView(View view)方法,调用关系如下:

从而可以看出,invalidate()方法是促使onDraw()方法被调用的力量。

那么,修改代码,将内部类MyView的onDraw()方法中的invalidate()注释取消,再看看运行效果:

控制台:

D/mark    (  248): this run onDraw() 5629 times!  D/mark    (  248): this run onDraw() 5630 times!  D/mark    (  248): this run onDraw() 5631 times!  D/mark    (  248): this run onDraw() 5632 times!  D/mark    (  248): this run onDraw() 5633 times!  D/mark    (  248): this run onDraw() 5634 times!  D/mark    (  248): this run onDraw() 5635 times!  D/mark    (  248): this run onDraw() 5636 times!  D/mark    (  248): this run onDraw() 5637 times!  D/mark    (  248): this run onDraw() 5638 times!  D/mark    (  248): this run onDraw() 5639 times!  D/mark    (  248): this run onDraw() 5640 times!  D/mark    (  248): this run onDraw() 5641 times!  D/mark    (  248): this run onDraw() 5642 times!  D/mark    (  248): this run onDraw() 5643 times!  D/mark    (  248): this run onDraw() 5644 times!  D/mark    (  248): this run onDraw() 5645 times!  D/mark    (  248): this run onDraw() 5646 times!

可以看出,invalidate()方法使onDraw()一直被调用,实现重绘的效果。

在invalidate()方法源码中,有这么一段注释:

/**      * Invalidate the whole view. If the view is visible, {
@link #onDraw} will * be called at some point in the future. This must be called from a * UI thread. To call from a non-UI thread, call {
@link #postInvalidate()}. */

这段话,说明了上面的实现(调用onDraw()方法)。但是在子线程中必须使用postInvalidate()方法。

invalidate()源码分析                                                                

public class ViewDrawTestActivity extends Activity {      // 用于测试      static int times = 1;        @Override      public void onCreate(Bundle savedInstanceState) {          super.onCreate(savedInstanceState);          MyView mView = new MyView(this);          mView.invalidate();          //setContentView(mView);      }        /**      * 内部类,继承View      *       * @author mark      */      class MyView extends View {            MyView(Context context) {              super(context);          }            Paint vPaint = new Paint(); // 绘制样式物件          int i = 0; // 弧形角度            @Override          protected void onDraw(Canvas canvas) {              super.onDraw(canvas);              Log.d("mark", "this run onDraw() " + (times++) + " times!");              // 设定绘图样式              vPaint.setColor(0xff00ffff); // 画笔颜色              vPaint.setAntiAlias(true); // 反锯齿              vPaint.setStyle(Paint.Style.STROKE);              // 绘制一个弧形              canvas.drawArc(new RectF(60, 120, 260, 320), 0, i, true, vPaint);              // 弧形角度              if ((i += 10) > 360) {                  i = 0;              }              // 重绘, 再一次执行onDraw 程序              // invalidate();          }      }  }

子没有多大的变化,只是在onCreate()方法中直接调用invalidate()方法,如:

mView.invalidate();

这样做的目的主要是想看看,自己调用View的invalidate()方法会不会触发onDraw()方法。运行一下:

nDraw()方法并没有执行!那么是不是因为没有调用setContentVIew()方法呢?修改onCreate()方法:

@Override      public void onCreate(Bundle savedInstanceState) {          super.onCreate(savedInstanceState);          MyView mView = new MyView(this);          mView.invalidate();          setContentView(mView);          mView.invalidate();      }

再次运行,效果:

D/mark    (  251): this run onDraw() 1 times!

说明,只有setContentVIew()方法中的invalidate()方法启了作用,自己调用View的invalidate()方法,mView.invalidate()没启任何作用。但是,在MyView的onDraw()方法中调用invalidate()方法可以循环调用onDraw()方法,类似递归。

分析一下,invalidate()方法的源码吧,在这里也许可以找到答案。

/**  * Invalidate the whole view. If the view is visible, {
@link #onDraw} will * be called at some point in the future. This must be called from a * UI thread. To call from a non-UI thread, call {
@link #postInvalidate()}. */ public void invalidate() { if (ViewDebug.TRACE_HIERARCHY) { ViewDebug.trace(this, ViewDebug.HierarchyTraceType.INVALIDATE); } if ((mPrivateFlags & (DRAWN | HAS_BOUNDS)) == (DRAWN | HAS_BOUNDS)) { mPrivateFlags &= ~DRAWN & ~DRAWING_CACHE_VALID; final ViewParent p = mParent; final AttachInfo ai = mAttachInfo; if (p != null && ai != null) { final Rect r = ai.mTmpInvalRect; r.set(0, 0, mRight - mLeft, mBottom - mTop); // Don't call invalidate -- we don't want to internally scroll // our own bounds p.invalidateChild(this, r); } } }

这里可以看到p.invalidateChild(this, r)(看源码只看关键部分,不然你会很晕!),其中p是ViewParent实例对象。ViewParent是一个接口,现在我们关心谁实现了这个接口?

通过千辛万苦的search,终于找到ViewParent的实现类ViewRoot:

/**  * The top of a view hierarchy, implementing the needed protocol between View  * and the WindowManager.  This is for the most part an internal implementation  * detail of {
@link WindowManagerImpl}. * * {@hide} */ @SuppressWarnings({
"EmptyCatchBlock"}) public final class ViewRoot extends Handler implements ViewParent, View.AttachInfo.Callbacks { }

那么,看看该类实现的invalidateChild()方法:

public void invalidateChild(View child, Rect dirty) {          checkThread();          if (DEBUG_DRAW) Log.v(TAG, "Invalidate child: " + dirty);          if (mCurScrollY != 0 || mTranslator != null) {              mTempRect.set(dirty);              dirty = mTempRect;              if (mCurScrollY != 0) {                 dirty.offset(0, -mCurScrollY);              }              if (mTranslator != null) {                  mTranslator.translateRectInAppWindowToScreen(dirty);              }              if (mAttachInfo.mScalingRequired) {                  dirty.inset(-1, -1);              }          }          mDirty.union(dirty);          if (!mWillDrawSoon) {              scheduleTraversals();          }      }

关键代码在这儿:

if (!mWillDrawSoon) {              scheduleTraversals();  }

这个方法是向Handler发送消息:

public void scheduleTraversals() {          if (!mTraversalScheduled) {              mTraversalScheduled = true;              sendEmptyMessage(DO_TRAVERSAL);          }  }

接下来,看看ViewRoot的Handler的handleMessage的实现:

public void handleMessage(Message msg) {      switch (msg.what) {      // 、、、      case DO_TRAVERSAL:      // 、、、           performTraversals();      }  }

performTraversals()方法,调用ViewRoot的私有方法private void draw(boolean fullRedrawNeeded),在该方法中有句代码很关键:

mView.draw(canvas);

其实这句代码,就是调用View的draw()方法 ,关键代码:

if (!dirtyOpaque) onDraw(canvas);

也就是说,满足这个方法,就会回调onDraw()方法。到此为止,您应该明白,当我们自己调用invalidate()方法时,想使onDraw()方法回调,必须满足条件。

调用关系,请看草图!

我是天王盖地虎的分割线                                    

本文转自我爱物联网博客园博客,原文链接:http://www.cnblogs.com/yydcdut/p/3851955.html,如需转载请自行联系原作者

你可能感兴趣的文章
论代码审查的重要性
查看>>
「docker实战篇」python的docker爬虫技术-导学(一)
查看>>
如何确定一个网站是用Wordpress开发的
查看>>
我的友情链接
查看>>
我的友情链接
查看>>
wdcp 安装
查看>>
C语言运算符优先级相关问题
查看>>
MP4视频播放器代码
查看>>
Nginx 匹配 iphone Android 微信
查看>>
ldap
查看>>
Yum软件仓库配置
查看>>
linux 压缩与解压总结
查看>>
mysql脚本1064 - You have an error in your SQL syntax; check the manual
查看>>
nessus 本地扫描(一)
查看>>
linux服务器磁盘陈列
查看>>
python----tcp/ip http
查看>>
我的友情链接
查看>>
第一本docker书学习笔记1-3章
查看>>
一個典型僵尸網絡淺析
查看>>
vmware克隆Centos6.4虚拟机网卡无法启动问题
查看>>