亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android加載View中Background詳解

2020-04-11 11:39:32
字體:
來源:轉載
供稿:網友

對大多數Android的開發者來說,最經常的操作莫過于對界面進行布局,View中背景圖片的加載是最經常做的。但是我們很少關注這個過程,這篇文章主要解析view中背景圖片加載的流程。了解view中背景圖片的加載(資源的加載)可以讓我們對資源加載的過程進行一些優化,另外當需要進行整個應用的換膚時,也可以更得心應手。

View圖片的加載,我們最常見的就是通過在XML文件當中進行drawable的設置,然后讓Android系統幫我們完成,或者手動寫代碼加載成Bitmap,然后加載到View上。這篇文章主要分析Android在什么時候以及怎么幫我們完成背景圖片的加載的,那么我們就從Activity.setContentView還是LayoutInflater.inflate(...)方法開始分析。

不管是從Activity.setContentView(...)還是LayoutInflater.inflate(...)方法進行View的初始化,最終都會到達LayoutInflater.inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot)這個方法中。在這里我們主要關注View的背景圖片加載,對于XML如何解析和加載就放過了。

復制代碼 代碼如下:

    public View inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot) {
        synchronized (mConstructorArgs) {
            final AttributeSet attrs = Xml.asAttributeSet(parser);
            Context lastContext = (Context)mConstructorArgs[0];
            mConstructorArgs[0] = mContext;
            View result = root;
            try {
                // Look for the root node.
                int type;
                while ((type = parser.next()) != XmlPullParser.START_TAG &&
                        type != XmlPullParser.END_DOCUMENT) {
                    // Empty
                }
                if (type != XmlPullParser.START_TAG) {
                    throw new InflateException(parser.getPositionDescription()
                            + ": No start tag found!");
                }
                final String name = parser.getName();
                if (DEBUG) {
                    System.out.println("**************************");
                    System.out.println("Creating root view: "
                            + name);
                    System.out.println("**************************");
                }
                if (TAG_MERGE.equals(name)) {
                    if (root == null || !attachToRoot) {
                        throw new InflateException("<merge /> can be used only with a valid "
                                + "ViewGroup root and attachToRoot=true");
                    }
                    rInflate(parser, root, attrs, false);
                } else {
                    // Temp is the root view that was found in the xml
                    View temp;
                    if (TAG_1995.equals(name)) {
                        temp = new BlinkLayout(mContext, attrs);
                    } else {
                        temp = createViewFromTag(root, name, attrs);
                    }
                    ViewGroup.LayoutParams params = null;
                    if (root != null) {
                        if (DEBUG) {
                            System.out.println("Creating params from root: " +
                                    root);
                        }
                        // Create layout params that match root, if supplied
                        params = root.generateLayoutParams(attrs);
                        if (!attachToRoot) {
                            // Set the layout params for temp if we are not
                            // attaching. (If we are, we use addView, below)
                            temp.setLayoutParams(params);
                        }
                    }
                    if (DEBUG) {
                        System.out.println("-----> start inflating children");
                    }
                     // Inflate all children under temp
                    rInflate(parser, temp, attrs, true);
                    if (DEBUG) {
                        System.out.println("-----> done inflating children");
                    }
                    // We are supposed to attach all the views we found (int temp)
                    // to root. Do that now.
                    if (root != null && attachToRoot) {
                        root.addView(temp, params);
                    }
                    // Decide whether to return the root that was passed in or the
                    // top view found in xml.
                    if (root == null || !attachToRoot) {
                        result = temp;
                    }
                }
            } catch (XmlPullParserException e) {
                InflateException ex = new InflateException(e.getMessage());
                ex.initCause(e);
                throw ex;
            } catch (IOException e) {
                InflateException ex = new InflateException(
                        parser.getPositionDescription()
                        + ": " + e.getMessage());
                ex.initCause(e);
                throw ex;
            } finally {
                // Don't retain static reference on context.
                mConstructorArgs[0] = lastContext;
                mConstructorArgs[1] = null;
            }
            return result;
        }
    }
