Android高手必須了解的關(guān)于View的細(xì)節(jié)(進(jìn)階必備)

關(guān)于Android的視圖體系,有一些位置梳庆、坐標(biāo)系暖途、定位的小細(xì)節(jié)可能開發(fā)者并沒有注意到,本文將指出一些讓人驚訝的小細(xì)節(jié)膏执,并以實(shí)例證明驻售。

一、Android的視圖組織體系不為人知的小細(xì)節(jié)

眾所周知Android中的view分為view和viewGroup更米,viewGroup又繼承了view欺栗,兩者組織起來成為一顆“視圖樹”。

Actiity并不直接承載view征峦,承載view的是Android中的Window迟几,他們具體的組織形式如下圖所畫。

image

其中DecorView作為根View栏笆,是Android中的一個(gè)自定義View类腮,看源碼可知它繼承了FrameLayout。它內(nèi)部有有三個(gè)部分:

  • 頂部的View蛉加,這個(gè)View起到一個(gè)占位的作用蚜枢,讓系統(tǒng)狀態(tài)欄能顯示出來。
  • 一個(gè)垂直方向的線性布局七婴,分為標(biāo)題欄和內(nèi)容欄祟偷,我們用setContentView給一個(gè)Activity設(shè)置的布局,就會加入到內(nèi)容欄部分打厘。
  • 底部View修肠,同樣起到一個(gè)占位的作用,讓系統(tǒng)的底部導(dǎo)航欄顯示出來户盯。

下圖是使用Android Studio Inspector Layout 抓取的視圖樹的截圖嵌施。


image

二、View的繪制過程的一個(gè)小細(xì)節(jié)

視圖樹的測量過程是從上到下的莽鸭,頂層View先測量自己的大小吗伤,然后遞歸調(diào)用子View的測量方法。這個(gè)很好理解硫眨,只有父View測量好了足淆,才能讓子View知道自己最大可以獲得多大空間。

關(guān)于測量不再多講,此處特別提出關(guān)于MeasureSpec的一些小知識巧号。

我們在自定義View的重寫onMeasure()發(fā)現(xiàn)有傳入兩個(gè)Int型的參數(shù)族奢,就是MeasureSpec,如下代碼所示丹鸿。

 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);

        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);

        int width, height;

        //一些判斷邏輯
        
        
        //最終設(shè)置自定義View寬高
        setMeasuredDimension(width, height);
    }

關(guān)于MeasureSpec有一個(gè)奇怪的現(xiàn)象越走,它明明是一個(gè)int值,但是卻保存兩個(gè)信息:

  • 模式信息靠欢,通過getMode()得到
  • 尺寸信息:通過getSize()得到

這是為什么呢廊敌?這可以通過查看源碼得到,MeasureSpec是一個(gè)32位的int值沒錯门怪,但是它的高2位儲存了SpecMode信息骡澈,低30位儲存了SpecSize尺寸信息。

 private static final int MODE_SHIFT = 30;
 
 private static final int MODE_MASK  = 0x3 << MODE_SHIFT;
 
 public static int getMode(int measureSpec) { 
           //通過與掩碼與運(yùn)算獲得模式信息
            return (measureSpec & MODE_MASK);
        }

 public static int getSize(int measureSpec) {
            //通過與掩碼的反碼與運(yùn)算獲得尺寸信息
            return (measureSpec & ~MODE_MASK);
        }

如上述源碼所示掷空,定義一個(gè)掩碼 原始值 0000 0000 0000 0000 0000 0000 0000 0011

通過左移運(yùn)算符 移動30位秧廉,變?yōu)? 1100 0000 0000 0000 0000 0000 0000 0000

getMode()中通過與掩碼與運(yùn)算獲得模式信息(高2位留存)。

getSize()中通過與掩碼的反碼與運(yùn)算獲得尺寸信息(低30位留存)拣帽。

講到這,是不是解決了一塊小小的心步莱减拭?

更多自定View的基礎(chǔ)知識可以參看本人另一篇博客:Android自定義View講解加示例

三、View定位體系不為人知的細(xì)節(jié)区丑。

