借鉴出处:
1.http://blog.csdn.net/guolin_blog/article/details/12921889
2.http://blog.csdn.net/lmj623565791/article/details/38171465
3.http://www.cnblogs.com/top5/archive/2012/05/04/2482328.html
4.http://www.2cto.com/kf/201407/313054.html
layoutInflater简介
简介:
在实际开发中LayoutInflater这个类还是非常有用的,它的作用类似于findViewById()。不同点是LayoutInflater是用来找res/layout/下的xml布局文件,并且实例化;而findViewById()是找xml布局文件下的具体widget控件(如Button、TextView等)。
具体作用:
1、对于一个没有被载入或者想要动态载入的界面,都需要使用LayoutInflater.inflate()来载入;
2、对于一个已经载入的界面,就可以使用Activiyt.findViewById()方法来获得其中的界面元素。
LayoutInflater 是一个抽象类,在文档中如下声明:
public abstract class LayoutInflater extends Object
获得 LayoutInflater 实例的三种方式
1. LayoutInflater inflater = getLayoutInflater();//调用Activity的getLayoutInflater()
2. LayoutInflater inflater = LayoutInflater.from(context);
3. LayoutInflater inflater = (LayoutInflater)context.getSystemService
(Context.LAYOUT_INFLATER_SERVICE);
其实,这三种方式本质是相同的,从源码中可以看出:
getLayoutInflater():
Activity 的 getLayoutInflater() 方法是调用 PhoneWindow 的getLayoutInflater()方法
看一下该源代码:
public PhoneWindow(Context context)
{
super(context);
mLayoutInflater = LayoutInflater.from(context);
}
可以看出它其实是调用 LayoutInflater.from(context)。
LayoutInflater.from(context):
public static LayoutInflater from(Context context)
{
LayoutInflater LayoutInflater = (LayoutInflater) context.getSystemService
(Context.LAYOUT_INFLATER_SERVICE);
if (LayoutInflater == null)
{
throw new AssertionError("LayoutInflater not found.");
}
return LayoutInflater;
}
可以看出它其实调用 context.getSystemService()。
结论:所以这三种方式最终本质是都是调用的Context.getSystemService()。
inflate 方法
通过 sdk 的 api 文档,可以知道该方法有以下几种过载形式,返回值均是 View 对象,如下:
1.public View inflate (int resource, ViewGroup root)
2.public View inflate (XmlPullParser parser, ViewGroup root)
3.public View inflate (XmlPullParser parser, ViewGroup root, boolean attachToRoot)
4.public View inflate (int resource, ViewGroup root, boolean attachToRoot)
注意:
1·inflate 方法与 findViewById 方法不同;
2.·inflater 是用来找 res/layout 下的 xml 布局文件,并且实例化;
3·findViewById() 是找具体 xml 布局文件中的具体 widget 控件(如:Button、TextView 等)。
布局加载:在Activity中调用setContentView()方法来完成加载布局。然而setContentView()方法的内部则是使用LayoutInflater来加载布局的,只不过这部分源码是internal的,不太容易查看到。
LayoutInflater的使用
要使用layoutInflater,那么必须要先获取layoutInflater的实例
1. LayoutInflater layoutInflater = LayoutInflater.from(context);
2. LayoutInflater layoutInflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
获取layoutInflater的实例之后,调用它的inflate()方法来加载布局
layoutInflater.inflate(resourceId, root);
inflate()方法一般接收两个参数:
1. 要加载的布局id。
2. 是指给该布局的外部再嵌套一层父布局,如果不需要就直接传null。
实例(代码及截图):
1.创建MainAcivity布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/main_layout"
android:layout_width="match_parent"
android:layout_height="match_parent" >
</LinearLayout>
2.创建任意控件布局或其他类型布局
(我创建的是一个button布局,就是直接使用button,不使用LinearLayout之类的布局)
<?xml version="1.0" encoding="utf-8"?>
<Button xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="100dp"
android:text="button">
</Button>
3.在MainActivity中加载创建好的button布局
private LinearLayout layout;
layout = (LinearLayout)findViewById(R.id.main);
LayoutInflater layoutinflater=LayoutInflater.from(this);
View button = layoutinflater.inflate(R.layout.layout,null);
layout.addView(button);
借鉴一下:
运行截图:
可以看到button已经加载进去了;
详解LayoutInflater的工作机制
1.不管使用的哪个inflate()方法的重载,最终都会辗转调用到LayoutInflater的如下代码中:
public View inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot) {
synchronized (mConstructorArgs) {
final AttributeSet attrs = Xml.asAttributeSet(parser);
mConstructorArgs[0] = mContext;
View result = root;
try {
int type;
while ((type = parser.next()) != XmlPullParser.START_TAG &&
type != XmlPullParser.END_DOCUMENT) {
}
if (type != XmlPullParser.START_TAG) {
throw new InflateException(parser.getPositionDescription()
+ ": No start tag found!");
}
final String name = parser.getName();
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);
} else {
View temp = createViewFromTag(name, attrs);
ViewGroup.LayoutParams params = null;
if (root != null) {
params = root.generateLayoutParams(attrs);
if (!attachToRoot) {
temp.setLayoutParams(params);
}
}
rInflate(parser, temp, attrs);
if (root != null && attachToRoot) {
root.addView(temp, params);
}
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;
}
return result;
}
}
结论:LayoutInflater其实就是使用Android提供的pull解析方式来解析布局文件的。
pull解析:
1.http://www.cnblogs.com/hxsyl/p/3652058.html
2.http://www.jb51.net/article/85781.htm
3.http://www.jb51.net/article/82092.htm
此处调用了createViewFromTag()这个方法,并把节点名和参数传了进去。它是用于根据节点名来创建View对象的。在createViewFromTag()方法的内部又会去调用createView()方法,然后使用反射的方式创建出View的实例并返回。
private void rInflate(XmlPullParser parser, View parent, final AttributeSet attrs)
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;
}
final String name = parser.getName();
if (TAG_REQUEST_FOCUS.equals(name)) {
parseRequestFocus(parser, parent);
} else if (TAG_INCLUDE.equals(name)) {
if (parser.getDepth() == 0) {
throw new InflateException("<include /> cannot be the root element");
}
parseInclude(parser, parent, attrs);
} else if (TAG_MERGE.equals(name)) {
throw new InflateException("<merge /> must be the root element");
} else {
final View view = createViewFromTag(name, attrs);
final ViewGroup viewGroup = (ViewGroup) parent;
final ViewGroup.LayoutParams params = viewGroup.generateLayoutParams(attrs);
rInflate(parser, view, attrs);
viewGroup.addView(view, params);
}
}
parent.onFinishInflate();
}
此处行同样是调用createViewFromTag()方法来创建View的实例,然后接下来还会递归调用rInflate()方法来查找这个View下的子元素,每次递归完成后则将这个View添加到父布局当中。
结论:整个布局文件都解析完成后就形成了一个完整的DOM结构,最终会把最顶层的根布局返回。
不懂的请参照大神的解析:http://blog.csdn.net/guolin_blog/article/details/12921889
开发实例:
我的百度云链接:http://pan.baidu.com/s/1bpHwRxX