Androidにおけるinflate()の原理解析

15649 ワード

いろいろ考えて、自分のブログを书いてみることにしました.初めてブログを书きました.何か悪いところがあったらご了承ください.间违ったところを教えてください.ok!くだらないことは言わないで、直接本題に入ります!
今日はinflate()メソッドのダイナミックロードレイアウトについて説明したいと思います.まずAndroidの基本的な使い方を教えてください!
LayoutInflater inflater = LayoutInflater.from(context);//  LayoutInflater  
//    ,layoutID layout         ;                    ,  ,   null
View view = inflater.inflate(layoutId, null);

レイアウトをロードする一般的な方法もあります.
//    ,layoutID layout         ;                    ,  ,   null
View view = View.inflate(context, layoutId, null);

これでres/layout/のレイアウトviewが得られます
しかし、第2の方法の本質は実は第1の方法を使って、ただそれをカプセル化しただけです.
一般的な例は、ListViewでAdapterを設定する場合、カスタムitemレイアウトを使用する場合に使用されます.
余談ですが、以前カスタムitemレイアウトでAdapterを書いていたとき、レイアウトのルートノードをRelativeLayoutに設定し、実行時に不思議なエラーが報告されていることに気づき、その後Linearlayoutに変更すれば大丈夫で、Android自体のどこかのバグか分かりません.つまり、ここで注意したいのは、いつレイアウトでルートノードをRelativLayoutに設定し、javaコードにレイアウトinflate()を入力し、実行エラーが検出されて終了した場合、私が言ったような問題が発生したかどうかです.
ダイナミックロードレイアウトinflate()の一般的な方法を説明した後、ソースコードの観点から解析してみます.
まずビューを見てinflate()のソースコード
public static View inflate(Context context, int resource, ViewGroup root) {
        LayoutInflater factory = LayoutInflater.from(context);
        return factory.inflate(resource, root);
    }

ここから分かるようにinflate()はLayoutInflater.inflate()がカプセル化されています.結局LayoutInflaterのinflate()に戻り、ソースコードを見てみましょう.
public View inflate(int resource, ViewGroup root) {
        return inflate(resource, root, root != null);
    }
public View inflate(int resource, ViewGroup root, boolean attachToRoot) {
        if (DEBUG) System.out.println("INFLATING from resource: " + resource);
        XmlResourceParser parser = getContext().getResources().getLayout(resource);
        try {
            return inflate(parser, root, attachToRoot);
        } finally {
            parser.close();
        }
    }

この方法では、まずレイアウトファイルresource(レイアウトファイルがxml形式であることはよく知られている)のxmlのpull解析器parserを取得します(具体的にどのように取得したのかについては、興味のある子供靴は自分で関連資料を探して自分で勉強することができますが、ここではあまり説明しません).次に、05行目のメソッドinflate(parser,root,attachToRoot)に着き、ソースコードは以下の通りです.
public View inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot) {
        synchronized (mConstructorArgs) {
            Trace.traceBegin(Trace.TRACE_TAG_VIEW, "inflate");

            final AttributeSet attrs = Xml.asAttributeSet(parser);
            Context lastContext = (Context)mConstructorArgs[0];
            mConstructorArgs[0] = mContext;
            View result = root;

            try {
/***1.start***/					
                // Look for the root node.
				//  parser.next()          , type==XmlPullParser.START_TAG,
				//     
                int type;
                while ((type = parser.next()) != XmlPullParser.START_TAG &&
                        type != XmlPullParser.END_DOCUMENT) {
                    // Empty
                }
				//    START_TAG,  END_DOCUMENT,       
                if (type != XmlPullParser.START_TAG) {
                    throw new InflateException(parser.getPositionDescription()
                            + ": No start tag found!");
                }
				//      ,          RelativeLayout LinearLayout 。。。
                final String name = parser.getName();
 /***1.end***/               
                if (DEBUG) {
                    System.out.println("**************************");
                    System.out.println("Creating root view: "
                            + name);
                    System.out.println("**************************");
                }
				//     merge  ,         , root   , attachToRoot  true
				//      
                if (TAG_MERGE.equals(name)) {
                    if (root == null || !attachToRoot) {
                        throw new InflateException(" 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 {
/***2.start***/
						//                ,     
						//             view,            
                        temp = createViewFromTag(root, name, attrs);
/***2.end***/
                    }

                    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");
                    }
/***3.start***/
                    // Inflate all children under temp
					//     parser  ,      temp        ,
					//     ,   temp               ,         
                    rInflate(parser, temp, attrs, true);
/***3.end***/
                    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;
            }

            Trace.traceEnd(Trace.TRACE_TAG_VIEW);

            return result;
        }
    }

まずルートラベル(1.start---->1.endのコード)を取得し、ルートラベルのタイプによって異なる動作を行い、一般的には2.start--->2.endでのtemp=createViewFromTag(root,name,attrs);ルートラベルのビューtempを作成し、最後にこのtempビューでparserを解析し続け、tempビューで追加サブビュー(3.start--->3.endのrInflate(parser,temp,attrs,true)を再帰的に作成する.)
ラベル名に基づいてビューを作成する方法を説明します.createView FromTag()ソースコードは次のとおりです.
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;
        }
    }

