电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Android view繪制流程詳解

瀏覽:5日期:2022-09-18 14:16:51
目錄繪制流程Measure 測(cè)量流程MeasureSpeclayout 布局流程draw 繪制流程相關(guān)類 & 概念DecorViewWindowViewRootActivity 視圖結(jié)構(gòu)繪制流程 measure 流程測(cè)量出 View 的寬高尺寸。 layout 流程確定 View 的位置及最終尺寸。 draw 流程將 View 繪制在屏幕上。Measure 測(cè)量流程

系統(tǒng)是通過 MeasureSpec 測(cè)量 View 的,在了解測(cè)量過程之前一定要了解這個(gè) MeasureSpec 。

MeasureSpec

MeasureSpec 是一個(gè) 32 位的 int 值打包而來(lái)的,打包為 MeasureSpec 主要是為了避免過多的對(duì)象內(nèi)存分配。

為了方便操作,MeasureSpec 提供了快捷的打包和解包的快捷方法。

MeasureSpec.makeMeasureSpec( int size, int mode) MeasureSpec.getMode(int measureSpec) MeasureSpec.getSize(int measureSpec)

MeasureSpec 其中前 2 位表示測(cè)量的模式 SpecMode,后邊 30 位表示某種測(cè)量模式下的尺寸 SpecSize。

MeasureSpec 中有三種測(cè)量模式

UNSPECIFIED 不指定具體尺寸,完全由 View 自己發(fā)揮。 EXACTLY 精確模式,這種模式下使用后邊的 specSize ,一般對(duì)應(yīng)于 LayoutParams 的 match_content 和設(shè)置的精確尺寸。 AT_MOST 最大模式,這種模式下 view 的最大尺寸不能超過后邊的 specSize ,一般對(duì)應(yīng)于 LayoutParams 的 wrap_content

在測(cè)量 View 的時(shí)候,系統(tǒng)會(huì)將自己的 LayoutParams 參數(shù)在父容器的 MeasureSpec 影響下轉(zhuǎn)換為自己的MeasureSpec ,然后再通過這個(gè) MeasureSpec 測(cè)量自身的寬高。

需要注意的是View 的MeasureSpec 不是唯一由 LayoutParams 決定的,是在父容器的共同影響下創(chuàng)建來(lái)的。

在 ViewGroup 的 measureChild() 可以看到具體的實(shí)現(xiàn)思路,getChildMeasureSpec() 里就是將 layoutParams 轉(zhuǎn)換為 measureSpec 的實(shí)現(xiàn)思路。