上面這么長一串代碼,其實思路很清晰,就是針對XML文件進行解析,然后根據XML解析出的每一個節點進行View的初始化,緊接著將View的Layout參數設置到View上,然后將View添加到它的父控件上。
為了了解View是怎么被加載出來的,我們只需要了解
 temp = createViewFromTag(root, name, attrs);
跟進去看看。
    /*
     * default visibility so the BridgeInflater can override it.
     */
    View createViewFromTag(View parent, String name, AttributeSet attrs) {
        if (name.equals("view")) {
            name = attrs.getAttributeValue(null, "class");
        }
        if (DEBUG) System.out.println("******** Creating view: " + name);
        try {
            View view;
            if (mFactory2 != null) view = mFactory2.onCreateView(parent, name, mContext, attrs);
            else if (mFactory != null) view = mFactory.onCreateView(name, mContext, attrs);
            else view = null;
            if (view == null && mPrivateFactory != null) {
                view = mPrivateFactory.onCreateView(parent, name, mContext, attrs);
            }
            if (view == null) {
                if (-1 == name.indexOf('.')) {
                    view = onCreateView(parent, name, attrs);
                } else {
                    view = createView(name, null, attrs);
                }
            }
            if (DEBUG) System.out.println("Created view is: " + view);
            return view;
        } catch (InflateException e) {
            throw e;
        } catch (ClassNotFoundException e) {
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Error inflating class " + name);
            ie.initCause(e);
            throw ie;
        } catch (Exception e) {
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Error inflating class " + name);
            ie.initCause(e);
            throw ie;
        }
    }

上面代碼的重點在于try...Catch里的內容。try包起來的東西就是對View進行初始化,注意到上面代碼中有幾個Factory,這些Factory可以在View進行初始化,也就是說其實我們可以在這里干預View的初始化。從上面代碼我們可以知道,如果我們自定義了一個Factory,那么當前要初始化的View會優先被我們自定義的Factory初始化,而不通過系統默認的Factory初始化。那么如果我們要自定義Factory,應該在哪里定義呢?容易想到,Factory必須要趕在資源加載前自定義完成,所以我們應該在onCreate(...)的this.setContentView(...)之前設置LayoutInflater.Factory。

  getLayoutInflater().setFactory(factory);
接下來我們看到上面函數里面的

復制代碼 代碼如下:

  if (-1 == name.indexOf('.')) {
        view = onCreateView(parent, name, attrs);
    } else {
        view = createView(name, null, attrs);
    }

這段函數就是對View進行初始化,有兩種情況,一種是系統自帶的View,它在

 if (-1 == name.indexOf('.'))
這里面進行初始化,因為如果是系統自帶的View,傳入的那么一般不帶系統的前綴"android.view."。另一個分支初始化的是我們自定義的View。我們跟進onCreateView看看。