ここのコードは多くなく、異なるファクトリクラスとラベル名に基づいてビューが作成されますが、どこのonCreateViewでもソースコードを見て、最終的には22行のcreateView()を呼び出してビューを作成します.
ソースコードは次のとおりです.
public final View createView(String name, String prefix, AttributeSet attrs)
            throws ClassNotFoundException, InflateException {

/***1.start**/
		//       sConstructorMap      name    
        Constructor extends View> constructor = sConstructorMap.get(name);
        Class extends View> clazz = null;

        try {
            Trace.traceBegin(Trace.TRACE_TAG_VIEW, name);

			/**       ,            ,           class  ,
			*         ,     sConstructorMap 
			*/
            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);
                    }
                }
            }

/***1.end***/
			Object[] args = mConstructorArgs;
            args[1] = attrs;

/***2.start***/
            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;
/***2.end***/
        } 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 ? "" : clazz.getName()));
            ie.initCause(e);
            throw ie;
        } finally {
            Trace.traceEnd(Trace.TRACE_TAG_VIEW);
        }
    }

ここのコードは理解に難くありません.まず1にあります.start--->1.endでクラスのコンストラクタを取得し、2.start--->2.endは、インスタンスを作成するオブジェクトを取り出して返す、ラベル名に基づいて作成されたビューを得ることができる.さて、前のinflate()メソッドの3に戻ります.stat--->3.endでのrInflate()メソッドでは、そのソースコードは次のとおりです.
void rInflate(XmlPullParser parser, View parent, final AttributeSet attrs,
            boolean finishInflate) throws XmlPullParserException, IOException {

        final int depth = parser.getDepth();
        int type;

        while (((type = parser.next()) != XmlPullParser.END_TAG ||
                parser.getDepth() > depth) && type != XmlPullParser.END_DOCUMENT) {

            if (type != XmlPullParser.START_TAG) {
                continue;
            }
/***1.start***/
			//         
            final String name = parser.getName();
/***1.end***/
            if (TAG_REQUEST_FOCUS.equals(name)) {
                parseRequestFocus(parser, parent);
            } else if (TAG_INCLUDE.equals(name)) {
                if (parser.getDepth() == 0) {
                    throw new InflateException(" cannot be the root element");
                }
                parseInclude(parser, parent, attrs);
            } else if (TAG_MERGE.equals(name)) {
                throw new InflateException(" must be the root element");
            } else if (TAG_1995.equals(name)) {
                final View view = new BlinkLayout(mContext, attrs);
                final ViewGroup viewGroup = (ViewGroup) parent;
                final ViewGroup.LayoutParams params = viewGroup.generateLayoutParams(attrs);
                rInflate(parser, view, attrs, true);
                viewGroup.addView(view, params);                
            } else {
/***2.stat***/
                final View view = createViewFromTag(parent, name, attrs);
                final ViewGroup viewGroup = (ViewGroup) parent;
                final ViewGroup.LayoutParams params = viewGroup.generateLayoutParams(attrs);
                rInflate(parser, view, attrs, true);
                viewGroup.addView(view, params);
/***2.end***/
            }
        }
		//     ,        ,     parent     ,     
        if (finishInflate) parent.onFinishInflate();
    }

まず1を見ます.start--->1.end出は、メソッドにおけるparserのサブタグ署名を取得し、そのサブタグに基づいて2に署名する.start--->2.endでは、メソッドcreateViewFromTagに基づいて新しいサブビューviewを作成し、rInflateに戻ってサブビューviewのサブビューを作成し、最後に親ビューparentに追加します.ここで再帰に使うので、理解しにくいかもしれませんが、仁兄の皆さんがゆっくりと論理を味わう必要があります.
これで、レイアウト全体のロードが完了しました!