關(guān)于View的定位體系很多開發(fā)者并沒深入了解拧粪,本節(jié)將提出一些問題看看諸位看客是否都有所了解。

1. View的定位模型和定位參數(shù)

我們在使用View定位的時(shí)候沧侥,可能會設(shè)置他們的一些位置信息

  • 比如top,left,right,bottom可霎,這些可以通過setTop()等方法進(jìn)行。

  • 或者會使用到setX()宴杀,setY()方法設(shè)置其X,Y坐標(biāo)癣朗。

  • 又有可能使用setTranslateX(),setTranslateY()旺罢。

  • 還有可能使用scrollBy()旷余,scrollTo()改變其位置信息。

那么關(guān)于上述的種種行為就需要理清楚View的坐標(biāo)體系扁达,請看下圖正卧。

image

從上圖可以看出:

  • View的定位是一種相對定位,即相對于父元素進(jìn)行定位跪解。

  • View的坐標(biāo)系和常規(guī)略有不同炉旷,x的正方向是向右,y軸正方向向下。

  • top和left含義很好理解窘行,但是請注意饥追,right的含義是子View右邊距距離父View左邊距的距離,botton的含義是子View下邊距距離父View上邊距的距離抽高。

  • View的X判耕,Y坐標(biāo)是指v左上角相對于父元素的坐標(biāo)。

  • 關(guān)于translateX和 translateY圖中并沒有畫出翘骂,這兩個(gè)屬性是View左上角相對于原來位置的偏移量壁熄。通過以下公式可能更加清晰:

          X=left+translateX
    

關(guān)于定位模型,是不是解決了你的一些疑惑碳竟?

2. View的scrollBy(x,y)和scrollTo(x,y)方法草丧。

可能有開發(fā)者經(jīng)常使用這兩個(gè)方法,但是你是否注意到兩個(gè)小點(diǎn):

  • x傳入正值向左移莹桅,y傳入正值向上移動(后面有代碼示范昌执。)

  • 這兩個(gè)方法只是移動了View的內(nèi)容,View的實(shí)際定位框架并沒有改變诈泼,還是用個(gè)圖來表示懂拾。

image

上圖可以說明這個(gè)概念,scroll移動的只是view的內(nèi)容铐达,view實(shí)際上的位置并沒有移動岖赋。

那肯定有人會表示懷疑,必須通過一個(gè)例子實(shí)踐證明瓮孙。

怎么驗(yàn)證上述結(jié)論呢唐断?其實(shí)很簡單,第一杭抠、如果View整體都移動了脸甘,那么應(yīng)該會引起整個(gè)視圖樹的重新定位,比如會發(fā)生后面的view會補(bǔ)充空白等現(xiàn)象偏灿。第二丹诀、我們實(shí)時(shí)獲取被移動View的 top、left屬性值菩混,看是否發(fā)生變化忿墅。好,我們寫一個(gè)demo沮峡。

寫一個(gè)布局文件疚脐,一個(gè)垂直線性布局,放入一個(gè)textView和一個(gè)button

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    >
    <TextView
        android:id="@+id/textview1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="TextView點(diǎn)擊移動自己"
        android:layout_marginTop="350dp"
        android:gravity="center"
        />

    <Button
        android:id="@+id/button2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="button觀察其是否跟隨移動補(bǔ)位"
        android:layout_marginTop="20dp"
        />
</LinearLayout>

寫activity的代碼邢疙,讓textView點(diǎn)擊的時(shí)候棍弄,向上移動自己的位置望薄,同時(shí)輸出位置信息

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        final TextView textView = (TextView) findViewById(R.id.textview1);
        textView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                textView.scrollBy(0, 10);//點(diǎn)擊移動自己
                //輸出位置參數(shù)
                Log.i("WQW", "Top參數(shù):" + textView.getTop());
                Log.i("WQW", "Left參數(shù):" + textView.getLeft());
                Log.i("WQW", "Right參數(shù):" + textView.getRight());
                Log.i("WQW", "Bottom參數(shù):" + textView.getBottom());
            }
        });
    }
}