protected void measureChild(View child, int parentWidthMeasureSpec,int parentHeightMeasureSpec) {//拿到子元素的 LayoutParams 參數(shù) final LayoutParams lp = child.getLayoutParams(); //創(chuàng)建子元素的 measureSpec final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec, mPaddingLeft + mPaddingRight, lp.width); final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec, mPaddingTop + mPaddingBottom, lp.height); //將測(cè)量傳遞到子元素 child.measure(childWidthMeasureSpec, childHeightMeasureSpec);}public static int getChildMeasureSpec(int spec, int padding, int childDimension) { //解析父容器的 measureSpec ,解析出模式和尺寸 int specMode = MeasureSpec.getMode(spec); int specSize = MeasureSpec.getSize(spec); int size = Math.max(0, specSize - padding); int resultSize = 0; int resultMode = 0; switch (specMode) { // 父容器是精確模式的情況,設(shè)置了精確尺寸。 case MeasureSpec.EXACTLY:if (childDimension >= 0) {//子元素本身是設(shè)置的精確尺寸,就是EXACTLY 模式,尺寸就是設(shè)置的尺寸。 resultSize = childDimension; resultMode = MeasureSpec.EXACTLY;} else if (childDimension == LayoutParams.MATCH_PARENT) { // 子元素設(shè)置的 match_content 充滿入容器,就把尺寸設(shè)置為入容器的尺寸,模式設(shè)置為EXACTLY resultSize = size; resultMode = MeasureSpec.EXACTLY;} else if (childDimension == LayoutParams.WRAP_CONTENT) { // 包裹模式下,子元素可以自己設(shè)置尺寸,但是不能超過夫容器的尺寸。模式為AT_MOST,尺寸為父容器的尺寸。 resultSize = size; resultMode = MeasureSpec.AT_MOST;}break; //父容器是最大模式 case MeasureSpec.AT_MOST:if (childDimension >= 0) { // 設(shè)置為子元素的尺寸,為精確模式 resultSize = childDimension; resultMode = MeasureSpec.EXACTLY;} else if (childDimension == LayoutParams.MATCH_PARENT) { // 子元素想充滿父容器,應(yīng)該設(shè)置為父容器的尺寸,但是父容器是最大模式,沒有精確尺寸。 // 所以將子元素設(shè)置為最大模式,不能超過父容器目前的尺寸。 resultSize = size; resultMode = MeasureSpec.AT_MOST;} else if (childDimension == LayoutParams.WRAP_CONTENT) { // 子元素沒有精確尺寸,想包裹自身,這種模式下,設(shè)置為最大模式,不超過父容器尺寸就好。 // bigger than us. resultSize = size; resultMode = MeasureSpec.AT_MOST;}break; // 父容器沒有限制,子元素自己發(fā)揮 case MeasureSpec.UNSPECIFIED:if (childDimension >= 0) { //子元素自己有設(shè)置的值,就好實(shí)用自己的值,設(shè)置為精確模式 resultSize = childDimension; resultMode = MeasureSpec.EXACTLY;} else if (childDimension == LayoutParams.MATCH_PARENT) { // 子元素想充滿父容器,那就找到父容器的尺寸,但父容器的尺寸未知,還是要自己發(fā)揮 UNSPECIFIED。 resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size; resultMode = MeasureSpec.UNSPECIFIED;} else if (childDimension == LayoutParams.WRAP_CONTENT) { // 只元素是包裹自身,父容器無(wú)法給出參考,所以讓子元素自己去隨意發(fā)揮,仍然是UNSPECIFIED resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size; resultMode = MeasureSpec.UNSPECIFIED;}break; } //使用打包方法,將子元素的模式和尺寸打包并返回 return MeasureSpec.makeMeasureSpec(resultSize, resultMode);}

measure 流程是在 ViewRoot 的 performMeasure() 里開始的。

在這里會(huì)將 DecorView 的 layoutParams 在 window 的 measureSpec 影響下轉(zhuǎn)換為自己的 measureSpec 。 然后調(diào)用 DecorView 的 measure() 將寬高的 measureSpec 傳入,在 measure() 里,decorView 開始自己的測(cè)量。

從 DecorView 的 measure() 開始,整個(gè) View 樹的測(cè)量流程就開始了。

View 的測(cè)量都是在 measure() 里進(jìn)行的,這是個(gè) final 類型的方法,里面的實(shí)現(xiàn)比較簡(jiǎn)單會(huì)有一些判斷調(diào)整,是否需要測(cè)量,會(huì)繼續(xù)調(diào)用 onMeasure() 將 measureSpec 傳進(jìn)來(lái),測(cè)量尺寸的確定最終是在 onMeasure() 里完成的。

通常我們自定義 View 都要重寫這個(gè)方法實(shí)現(xiàn)自己的測(cè)量邏輯,包括我們常用的控件都是自己重寫了這個(gè)方法實(shí)現(xiàn)自己的測(cè)量邏輯。

如果不重寫 onMeasure(),會(huì)導(dǎo)致自定義 view 的 wrap_content 參數(shù)無(wú)效,具體可以看一下 getDefaultSize() 實(shí)現(xiàn)。

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec), getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));}public static int getDefaultSize(int size, int measureSpec) { int result = size; int specMode = MeasureSpec.getMode(measureSpec); int specSize = MeasureSpec.getSize(measureSpec); switch (specMode) { case MeasureSpec.UNSPECIFIED:result = size;break; case MeasureSpec.AT_MOST: case MeasureSpec.EXACTLY://默認(rèn) 精確模式和最大模式下都是使用后邊的 specSize ,這會(huì)導(dǎo)致我們?cè)O(shè)置的 wrap_content 無(wú)效,始終是充滿父容器。result = specSize;break; } return result;}protected int getSuggestedMinimumHeight() { return (mBackground == null) ? mMinHeight : max(mMinHeight, mBackground.getMinimumHeight());} protected int getSuggestedMinimumWidth() { return (mBackground == null) ? mMinWidth : max(mMinWidth, mBackground.getMinimumWidth());}

View 和 ViewGroup 的測(cè)量過程是不同的。

單純的 View 只需要在 onMeasure() 里完成自己的測(cè)量就可以了,ViewGroup 除了完成自己的測(cè)量外,還有子元素的測(cè)量。

ViewGroup 的 onMeasure() 是沒有任何實(shí)現(xiàn)的,因?yàn)楦鱾€(gè)布局的特性不同,具體測(cè)量邏輯也是不同的,具體實(shí)現(xiàn)都在各個(gè)布局里。

但是 ViewGroup 里提供了 measureChildren() 方法,思路就是,遍歷所有需要顯示的子元素,取出他們的 LayoutParams 參數(shù)在自己 measureSpec 的影響下創(chuàng)建出子元素的 measureSpec ,然后將調(diào)用子元素的 measure() 將measureSpec 傳遞進(jìn)去。

這里就將測(cè)量傳遞到了子元素。如果子元素是單純的 View 控件只需要完成自己就可以了,如果是 ViewGroup 會(huì)繼續(xù)將測(cè)量遞歸下去,直至完成整個(gè) View 樹的測(cè)量。

protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {final int size = mChildrenCount;final View[] children = mChildren;for (int i = 0; i < size; ++i) { final View child = children[i]; if ((child.mViewFlags & VISIBILITY_MASK) != GONE) {//測(cè)量子元素,measureChild 見上面 MeasureSpec 里的代碼。measureChild(child, widthMeasureSpec, heightMeasureSpec); }} }

在完成測(cè)量流程之后就會(huì)進(jìn)入了 layout 流程了。

layout 布局流程

layout 這一流程會(huì)確定 View 的四個(gè)頂點(diǎn)位置,進(jìn)而確定在父容器中的位置和最終寬高。

layout 流程也是在 ViewRoot 里開始,是在 performLayout() 里首先調(diào)用 DecorView 的 layout() 方法開始整個(gè) View 樹的布局流程。

View 的布局流程都是在 layout() 方法里完成的,會(huì)在這里通過 setFrame() 設(shè)置自己四個(gè)頂點(diǎn)的位置。

設(shè)置完自己的位置后,會(huì)繼續(xù)調(diào)用 onLayout() 方法,如果是 ViewGroup 可以繼續(xù)在 onLayout 里確定子元素的位置。

View 的 onLayout() 是沒有任何實(shí)現(xiàn)的,因?yàn)樗菦]有子元素,ViewGroup 本身也是沒有實(shí)現(xiàn)的,也都是具體的各個(gè)布局里自己實(shí)現(xiàn)的。

思路也是遍歷所有需要布局的子元素,根據(jù)測(cè)量尺寸計(jì)算出他們的位置后調(diào)用子元素的 layout() 方法將位置參數(shù)穿進(jìn)去,讓子元素去完成自己的布局流程。

在這里也是將布局流程傳遞到了子元素,如果子元素是 ViewGroup 會(huì)繼續(xù)將布局流程傳遞,直到完成整個(gè) View 樹的布局流程。

layout() 確定自身的位置 onLayout() 確定子元素的位置

在完成 layout 流程后,就是最后一個(gè) draw 流程了。

draw 繪制流程

這個(gè)流程是將 View 繪制到屏幕上。

draw 流程也是在 ViewRoot 里開始的,具體是在 performDraw() 里開始,在這里會(huì)調(diào)用 DecorView 的 draw() 開始整個(gè) View 樹的繪制。

draw 的過程相對(duì)來(lái)說(shuō)較為簡(jiǎn)單,在 draw() 里可以看到整個(gè)步驟

繪制背景 drawBackground(canvas); 繪制自己的內(nèi)容 onDraw(canvas); 繪制子元素 dispatchDraw(canvas); 繪制裝飾 onDrawForeground(canvas);

我們自定義 View 都會(huì)在 onDraw() 里實(shí)現(xiàn)自己的繪制邏輯,View 的 dispatchDraw() 是沒有任何實(shí)現(xiàn)的,具體實(shí)現(xiàn)在 ViewGroup 里。

在 ViewGroup 后調(diào)用子元素的 draw() 將繪制流程傳遞到子元素,直到繪制完整個(gè) View 樹。

在完成整個(gè) View 樹的繪制后,就可以在屏幕上看見界面了。

相關(guān)類 & 概念

在 View 的繪制過程中,涉及到了很多類,這里就不做詳細(xì)的介紹了,只在這里簡(jiǎn)單列一下,知道這些個(gè)的作用。

DecorView

整個(gè) View 樹的根節(jié)點(diǎn),所有的繪制,事件都是從這個(gè) View 開始分發(fā)的。

它繼承自 FrameLayout 是一個(gè) ViewGroup ,內(nèi)部含有一個(gè) LinearLayout 。

這個(gè) LinearLayout 里有一個(gè) id 為 content 的 FrameLayout ,我們通常設(shè)置的 setContentView() 就是加載到了這個(gè) FrameLayout 里。

Window

每個(gè) Activity 都有一個(gè) window ,直譯就是“窗口”,是 Activity 的成員變量,也是應(yīng)用程序的視圖窗口,承載整個(gè) Activity 的視圖。 內(nèi)部含有一個(gè) DeocrView 成員變量,承載的視圖就是這個(gè) DeocrView 。

它目前只有一個(gè)實(shí)現(xiàn)類,PhoneWindow ,activity 里的 mWindow 就是這個(gè)實(shí)例。

ViewRoot

View Root 的作用很大,是連接 DecorView 和 Window Manager 的紐帶。 View 的繪制,觸屏,按鍵,屏幕刷新等事件分發(fā)都通過它完成的。

Activity 視圖結(jié)構(gòu)

Android view繪制流程詳解

以上就是Android view繪制流程詳解的詳細(xì)內(nèi)容,更多關(guān)于Android view繪制流程的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Android
相關(guān)文章:
主站蜘蛛池模板: 台湾阳明固态继电器-奥托尼克斯光电传感器-接近开关-温控器-光纤传感器-编码器一级代理商江苏用之宜电气 | 浙江浩盛阀门有限公司| 北京中航时代-耐电压击穿试验仪厂家-电压击穿试验机 | 线粒体膜电位荧光探针-细胞膜-标记二抗-上海复申生物科技有限公司 | 泥浆在线密度计厂家-防爆数字压力表-膜盒-远传压力表厂家-江苏大亚自控设备有限公司 | 二手Sciex液质联用仪-岛津气质联用仪-二手安捷伦气质联用仪-上海隐智科学仪器有限公司 | 混合气体腐蚀试验箱_盐雾/硫化氢/气体腐蚀试验箱厂家-北京中科博达 | 高博医疗集团上海阿特蒙医院| 隐形纱窗|防护纱窗|金刚网防盗纱窗|韦柏纱窗|上海青木装潢制品有限公司|纱窗国标起草单位 | T恤衫定做,企业文化衫制作订做,广告T恤POLO衫定制厂家[源头工厂]-【汉诚T恤定制网】 | 自清洗过滤器-全自动自清洗过反冲洗过滤器 - 中乂(北京)科技有限公司 | 防火门-专业生产甲级不锈钢钢质防火门厂家资质齐全-广东恒磊安防设备有限公司 | 武汉天安盾电子设备有限公司 - 安盾安检,武汉安检门,武汉安检机,武汉金属探测器,武汉测温安检门,武汉X光行李安检机,武汉防爆罐,武汉车底安全检查,武汉液体探测仪,武汉安检防爆设备 | 电缆接头_防水接头_电缆防水接头 - 乐清市新豪电气有限公司 | 防堵吹扫装置-防堵风压测量装置-电动操作显示器-兴洲仪器 | NMRV减速机|铝合金减速机|蜗轮蜗杆减速机|NMRV减速机厂家-东莞市台机减速机有限公司 | 建筑资质代办_工程施工资质办理_资质代办公司_北京众聚企服 | 工业铝型材-铝合金电机壳-铝排-气动执行器-山东永恒能源集团有限公司 | 气动隔膜阀_气动隔膜阀厂家_卫生级隔膜阀价格_浙江浙控阀门有限公司 | 涡街流量计_LUGB智能管道式高温防爆蒸汽温压补偿计量表-江苏凯铭仪表有限公司 | 上海瑶恒实业有限公司|消防泵泵|离心泵|官网 | 防爆暖风机_防爆电暖器_防爆电暖风机_防爆电热油汀_南阳市中通智能科技集团有限公司 | 专业的新乡振动筛厂家-振动筛品质保障-环保振动筛价格—新乡市德科筛分机械有限公司 | 高温链条油|高温润滑脂|轴承润滑脂|机器人保养用油|干膜润滑剂-东莞卓越化学 | bkzzy在职研究生网 - 在职研究生招生信息咨询平台 | 纯化水设备-纯水设备-超纯水设备-[大鹏水处理]纯水设备一站式服务商-东莞市大鹏水处理科技有限公司 | 自动化生产线-自动化装配线-直流电机自动化生产线-东莞市慧百自动化有限公司 | 不锈钢散热器,冷却翅片管散热器厂家-无锡市烨晟化工装备科技有限公司 | 博医通医疗器械互联网供应链服务平台_博医通 | 手机存放柜,超市储物柜,电子储物柜,自动寄存柜,行李寄存柜,自动存包柜,条码存包柜-上海天琪实业有限公司 | 合肥宠物店装修_合肥宠物美容院装修_合肥宠物医院设计装修公司-安徽盛世和居装饰 | 跨境物流_美国卡派_中大件运输_尾程派送_海外仓一件代发 - 广州环至美供应链平台 | 钢板仓,大型钢板仓,钢板库,大型钢板库,粉煤灰钢板仓,螺旋钢板仓,螺旋卷板仓,骨料钢板仓 | 胃口福饺子加盟官网_新鲜现包饺子云吞加盟 - 【胃口福唯一官网】 | 振动筛,震动筛,圆形振动筛,振动筛价格,振动筛厂家-新乡巨宝机电 蒸汽热收缩机_蒸汽发生器_塑封机_包膜机_封切收缩机_热收缩包装机_真空机_全自动打包机_捆扎机_封箱机-东莞市中堡智能科技有限公司 | 培训中心-翰香原香酥板栗饼加盟店总部-正宗板栗酥饼技术 | 净气型药品柜-试剂柜-无管道净气型通风柜-苏州毕恩思 | 泰来华顿液氮罐,美国MVE液氮罐,自增压液氮罐,定制液氮生物容器,进口杜瓦瓶-上海京灿精密机械有限公司 | 带式过滤机厂家_价格_型号规格参数-江西核威环保科技有限公司 | 无线讲解器-导游讲解器-自助讲解器-分区讲解系统 品牌生产厂家[鹰米讲解-合肥市徽马信息科技有限公司] | MES系统-WMS系统-MES定制开发-制造执行MES解决方案-罗浮云计算 |