復制代碼 代碼如下:

  protected View onCreateView(String name, AttributeSet attrs)
            throws ClassNotFoundException {
        return createView(name, "android.view.", attrs);
    }
    public final View createView(String name, String prefix, AttributeSet attrs)
            throws ClassNotFoundException, InflateException {
        Constructor<? extends View> constructor = sConstructorMap.get(name);
        Class<? extends View> clazz = null;
        try {
            if (constructor == null) {
                // Class not found in the cache, see if it's real, and try to add it
                clazz = mContext.getClassLoader().loadClass(
                        prefix != null ? (prefix + name) : name).asSubclass(View.class);
                if (mFilter != null && clazz != null) {
                    boolean allowed = mFilter.onLoadClass(clazz);
                    if (!allowed) {
                        failNotAllowed(name, prefix, attrs);
                    }
                }
                constructor = clazz.getConstructor(mConstructorSignature);
                sConstructorMap.put(name, constructor);
            } else {
                // If we have a filter, apply it to cached constructor
                if (mFilter != null) {
                    // Have we seen this name before?
                    Boolean allowedState = mFilterMap.get(name);
                    if (allowedState == null) {
                        // New class -- remember whether it is allowed
                        clazz = mContext.getClassLoader().loadClass(
                                prefix != null ? (prefix + name) : name).asSubclass(View.class);
                        boolean allowed = clazz != null && mFilter.onLoadClass(clazz);
                        mFilterMap.put(name, allowed);
                        if (!allowed) {
                            failNotAllowed(name, prefix, attrs);
                        }
                    } else if (allowedState.equals(Boolean.FALSE)) {
                        failNotAllowed(name, prefix, attrs);
                    }
                }
            }
            Object[] args = mConstructorArgs;
            args[1] = attrs;
            final View view = constructor.newInstance(args);
            if (view instanceof ViewStub) {
                // always use ourselves when inflating ViewStub later
                final ViewStub viewStub = (ViewStub) view;
                viewStub.setLayoutInflater(this);
            }
            return view;
        } catch (NoSuchMethodException e) {
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Error inflating class "
                    + (prefix != null ? (prefix + name) : name));
            ie.initCause(e);
            throw ie;
        } catch (ClassCastException e) {
            // If loaded class is not a View subclass
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Class is not a View "
                    + (prefix != null ? (prefix + name) : name));
            ie.initCause(e);
            throw ie;
        } catch (ClassNotFoundException e) {
            // If loadClass fails, we should propagate the exception.
            throw e;
        } catch (Exception e) {
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Error inflating class "
                    + (clazz == null ? "<unknown>" : clazz.getName()));
            ie.initCause(e);
            throw ie;
        }
    }

從onCreateView(...)中我們知道,其實createViewFromTag(...)中對View的初始化最終都是通過createView(...)這個函數進行初始化的,不同只在于系統控件需要通過onCreateView(...)加上前綴,以便類加載器(ClassLoader)正確地通過類所在的包初始化這個類。createView(...)這個函數的思路很清晰,不看catch里面的內容,try里面開頭的兩個分支就是用來將所要用的類構造函數提取出來,Android系統會對使用過的類構造函數進行緩存,因為像TextView這些常用的控件可能會被使用很多次。接下來,就是通過類構造函數對View進行初始化了。我們注意到傳入構造函數的mConstructorArgs是一個包含兩個元素的數組。

 final Object[] mConstructorArgs = new Object[2];
那么我們就很清楚了,它就是調用系統控件中對應兩個參數的構造函數。為了方便,我們就從最基礎的View進行分析。