來看下Gif運(yùn)行結(jié)果,檢查一下下面的Button是否會補(bǔ)位呼畸。從下面結(jié)果中可以看到Button并不會隨之向上移動補(bǔ)位痕支。

image

再來看一下Log輸出,可以發(fā)現(xiàn)top等位置屬性值都沒有發(fā)生變化蛮原。至此卧须,已經(jīng)證明完畢,scrollBy()等方法只是移動了View的內(nèi)容儒陨。

11-08 04:24:18.678 32105-32105/com.example.administrator.myapplication I/WQW: Top參數(shù):1050
11-08 04:24:18.678 32105-32105/com.example.administrator.myapplication I/WQW: Left參數(shù):0
11-08 04:24:18.678 32105-32105/com.example.administrator.myapplication I/WQW: Right參數(shù):1080
11-08 04:24:18.678 32105-32105/com.example.administrator.myapplication I/WQW: Bottom參數(shù):1107
11-08 04:24:20.603 32105-32105/com.example.administrator.myapplication I/WQW: Top參數(shù):1050
11-08 04:24:20.603 32105-32105/com.example.administrator.myapplication I/WQW: Left參數(shù):0
11-08 04:24:20.603 32105-32105/com.example.administrator.myapplication I/WQW: Right參數(shù):1080
11-08 04:24:20.603 32105-32105/com.example.administrator.myapplication I/WQW: Bottom參數(shù):1107
11-08 04:24:23.867 32105-32105/com.example.administrator.myapplication I/WQW: Top參數(shù):1050
11-08 04:24:23.867 32105-32105/com.example.administrator.myapplication I/WQW: Left參數(shù):0
11-08 04:24:23.867 32105-32105/com.example.administrator.myapplication I/WQW: Right參數(shù):1080
11-08 04:24:23.867 32105-32105/com.example.administrator.myapplication I/WQW: Bottom參數(shù):1107

三花嘶、總結(jié)

本文總結(jié)了三個(gè)方面的一些開發(fā)者沒有關(guān)注到的小細(xì)節(jié),如Android的視圖組織體系蹦漠,繪制過程和View定位模型定位參數(shù)椭员,將來如發(fā)現(xiàn)更多值得補(bǔ)充的內(nèi)容,仍舊會更新在本文當(dāng)中笛园。

如果覺得本文對你有幫助隘击,請關(guān)注、留言研铆、點(diǎn)贊我埋同,謝謝!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末棵红,一起剝皮案震驚了整個(gè)濱河市莺禁,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌窄赋,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件楼熄,死亡現(xiàn)場離奇詭異忆绰,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)可岂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進(jìn)店門错敢,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人缕粹,你說我怎么就攤上這事稚茅。” “怎么了平斩?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵亚享,是天一觀的道長。 經(jīng)常有香客問我绘面,道長欺税,這世上最難降的妖魔是什么侈沪? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮晚凿,結(jié)果婚禮上亭罪,老公的妹妹穿的比我還像新娘。我一直安慰自己歼秽,他們只是感情好应役,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著燥筷,像睡著了一般箩祥。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上荆责,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天滥比,我揣著相機(jī)與錄音,去河邊找鬼做院。 笑死盲泛,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的键耕。 我是一名探鬼主播寺滚,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼屈雄!你這毒婦竟也來了村视?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤酒奶,失蹤者是張志新(化名)和其女友劉穎蚁孔,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體惋嚎,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡杠氢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了另伍。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鼻百。...
    茶點(diǎn)故事閱讀 38,599評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖摆尝,靈堂內(nèi)的尸體忽然破棺而出温艇,到底是詐尸還是另有隱情,我是刑警寧澤堕汞,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布勺爱,位于F島的核電站,受9級特大地震影響讯检,放射性物質(zhì)發(fā)生泄漏邻寿。R本人自食惡果不足惜蝎土,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望绣否。 院中可真熱鬧誊涯,春花似錦、人聲如沸蒜撮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽段磨。三九已至取逾,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間苹支,已是汗流浹背砾隅。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留债蜜,地道東北人晴埂。 一個(gè)月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像寻定,于是被迫代替她去往敵國和親儒洛。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內(nèi)容