View.toString()

我們在對View執行toString()方法的時候,會看到一段輸出:

*.View {33df71a V.E...... ......I. 0,0-0,0 #7f100102 app:id/view}

一般對象的toString輸出是一段hashCode,而 View 的輸出是這麼一串的東西,它是不是有意義呢?我們就來看下View的toString方法的實現。

今天的主角是這一個方法:

public String toString() {
    StringBuilder out = new StringBuilder(128);
    out.append(getClass().getName());
    out.append('{');
    out.append(Integer.toHexString(System.identityHashCode(this)));
    out.append(' ');
    switch (mViewFlags&VISIBILITY_MASK) {
        case VISIBLE: out.append('V'); break;
        case INVISIBLE: out.append('I'); break;
        case GONE: out.append('G'); break;
        default: out.append('.'); break;
    }
    out.append((mViewFlags&FOCUSABLE_MASK) == FOCUSABLE ? 'F' : '.');
    out.append((mViewFlags&ENABLED_MASK) == ENABLED ? 'E' : '.');
    out.append((mViewFlags&DRAW_MASK) == WILL_NOT_DRAW ? '.' : 'D');
    out.append((mViewFlags&SCROLLBARS_HORIZONTAL) != 0 ? 'H' : '.');
    out.append((mViewFlags&SCROLLBARS_VERTICAL) != 0 ? 'V' : '.No');
    out.append((mViewFlags&CLICKABLE) != 0 ? 'C' : '.');
    out.append((mViewFlags&LONG_CLICKABLE) != 0 ? 'L' : '.');
    out.append((mViewFlags&CONTEXT_CLICKABLE) != 0 ? 'X' : '.');
    out.append(' ');
    out.append((mPrivateFlags&PFLAG_IS_ROOT_NAMESPACE) != 0 ? 'R' : '.');
    out.append((mPrivateFlags&PFLAG_FOCUSED) != 0 ? 'F' : '.');
    out.append((mPrivateFlags&PFLAG_SELECTED) != 0 ? 'S' : '.');
    if ((mPrivateFlags&PFLAG_PREPRESSED) != 0) {
        out.append('p');
    } else {
        out.append((mPrivateFlags&PFLAG_PRESSED) != 0 ? 'P' : '.');
    }
    out.append((mPrivateFlags&PFLAG_HOVERED) != 0 ? 'H' : '.');
    out.append((mPrivateFlags&PFLAG_ACTIVATED) != 0 ? 'A' : '.');
    out.append((mPrivateFlags&PFLAG_INVALIDATED) != 0 ? 'I' : '.');
    out.append((mPrivateFlags&PFLAG_DIRTY_MASK) != 0 ? 'D' : '.');
    out.append(' ');
    out.append(mLeft);
    out.append(',');
    out.append(mTop);
    out.append('-');
    out.append(mRight);
    out.append(',');
    out.append(mBottom);
    final int id = getId();
    if (id != NO_ID) {
        out.append(" #");
        out.append(Integer.toHexString(id));
        final Resources r = mResources;
        if (id > 0 && Resources.resourceHasPackage(id) && r != null) {
            try {
                String pkgname;
                switch (id&0xff000000) {
                    case 0x7f000000:
                        pkgname="app";
                        break;
                    case 0x01000000:
                        pkgname="android";
                        break;
                    default:
                        pkgname = r.getResourcePackageName(id);
                        break;
                }
                String typename = r.getResourceTypeName(id);
                String entryname = r.getResourceEntryName(id);
                out.append(" ");
                out.append(pkgname);
                out.append(":");
                out.append(typename);
                out.append("/");
                out.append(entryname);
            } catch (Resources.NotFoundException e) {
            }
        }
    }
    out.append("}");
    return out.toString();
}

代碼非常簡單,從這裏我們知道,toString 裏面把View的一些狀態都標明好了,這對我們 debug 非常有幫助,我們來簡單看一下里面的內容:
先是 用一對括號{}把信息包起來

一段 hashcode

標明View是否可見,值可選爲 V/I/G/.,其中.是意外情況,一般不會有這種狀態出現
標明View是否可獲得焦點(Focusable),值可選爲F/.
標明View是否可用(Enable),值可選爲E/.
標明View是否需要繪製(WillNotDraw),值可選爲D/.
標明View的橫向滾動條是否顯示,值可選爲H/.
標明View的縱向滾動條是否顯示,值可選爲Z/.
標明View是否可點擊(Clickable),值可選爲C/.
標明View是否可長按(Long Clickable),值可選爲L/.
標明View是否可按出上下文菜單(鼠標右鍵點擊,Context Clickable),值可選爲X/.
標明View是否在根命名空間中(Android內部使用),值可選爲R/.
標明View是否獲得焦點(Focused),值可選爲F/.
標明View是否是選擇狀態(Selected),值可選爲S/.
標明View是否是預按下的狀態,值可選爲p/.
標明View是否是按下的狀態,值可選爲P/.
標明View是否是 Hover 的狀態,值可選爲H/.
標明View是否是 Active 的狀態,值可選爲A/.
標明View是否是 Invalidate 的狀態,值可選爲I/.
標明View是否是 Dirty 的狀態,值可選爲D/.
接下去是一個空格,然後是View的尺寸座標信息:left,top-right,bottom
最後一個就是 ID 信息,如果是應用定義的 ID 就是app:id/xxxx

當然,如果你需要 Debug 一些 View 狀態的話,記住這個順序,和字母代表的意思,就可以一眼看到View的狀態,是非常不錯的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章