復制代碼 代碼如下:

  public View(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }
    public View(Context context, AttributeSet attrs, int defStyle) {
     this(context);
     TypedArray a = context.obtainStyledAttributes(attrs, com.android.internal.R.styleable.View,
             defStyle, 0);
     Drawable background = null;
     int leftPadding = -1;
     int topPadding = -1;
     int rightPadding = -1;
     int bottomPadding = -1;
     int startPadding = UNDEFINED_PADDING;
     int endPadding = UNDEFINED_PADDING;
     int padding = -1;
     int viewFlagValues = 0;
     int viewFlagMasks = 0;
     boolean setScrollContainer = false;
     int x = 0;
     int y = 0;
     float tx = 0;
     float ty = 0;
     float rotation = 0;
     float rotationX = 0;
     float rotationY = 0;
     float sx = 1f;
     float sy = 1f;
     boolean transformSet = false;
     int scrollbarStyle = SCROLLBARS_INSIDE_OVERLAY;
     int overScrollMode = mOverScrollMode;
     boolean initializeScrollbars = false;
     boolean leftPaddingDefined = false;
     boolean rightPaddingDefined = false;
     boolean startPaddingDefined = false;
     boolean endPaddingDefined = false;
     final int targetSdkVersion = context.getApplicationInfo().targetSdkVersion;
     final int N = a.getIndexCount();
      for (int i = 0; i < N; i++) {
          int attr = a.getIndex(i);
          switch (attr) {
              case com.android.internal.R.styleable.View_background:
                  background = a.getDrawable(attr);
                  break;
              case com.android.internal.R.styleable.View_padding:
                  padding = a.getDimensionPixelSize(attr, -1);
                  mUserPaddingLeftInitial = padding;
                  mUserPaddingRightInitial = padding;
                  leftPaddingDefined = true;
                  rightPaddingDefined = true;
                  break;
   //省略一大串無關的函數
 }

由于我們只關注View中的背景圖是怎么加載的,注意這個函數其實就是遍歷AttributeSet attrs這個東西,然后對View的各個屬性進行初始化。我們直接進入

 background = a.getDrawable(attr);
這里看看(TypedArray.getDrawable)。

復制代碼 代碼如下:

    public Drawable getDrawable(int index) {
        final TypedValue value = mValue;
        if (getValueAt(index*AssetManager.STYLE_NUM_ENTRIES, value)) {
            if (false) {
                System.out.println("******************************************************************");
                System.out.println("Got drawable resource: type="
                                   + value.type
                                   + " str=" + value.string
                                   + " int=0x" + Integer.toHexString(value.data)
                                   + " cookie=" + value.assetCookie);
                System.out.println("******************************************************************");
            }
            return mResources.loadDrawable(value, value.resourceId);
        }
        return null;
    }

我們發現它調用mResources.loadDrawable(...),進去看看。

復制代碼 代碼如下:

    /*package*/ Drawable loadDrawable(TypedValue value, int id)
            throws NotFoundException {
        if (TRACE_FOR_PRELOAD) {
            // Log only framework resources
            if ((id >>> 24) == 0x1) {
                final String name = getResourceName(id);
                if (name != null) android.util.Log.d("PreloadDrawable", name);
            }
        }
        boolean isColorDrawable = false;
        if (value.type >= TypedValue.TYPE_FIRST_COLOR_INT &&
                value.type <= TypedValue.TYPE_LAST_COLOR_INT) {
            isColorDrawable = true;
        }
        final long key = isColorDrawable ? value.data :
                (((long) value.assetCookie) << 32) | value.data;
        Drawable dr = getCachedDrawable(isColorDrawable ? mColorDrawableCache : mDrawableCache, key);
        if (dr != null) {
            return dr;
        }
        Drawable.ConstantState cs = isColorDrawable
                ? sPreloadedColorDrawables.get(key)
                : (sPreloadedDensity == mConfiguration.densityDpi
                        ? sPreloadedDrawables.get(key) : null);
        if (cs != null) {
            dr = cs.newDrawable(this);
        } else {
            if (isColorDrawable) {
                dr = new ColorDrawable(value.data);
            }
            if (dr == null) {
                if (value.string == null) {
                    throw new NotFoundException(
                            "Resource is not a Drawable (color or path): " + value);
                }
                String file = value.string.toString();
                if (TRACE_FOR_MISS_PRELOAD) {
                    // Log only framework resources
                    if ((id >>> 24) == 0x1) {
                        final String name = getResourceName(id);
                        if (name != null) android.util.Log.d(TAG, "Loading framework drawable #"
                                + Integer.toHexString(id) + ": " + name
                                + " at " + file);
                    }
                }
                if (DEBUG_LOAD) Log.v(TAG, "Loading drawable for cookie "
                        + value.assetCookie + ": " + file);
                if (file.endsWith(".xml")) {
                    try {
                        XmlResourceParser rp = loadXmlResourceParser(
                                file, id, value.assetCookie, "drawable");
                        dr = Drawable.createFromXml(this, rp);
                        rp.close();
                    } catch (Exception e) {
                        NotFoundException rnf = new NotFoundException(
                            "File " + file + " from drawable resource ID #0x"
                            + Integer.toHexString(id));
                        rnf.initCause(e);
                        throw rnf;
                    }
                } else {
                    try {
                        InputStream is = mAssets.openNonAsset(
                                value.assetCookie, file, AssetManager.ACCESS_STREAMING);
        //                System.out.println("Opened file " + file + ": " + is);
                        dr = Drawable.createFromResourceStream(this, value, is,
                                file, null);
                        is.close();
        //                System.out.println("Created stream: " + dr);
                    } catch (Exception e) {
                        NotFoundException rnf = new NotFoundException(
                            "File " + file + " from drawable resource ID #0x"
                            + Integer.toHexString(id));
                        rnf.initCause(e);
                        throw rnf;
                    }
                }
            }
        }
        if (dr != null) {
            dr.setChangingConfigurations(value.changingConfigurations);
            cs = dr.getConstantState();
            if (cs != null) {
                if (mPreloading) {
                    if (verifyPreloadConfig(value, "drawable")) {
                        if (isColorDrawable) {
                            sPreloadedColorDrawables.put(key, cs);
                        } else {
                            sPreloadedDrawables.put(key, cs);
                        }
                    }
                } else {
                    synchronized (mTmpValue) {
                        //Log.i(TAG, "Saving cached drawable @ #" +
                        //        Integer.toHexString(key.intValue())
                        //        + " in " + this + ": " + cs);
                        if (isColorDrawable) {
                            mColorDrawableCache.put(key, new WeakReference<Drawable.ConstantState>(cs));
                        } else {
                            mDrawableCache.put(key, new WeakReference<Drawable.ConstantState>(cs));
                        }
                    }
                }
            }
        }
        return dr;
    }

就是這個函數了,所有View的背景的加載都在這里了。這個函數的邏輯就比較復雜了,大體說來就是根據背景的類型(純顏色、定義在XML文件中的,或者是一張靜態的背景),如果緩存里面有,就直接用緩存里的。

總結一下,經過上面的分析,我們知道了,Android就是在Activity.setContentView(...)中為我們進行資源文件的加載,精確到具體的函數的話,資源文件的加載就是在每一個被初始化的View的構造函數中進行加載的。

以上就是本文的全部內容了,希望對大家能夠有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
成人免费网站在线看| 久久伊人91精品综合网站| 国产精品高潮呻吟久久av黑人| 欧美激情精品久久久久久久变态| 成人97在线观看视频| 国外日韩电影在线观看| 亚洲天堂av在线播放| 亚洲美女av在线播放| 亚洲精品国产品国语在线| 国产一区二区三区免费视频| 日韩的一区二区| 国产91精品久久久久久| 97在线免费视频| 亚洲97在线观看| 91日韩在线视频| 亚洲韩国青草视频| 69影院欧美专区视频| 久久精品国亚洲| 国产在线日韩在线| 亚洲欧美在线免费观看| 久久久久国产一区二区三区| 亚洲国产欧美一区二区三区同亚洲| 日韩精品在线观看网站| 国产精品国产亚洲伊人久久| 成人免费淫片视频软件| 日韩不卡中文字幕| 91夜夜揉人人捏人人添红杏| 欧美日韩高清在线观看| 91九色综合久久| 欧美成人免费全部观看天天性色| 日韩高清中文字幕| 国产亚洲精品一区二区| 亚洲aaa激情| 色久欧美在线视频观看| 91免费看片网站| 亚洲欧美激情四射在线日| 亚洲高清久久网| 国产欧美久久一区二区| 国产日韩精品电影| 97成人超碰免| 亚洲男人av电影| 国产精品成人一区| 欧美精品手机在线| 成人在线观看视频网站| 久久久精品视频成人| 91精品久久久久久久久不口人| 欧美亚洲视频一区二区| 国产精品久久久久999| 成人中文字幕+乱码+中文字幕| 91久久国产精品| 97在线视频国产| 91视频国产精品| 2021国产精品视频| 国产91露脸中文字幕在线| 91精品久久久久久久久久另类| 日韩二区三区在线| 国产最新精品视频| 亚洲第一精品久久忘忧草社区| 亚洲欧洲一区二区三区在线观看| 精品亚洲一区二区| 国产精品91久久久| 亚洲电影成人av99爱色| 国产欧美中文字幕| 亚洲娇小xxxx欧美娇小| 亚洲美女av在线播放| 久久精品视频在线观看| 欧美精品情趣视频| 欧美wwwwww| 久久影视电视剧免费网站清宫辞电视| 成人在线播放av| 国内精品久久久久久久| 亚洲一区二区久久| 久久成人亚洲精品| 欧美性一区二区三区| 中文字幕一区电影| 国产日韩视频在线观看| 久久精品精品电影网| 亚洲色图偷窥自拍| 亚洲自拍欧美另类| 国产精品v日韩精品| 国产精品看片资源| 91在线无精精品一区二区| 久久久综合免费视频| 96精品久久久久中文字幕| 日韩激情av在线播放| 欧美在线视频一区二区| 懂色av一区二区三区| 日韩成人激情影院| 欧美性69xxxx肥| 亚洲va欧美va国产综合久久| 久久亚洲成人精品| 精品欧美激情精品一区| 亚洲影院污污.| 97成人在线视频| 97在线观看视频国产| 日韩精品欧美国产精品忘忧草| 欧美色道久久88综合亚洲精品| 成人午夜激情免费视频| 国产精品老女人精品视频| 欧美成人免费va影院高清| 国产成人a亚洲精品| 亚洲国模精品私拍| 日韩一区二区欧美| 亚洲精品网址在线观看| 国产精品第三页| 国产精品高潮呻吟久久av黑人| 亚洲大胆人体在线| 一区二区欧美亚洲| 91探花福利精品国产自产在线| www.日韩av.com| 欧美亚洲激情视频| 伊人伊成久久人综合网小说| 精品国产一区二区在线| 亚洲网址你懂得| 日韩成人在线视频| 成人午夜一级二级三级| 久久久这里只有精品视频| 精品国产依人香蕉在线精品| 黑人狂躁日本妞一区二区三区| 久久久久久久久中文字幕| 成人免费网站在线| 日韩国产在线播放| 亚洲视频视频在线| 亚洲热线99精品视频| 久久好看免费视频| 韩国美女主播一区| 久久久久久久香蕉网| 国产一区二区在线免费| 午夜精品久久久久久久99黑人| 亚洲欧美日韩国产成人| 亚洲精品在线不卡| 日韩在线精品视频| 久久久久久com| 亚洲福利在线播放| 97精品视频在线观看| 欧美性猛交xxxx乱大交3| 亚洲女人天堂网| 亚洲精品美女在线观看播放| 91精品在线影院| 久久久国产影院| 亚洲乱码国产乱码精品精| 欧美极品欧美精品欧美视频| 欧美激情亚洲另类| 欧美大全免费观看电视剧大泉洋| 奇米四色中文综合久久| 欧美视频在线免费看| 亚洲国产精品国自产拍av秋霞| 国产日韩欧美中文| 欧美专区在线观看| 中文字幕综合一区| 精品久久香蕉国产线看观看gif| 欧美一区二区.| 高清日韩电视剧大全免费播放在线观看| 欧美视频在线免费看| 亚洲一区二区三区777| 欧美老肥婆性猛交视频| 亚洲男人第一网站| 久久久www成人免费精品张筱雨| 亚洲香蕉在线观看| 久久久久久久香蕉网| 日韩在线免费av| 成人在线播放av| 中文字幕日韩av综合精品| 色伦专区97中文字幕|