我们知道,在Android系统中,每一个应用程序一般都会配置很多资源,用来适配不同密度、大小和方向的屏幕,以及适配不同的国家、地区和语言等等。这些资源是在应用程序运行时自动根据设备的当前配置信息进行适配的。这也就是说,给定一个相同的资源ID,在不同的设备配置之下,查找到的可能是不同的资源。这个资源查找过程对应用程序来说,是完全透明的。在本文中,我们就详细分析资源管理框架是如何根据ID来查找资源的。
从前面Android应用程序资源管理器(Asset Manager)的创建过程分析一文可以知道,Android资源管理框架实际就是由AssetManager
和Resources
两个类来实现的。其中,Resources
类可以根据ID来查找资源
,而AssetManager
类根据文件名来查找资源
。事实上,**如果一个资源ID对应的是一个文件,那么Resources
类是先根据ID来找到资源文件名称
,然后再将该文件名称交给AssetManager类来打开对应的文件的,这个过程如图1所示。
在图1中,Resources
类根据资源ID来查到资源名称实际上也是要通过AssetManager
类来实现的,这是因为资源ID与资源名称的对应关系是由打包在APK里面的resources.arsc
文件中的。当Resources
类查找的资源对应的是一个文件的时候,它就会再次将资源名称交给AssetManager
,以便后者可以打开对应的文件,否则的话,上一步找到的资源名称就是最终的查找结果。
从前面Android应用程序资源的编译和打包过程分析一文可以知道,APK包里面的resources.arsc
文件是在编译应用程序资源的时候生成的,然后连同其它被编译的以及原生的资源一起打包在一个APK包里面。
从前面Android资源管理框架(Asset Manager)
一文又可以知道,Android应用程序资源是可以划分是很多类别的,但是从资源查找的过程来看,它们可以归结为两大类。第一类资源是不对应有文件的,而第二类资源是对应有文件的,例如,字符串资源是直接编译在resources.arsc
文件中的,而界面布局资源是在APK包里面是对应的单独的文件的。如上所述,不对应文件的资源只需要执行从资源ID到资源名称的转换即可,而对应有文件的资源还需要根据资源名称来打开对应的文件。在本文中,我们就以界面布局资源的查找过程为例,来说明Android资源管理框架查找资源的过程。
界面布局资源的查找过程
我们知道,每一个Activity组件创建的时候,它的成员函数onCreate
都会被调用,而在Activity组件的成员函数onCreate
中,我们基本上都无一例外地调用setContentView
来设置Activity组件的界面。在调用Activity组件的成员函数setContentView
的时候,需要指定一个layout类型
的资源ID
,以便Android资源管理框架可以找到指定的Xml资源文件
来填充(inflate)为Activity组件的界面。接下来,我们就从Activity类的成员函数setContentView
开始,分析Android资源管理框架查找layout
资源的过程,如图2所示。
这个过程可以分为22个步骤,接下来我们就详细分析每一个步骤。
Step 1. Activity.setContentView
public class Activity extends ContextThemeWrapper
implements LayoutInflater.Factory,
Window.Callback, KeyEvent.Callback,
OnCreateContextMenuListener, ComponentCallbacks {
......
private Window mWindow;
......
public Window getWindow() {
return mWindow;
}
.....
public void setContentView(int layoutResID) {
getWindow().setContentView(layoutResID);
}
......
}
这个函数定义在文件frameworks/base/core/java/android/app/Activity.java中。
从前面Android全面解析之Window机制一文可以知道,Activity类的成员变量mWindow
指向的是一个PhoneWindow
对象,因此,Activity
类的成员函数setContentView
实际上是调用PhoneWindow
类的成员函数setContentView
来进一步操作。
Step 2. PhoneWindow.setContentView
public class PhoneWindow extends Window implements MenuBuilder.Callback {
......
// This is the view in which the window contents are placed. It is either
// mDecor itself, or a child of mDecor where the contents go.
private ViewGroup mContentParent;
......
private LayoutInflater mLayoutInflater;
......
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor();
} else {
mContentParent.removeAllViews();
}
mLayoutInflater.inflate(layoutResID, mContentParent);
final Callback cb = getCallback();
if (cb != null) {
cb.onContentChanged();
}
}
......
}
这个函数定义在文 frameworks/base/policy/src/com/android/internal/policy/impl/PhoneWindow.java中。
PhoneWindow
类的成员变量mContentParent
用来描述一个类型为DecorView
的视图对象,或者这个类型为DecorView
的视图对象的一个子视图对象,用作UI容器。当它的值等于null
的时候,就说明当前正在处理的Activity组件的视图对象还没有创建。在这种情况下,就会调用成员函数installDecor
来创建当前正在处理的Activity组件的视图对象。否则的话,就说明是要重新设置当前正在处理的Activity组件的视图。在重新设置之前,首先调用成员变量mContentParent
所描述的一个ViewGroup对象来移除原来的UI内容。
PhoneWindow
类的成员变量mLayoutInflater
指向的是一个PhoneLayoutInflate
r对象。PhoneLayoutInflater
类是从LayoutInflater
类继承下来的,同时它也继承了LayoutInflater
类的成员函数·inflate·。通过调用·PhoneWindow·类的成员变量·mLayoutInflater·所指向的一个PhoneLayoutInflater
对象的成员函数inflate
,也就是从父类继承下来的成员函数inflate
,就可以将参数layoutResID
所描述的一个UI布局设置到mContentParent
所描述的一个视图容器中去。这样就可以将当前正在处理的Activity组件的UI创建出来。
最后,PhoneWindow
类的成员函数还会调用一个Callback
接口的成员函数onContentChanged
来通知当前正在处理的Activity组件,它的视图内容发生改变了。从前面Android应用程序窗口(Activity)的窗口对象(Window)的创建过程分析一文可以知道,每一个Activity组件都实现了一个Callback
接口,并且将这个Callback接口
设置到了与它所关联的PhoneWindow
的内部去,因此,最后调用的实际上是Activity类的成员函数onContentChanged
。
接下来,我们就继续分析LayoutInflater
类的成员函数inflate
的实现,以便可以了解Android资源管理框架是如何找到参数layoutResID
所描述的UI布局文件的。
Step 3. LayoutInflater.inflate
public abstract class LayoutInflater {
......
public View inflate(int resource, ViewGroup root) {
return inflate(resource, root, root != null);
}
......
public View inflate(int resource, ViewGroup root, boolean attachToRoot) {
......
XmlResourceParser parser = getContext().getResources().getLayout(resource);
try {
return inflate(parser, root, attachToRoot);
} finally {
parser.close();
}
}
......
}
这个函数定义在文件frameworks/base/core/java/android/view/LayoutInflater.java中。
LayoutInflater
类两个参数版本的成员函数inflate
通过调用三个参数版本的成员函数inflate
来查找参数resource
所描述的UI布局文件。
在LayoutInflater
类三个参数版本的成员函数inflate
中,首先是获得用来描述当前运行上下文环境的一个Resources
对象,然后接调用这个Resources对象
的成员函数getLayout
来查找参数resource所描述的UI布局
文件。
Resources
类的成员函数getLayout
找到了指定的UI布局文件之后,就会打开它。由于Android系统的UI布局文件是一个Xml文件,因此,Resources类
的成员函数getLayout
打开它之后,得到的是一个XmlResourceParser
对象。有了这个XmlResourceParser
对象之后,LayoutInflater
类三个参数版本的成员函数inflate
就将它传递给另外一个三个参数版本的成员函数inflate
,以便后者可以通过它来创建一个UI界面。
接下来,我们就首先分析Resources
类的成员函数getLayout
的实现,然后再分析LayoutInflater
类的另外一个三个参数版本的成员函数inflate的实现。
Step 4. Resources.getLayout
public class Resources {
......
public XmlResourceParser getLayout(int id) throws NotFoundException {
return loadXmlResourceParser(id, "layout");
}
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/Resources.java中。
Resources
类的成员函数getLayout
的实现很简单,它通过调用另外一个成员函数loadXmlResourceParser
来查找并且打开由参数id所描述的一个UI布局文件。
Step 5. Resources.loadXmlResourceParser
public class Resources {
......
/*package*/ XmlResourceParser loadXmlResourceParser(int id, String type)
throws NotFoundException {
synchronized (mTmpValue) {
TypedValue value = mTmpValue;
getValue(id, value, true);
if (value.type == TypedValue.TYPE_STRING) {
return loadXmlResourceParser(value.string.toString(), id,
value.assetCookie, type);
}
throw new NotFoundException(
"Resource ID #0x" + Integer.toHexString(id) + " type #0x"
+ Integer.toHexString(value.type) + " is not valid");
}
}
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/Resources.java中。
参数id
描述的是一个资源ID
,Resources
类的成员函数loadXmlResourceParser
首先调用另外一个成员函数getValue
来获得该资源ID所对应的资源值
,并且保存在一个类型为TypedValue
的变量value
中。在我们这个情景中,参数id
描述的是一个类型为layout的资源ID
,从前面Android应用程序资源的编译和打包过程分析一文可以知道,类型为layout
的资源ID对应的资源值
即为一个UI布局文件名称
。有了这个UI布局文件名称之后,Resources
类的成员函数loadXmlResourceParser
接着再调用另外一个四个参数版本的成员函数loadXmlResourceParser
来加载对应的UI布局文件,并且得到一个XmlResourceParser
对象返回给调用者。
注意,如果
Resources
类的成员函数getValue
没有找到与参数id
所描述的资源,或者找到的资源的值不是字符串类型
的,那么Resources
类的成员函数loadXmlResourceParser
就会抛出一个类型为NotFoundException
的异常。
接下来,我们就首先分析Resources
类的成员函数getValue
的实现,接着再分析Resources
类四个参数版本的成员函数loadXmlResourceParser
的实现。
Step 6. Resources.getValue
public class Resources {
......
/*package*/ final AssetManager mAssets;
......
public void getValue(int id, TypedValue outValue, boolean resolveRefs)
throws NotFoundException {
boolean found = mAssets.getResourceValue(id, outValue, resolveRefs);
if (found) {
return;
}
throw new NotFoundException("Resource ID #0x"
+ Integer.toHexString(id));
}
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/Resources.java中。
Resources
类的成员变量mAssets
指向的是一个AssetManager
对象,Resources
类的成员函数getValue
通过调用它的成员函数getResourceValue
来获得与参数id
所对应的资源的值。注意,如果AssetManager类的成员函数getResourceValue查找不到与参数id所对应的资源,那么Resources类的成员函数getValue就会抛出一个类型为NotFoundException
的异常。
接下来,我们就继续分析AssetManager
类的成员函数getResourceValue
的实现。
Step 7. AssetManager.getResourceValue
public final class AssetManager {
......
private StringBlock mStringBlocks[] = null;
......
/*package*/ final boolean getResourceValue(int ident,
TypedValue outValue,
boolean resolveRefs)
{
int block = loadResourceValue(ident, outValue, resolveRefs);
if (block >= 0) {
if (outValue.type != TypedValue.TYPE_STRING) {
return true;
}
outValue.string = mStringBlocks[block].get(outValue.data);
return true;
}
return false;
}
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/AssetManager.java中。
AssetManager
类的成员函数getResourceValue
通过调用另外一个成员函数loadResourceValue
来加载参数ident
所描述的资源。如果加载成功,那么结果就会保存在参数outValue
所描述的一个TypedValue
对象中,并且AssetManager
类的成员函数loadResourceValue
的返回值block大于等于0
。
从前面Android应用程序资源管理器(Asset Manager)的创建过程分析一文可以知道,AssetManager
类的成员变量mStringBlock
描述的是一个StringBlock数组
。这个StringBlock
数组中的每一个StringBlock对象
描述的都是当前应用程序使用的每一个资源索引表的资源项值字符串资源池。关于资源索引表的格式以及生成过程,可以参考前面Android应用程序资源的编译和打包过程分析。
了解了上述背景之后,我们就可以知道,当AssetManager
类的成员函数loadResourceValue
的返回值block大于等于0
的时候,实际上就表示参数ident
所描述的资源项在当前应用程序使用的第block个
资源索引表中,而当参数ident
所描述的资源项是一个字符串
时,那么就可以在第block个
资源索引表的资源项值字符串资源池中找到对应的字符串,并且保存在参数outValue
所描述的一个TypedValue
对象的成员变量string
中,以便返回给调用者使用。注意,最终得到的字符串在第block个资源索引表的资源项值字符串资源池中的位置就保存在参数outValue
所描述的一个TypedValue对象的成员变量data中。
接下来,我们就继续分析AssetManager类的成员函数loadResourceValue的实现。
Step 8. AssetManager.loadResourceValue
public final class AssetManager {
......
/** Returns true if the resource was found, filling in mRetStringBlock and
* mRetData. */
private native final int loadResourceValue(int ident, TypedValue outValue,
boolean resolve);
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/AssetManager.java中。
AssetManager
类的成员函数loadResourceValue
是一个JNI方法,它是由C++层的函数android_content_AssetManager_loadResourceValue
来实现的,如下所示:
static jint android_content_AssetManager_loadResourceValue(JNIEnv* env, jobject clazz,
jint ident,
jobject outValue,
jboolean resolve)
{
AssetManager* am = assetManagerForJavaObject(env, clazz);
if (am == NULL) {
return 0;
}
const ResTable& res(am->getResources());
Res_value value;
ResTable_config config;
uint32_t typeSpecFlags;
ssize_t block = res.getResource(ident, &value, false, &typeSpecFlags, &config);
......
uint32_t ref = ident;
if (resolve) {
block = res.resolveReference(&value, block, &ref);
......
}
return block >= 0 ? copyValue(env, outValue, &res, value, ref, block, typeSpecFlags, &config) : block;
}
这个函数定义在文件frameworks/base/core/jni/android_util_AssetManager.cpp中。
函数android_content_AssetManager_loadResourceValue
主要是执行以下五个操作:
- 调用函数
assetManagerForJavaObject
来将参数clazz
所描述的一个Java层的AssetManager对象
的成员变量mObject
转换为一个C++层的AssetManager
对象。
- 调用函数
- 调用上述得到的
C++层的AssetManager
对象的成员函数getResources
来获得一个ResTable
对象,这个ResTable
对象描述的是一个资源表。
- 调用上述得到的
- 调用上述得到的
ResTable
对象的成员函数getResource
来获得与参数ident
所对应的资源项值及其配置信息,并且保存在类型为Res_value
的变量value
以及类型为ResTable_config
的变量config
中。
- 调用上述得到的
- 如果参数
resolve
的值等于true
,那么就继续调用上述得到的ResTable
对象的成员函数resolveReference
来解析前面所得到的资源项值。
- 如果参数
- 调用函数
copyValue
将上述得到的资源项值及其配置信息拷贝到参数outValue
所描述的一个Java层的TypedValue对象
中去,返回调用者可以获得与参数ident
所对应的资源项内容。
- 调用函数
接下来,我们就主要分析第2~4操作,即AssetManager对象的成员函数getResources
以及ResTable
类的成员函数getResource
和resolveReference
的实现,以便可以了解Android应用程序资源的查找过程。
Step 9. AssetManager.getResources
const ResTable& AssetManager::getResources(bool required) const
{
const ResTable* rt = getResTable(required);
return *rt;
}
这个函数定义在文件frameworks/base/libs/utils/AssetManager.cpp中。
AssetManager
类的成员函数getResources
通过调用另外一个成员函数getResTable
来获得当前应用程序所使用的资源表,后者的实现如下所示:
const ResTable* AssetManager::getResTable(bool required) const
{
ResTable* rt = mResources;
if (rt) {
return rt;
}
// Iterate through all asset packages, collecting resources from each.
AutoMutex _l(mLock);
if (mResources != NULL) {
return mResources;
}
......
const size_t N = mAssetPaths.size();
for (size_t i=0; i<N; i++) {
Asset* ass = NULL;
ResTable* sharedRes = NULL;
bool shared = true;
const asset_path& ap = mAssetPaths.itemAt(i);
Asset* idmap = openIdmapLocked(ap);
......
if (ap.type != kFileTypeDirectory) {
if (i == 0) {
// The first item is typically the framework resources,
// which we want to avoid parsing every time.
sharedRes = const_cast<AssetManager*>(this)->
mZipSet.getZipResourceTable(ap.path);
}
if (sharedRes == NULL) {
ass = const_cast<AssetManager*>(this)->
mZipSet.getZipResourceTableAsset(ap.path);
if (ass == NULL) {
......
ass = const_cast<AssetManager*>(this)->
openNonAssetInPathLocked("resources.arsc",
Asset::ACCESS_BUFFER,
ap);
if (ass != NULL && ass != kExcludedAsset) {
ass = const_cast<AssetManager*>(this)->
mZipSet.setZipResourceTableAsset(ap.path, ass);
}
}
if (i == 0 && ass != NULL) {
// If this is the first resource table in the asset
// manager, then we are going to cache it so that we
// can quickly copy it out for others.
LOGV("Creating shared resources for %s", ap.path.string());
sharedRes = new ResTable();
sharedRes->add(ass, (void*)(i+1), false, idmap);
sharedRes = const_cast<AssetManager*>(this)->
mZipSet.setZipResourceTable(ap.path, sharedRes);
}
}
} else {
......
Asset* ass = const_cast<AssetManager*>(this)->
openNonAssetInPathLocked("resources.arsc",
Asset::ACCESS_BUFFER,
ap);
shared = false;
}
if ((ass != NULL || sharedRes != NULL) && ass != kExcludedAsset) {
if (rt == NULL) {
mResources = rt = new ResTable();
updateResourceParamsLocked();
}
......
if (sharedRes != NULL) {
......
rt->add(sharedRes);
} else {
......
rt->add(ass, (void*)(i+1), !shared, idmap);
}
if (!shared) {
delete ass;
}
}
if (idmap != NULL) {
delete idmap;
}
}
......
if (!rt) {
mResources = rt = new ResTable();
}
return rt;
}
这个函数定义在文件frameworks/base/libs/utils/AssetManager.cpp中。
AssetManager
类的成员函数getResources
的实现看起来比较复杂,但是它要做的事情就是解析当前应用程序所使用的资源包里面的resources.arsc文件。从前面Android应用程序资源管理器(Asset Manager)的创建过程分析一文可以知道,当前应用程序所使用的资源包有两个,其中一个是系统资源包
,即/system/framework/framework-res.apk
,另外一个就是自己的APK文件
。这些APK文件的路径都分别使用一个asset_path
对象来描述,并且保存在AssetManager
类的成员变量mAssetPaths
中。
AssetManager
类的成员变量mResources
指向的是一个ResTable
对象,如果它的值不等于NULL
,那么就说明当前应用程序已经解析过它使用的资源包里面的resources.arsc文件,因此,这时候AssetManager
类的成员函数getResources
就可以直接将该ResTable
对象返回给调用者。
如果当前应用程序还没有解析过它使用的资源包里面的resources.arsc文件,那么AssetManager
类的成员函数getResources
就会先获取由成员变量mLock
所描述的一个互斥锁
,避免多个线程同时去解析当前应用程序还没有解析过它使用的资源包里面的resources.arsc文件。注意
,获取锁成功之后,有可能其它线程已经抢先一步解析了当前应用程序使用的资源包里面的resources.arsc文件了,因此,这时候就需要再次判断 AssetManager类的成员函数mResources是否等于NULL。如果不等于NULL,就可以将它所指向的ResTable对象返回给调用者了。
AssetManager
类的成员函数getResources
接下来按照以下步骤来解析当前应用程序所使用的每一个资源包里面的resources.arsc
文件:
- 检查资源包里面的
resources.arsc
文件已经提取出来。如果已经提取出来的话,那么以当前正在处理的资源包路径为参数,调用当前正在处理的AssetManager
对象的成员变量mZipSet
所指向的一个ZipSet对象
的成员函数getZipResourceTableAsset
就可以获得一个对应的Asset对象
。
- 检查资源包里面的
- 如果资源包里面的
resources.arsc
文件还没有提取出来,那么就会调用当前正在处理的AssetManager
对象的成员函数openNonAssetInPathLocked
来将该resources.arsc
文件提取出来。提取的结果就是获得一个对应的Asset对象,保存在变量ass中。注意,如果当前提取出来的Asset对象的地址值不等于全局变量kExcludedAsset
的值,那么就将该Asset
对象设置为当前正在处理的AssetManager
对象的成员变量mZipSet
所指向的一个ZipSet
对象中去,这是通过调用该ZipSet
对象的成员函数setZipResourceTableAsset
来实现的。
- 如果资源包里面的
- 将上面获得的用来描述
resources.arsc
文件的Asset
对象ass
添加到变量rt
所描述的一个ResTable
对象中去,这是通过调用该ResTable
对象的成员函数add
来实现的。注意,如果该ResTable
对象还没有创建,那么它就会首先被创建,并且同时保存在AssetManager
类的成员变量mResources
和变量rt
中。另外一个地方需要注意的是,ResTable类的成员函数add
在增加一个Asset对象时,会对该Asset
对象所描述的resources.arsc
文件的内容进行解析,结果就是得到一个系列的Package
信息。每一个Package
又包含了一个资源类型字符串资源池
和一个资源项名称字符串资源池,以及一系列的资源类型规范数据块和一系列的资源项数据块。这些内容可以参考前面Android应用程序资源的编译和打包过程分析。还有第三个地方需要注意的是,每一个资源包里面的所有Pacakge形成一个PackageGroup,保存在变量rt所指向的一个ResTable对象的成员变量mPackageGroups
中。总之,ResTable类的作用就类似于在前面Android应用程序资源的编译和打包过程分析所介绍的ResourceTable
类。
- 将上面获得的用来描述
一般来说,在AssetManager
类的成员变量mAssetPaths
中,第一个资源包路径指向的就是系统资源包,而系统资源包在当前正在处理的AssetManager对象创建的时候,可能就已经提取或者初始化过了,也就是它的resources.arsc文件已经提取或者解析过了。如果已经解析过,那么在代码中的for循环中,当i等于0时
,调用当前正在处理的AssetManager对象的成员变量mZipSet
所指向的一个ZipSet
对象的成员函数getZipResourceTable
就可以获得一个ResTable
对象,并且保存在变量sharedRes
中,最终就可以直接将该ResTable
对象添加到变量rt
所指向的一个ResTable
对象中去,也就是添加到AssetManager
类的成员变量mResources
所指向的一个ResTable
对象中去。如果只是提取过,但是还没有解析过,那么就会首先对它进行解析,并且将得到的ResTable对象保存在变量sharedRes中,最后再将该ResTable对象添加到变量rt所指向的一个ResTable对象中去。
此外,AssetManager
类的成员变量mAssetPaths
保存的资源包路径指向可能不是一个APK文件,而是一个目录文件。在这种情况下,AssetManager
类的成员函数getResources
就会直接调用另外一个成员函数openNonAssetInPathLocked
来打开该目录下的resources.arsc
文件,并且获得一个Asset对象
,同样是保存在变量ass
中。
在前面Android应用程序资源管理器(Asset Manager)的创建过程分析一文还提到,Android系统的资源管理框架提供了一种idmap机制
,用来个性化定制一个资源包里面已有的资源项,也就是说,每一个资源包都可能有一个对应的idmap文件
,用来描述它所个性化定制的资源项。在提取和解析资源包的过程中,如果该资源包存在idmap
文件,那么该idmap
文件也会被解析,并且解析得到的一个Asset对象也会同时被增加到变量rt
所指向的一个ResTable
对象中去。
经过上述的一系列操作之后,AssetManager
类的成员变量mResources
所指向的一个ResTable
对象就包含了当前应用程序所使用的资源包的所有信息,该ResTable
对象最后就会返回给调用者来使用。
这一步执行完成之后,回到前面的Step 8中,即AssetManager
类的成员函数loadResourceValue
中,接下来它就会调用前面所获得的一个ResTable
对象的成员函数getResource
来获得指定的资源项内容。
Step 10. ResTable.getResource
ssize_t ResTable::getResource(uint32_t resID, Res_value* outValue, bool mayBeBag,
uint32_t* outSpecFlags, ResTable_config* outConfig) const
{
......
const ssize_t p = getResourcePackageIndex(resID);
const int t = Res_GETTYPE(resID);
const int e = Res_GETENTRY(resID);
......
const Res_value* bestValue = NULL;
const Package* bestPackage = NULL;
ResTable_config bestItem;
memset(&bestItem, 0, sizeof(bestItem)); // make the compiler shut up
if (outSpecFlags != NULL) *outSpecFlags = 0;
// Look through all resource packages, starting with the most
// recently added.
const PackageGroup* const grp = mPackageGroups[p];
......
size_t ip = grp->packages.size();
while (ip > 0) {
ip--;
int T = t;
int E = e;
const Package* const package = grp->packages[ip];
if (package->header->resourceIDMap) {
uint32_t overlayResID = 0x0;
status_t retval = idmapLookup(package->header->resourceIDMap,
package->header->resourceIDMapSize,
resID, &overlayResID);
if (retval == NO_ERROR && overlayResID != 0x0) {
// for this loop iteration, this is the type and entry we really want
......
T = Res_GETTYPE(overlayResID);
E = Res_GETENTRY(overlayResID);
} else {
// resource not present in overlay package, continue with the next package
continue;
}
}
const ResTable_type* type;
const ResTable_entry* entry;
const Type* typeClass;
ssize_t offset = getEntry(package, T, E, &mParams, &type, &entry, &typeClass);
if (offset <= 0) {
// No {entry, appropriate config} pair found in package. If this
// package is an overlay package (ip != 0), this simply means the
// overlay package did not specify a default.
// Non-overlay packages are still required to provide a default.
if (offset < 0 && ip == 0) {
......
return offset;
}
continue;
}
if ((dtohs(entry->flags)&entry->FLAG_COMPLEX) != 0) {
......
continue;
}
......
const Res_value* item =
(const Res_value*)(((const uint8_t*)type) + offset);
ResTable_config thisConfig;
thisConfig.copyFromDtoH(type->config);
if (outSpecFlags != NULL) {
if (typeClass->typeSpecFlags != NULL) {
*outSpecFlags |= dtohl(typeClass->typeSpecFlags[E]);
} else {
*outSpecFlags = -1;
}
}
if (bestPackage != NULL &&
(bestItem.isMoreSpecificThan(thisConfig) || bestItem.diff(thisConfig) == 0)) {
// Discard thisConfig not only if bestItem is more specific, but also if the two configs
// are identical (diff == 0), or overlay packages will not take effect.
continue;
}
bestItem = thisConfig;
bestValue = item;
bestPackage = package;
}
......
if (bestValue) {
outValue->size = dtohs(bestValue->size);
outValue->res0 = bestValue->res0;
outValue->dataType = bestValue->dataType;
outValue->data = dtohl(bestValue->data);
if (outConfig != NULL) {
*outConfig = bestItem;
}
......
return bestPackage->header->index;
}
return BAD_VALUE;
}
这个函数定义在文件frameworks/base/libs/utils/ResourceTypes.cpp中。
参数resID
描述的是要查找的资源的ID
,ResTable
类的成员函数getResource
分别获得它的Pakcage ID
、Type ID
以及Entry ID
,保存在变量p
、t
以及e
中。知道了Pakcage ID
之后,就可以在ResTable
类的成员变量mPackageGroups
中找到对应的PakcageGroup
。
注意,前面获得的PakcageGroup可能包含有多个Package,这些Package都保存在PakcageGroup的成员变量
packages
所描述的一个数组中,因此,ResTable类的成员函数getResource
就通过一个while循环
来在每一个Package中查找最符合条件的资源项。
如果当前正在处理的Package的成员变量header
所描述的一个Header对象的成员变量resourceIDMap
的值不等于NULL
,那么它所指向的就是一个idmap
,同时也说明当前正在处理的Package是一个Overlay Package
,也就是说,它是用来覆盖已存在的资源项的。在这种情况下,ResTable类的成员函数getResource
就会将参数resID
所描述的资源ID映射为覆盖后的资源ID。注意,如果不能将参数resID所描述的资源ID映射为覆盖后的ID,那么当前正在处理的Package就会被跳过。
无论当前正在处理的Package是否是一个Overlay Package,最想要找到的资源项的Type ID和Entry ID都保存在变量 T
和E
中,接下来ResTable类的成员函数getResource
就会以这两个变量为参数来调用另外一个成员函数getEntry
来在当前正在处理的Package中检查是否存在符合条件的资源项。如果存在的话,那么调用ResTable类的成员函数getEntry
得到的返回值offset
就会大于0,同时还会得到三个类型分别为ResTable_type
、ResTable_entry
和Type
结构体,分别保存在变量type
、entry
和typeClass
中。其中,ResTable_type用来描述一个资源类型,ResTable_entry用来描述一个资源项,而Type用来描述一个资源类型规范,关于这些结构的详细解释可以参考前面Android应用程序资源的编译和打包过程分析。
从前面Android应用程序资源的编译和打包过程分析还可以知道,对于一个普通的资源项来说,它在资源表文件resources.arsc
中,由一个ResTable_entry和一个Res_value结构体前面连接在一起表示,其中,结构体ResTable_entry用来描述资源项的头部,而结构体Res_value用来描述资源项的值,并且这两个结构体是嵌入在一个ResTable_type结构体里面的。
理解了上述背景知道之后,我们就可以解释前面得到的返回值offset
的含义了,它表示一个Res_value结构体在一个ResTable_type结构体中的偏移,也就是说,将前面获得的ResTable_type结构体type的开始地址,再加偏移量offset,就可以得到一个Res_value结构体item,而这个Res_value结构体就是表示资源ID等于resID的资源项的值。
注意,在调用ResTable类的成员函数
getEntry
来在当前正在处理的Package中查找与参数resID
对应的资源项时,还会指定设备的当前配置信息。设备的当前配置信息是由ResTable类的成员变量mParams
所指向的一个ResTable_config结构体来描述的。如果ResTable类的成员函数getEntry
能成功找到一个匹配的资源项,那么它还会通过ResTable_type结构体type
的成员变量config
所指向的一个ResTable_config结构体来返回该资源项的实际配置信息,并且保存在另外一个ResTable_config结构体thisConfig中。
现在一切就准备就绪了,ResTable类的成员函数getResource
要做的事情就是比较在前后两个Package中找到的两个资源项中,哪一个资源项更匹配设备的当前配置信息。注意,在前一个Package中找到的资源项的值及其所对应的Package和配置信息分别保存在Res_value结构体bestValue
、Package结构体bestPackage
和ResTable_config结构体bestItem
中,而在后一个Package中找到的资源项对应的配置信息保存在ResTable_config结构体thisConfig
中。
如果ResTable_config结构体bestItem
描述的配置信息比ResTable_config结构体thisConfig描述的配置信息更具体,或者它们完全是一样的,那么ResTable类的成员函数getResource就会认为在前一个Package中找到的资源项更匹配设备的当前配置信息,于是就保持Res_value结构体bestValue
、Package结构体bestPackage
和ResTable_config结构体bestItem
的值不变,否则的话,就会将在后一个Package中找到的资源项的值及其所对应的Package和配置信息分别保存在Res_value结构体bestValue
、Package结构体bestPackage
和ResTable_config结构体bestItem
中。
ResTable类的成员函数getResource
执行完成中间的while循环之后,最终得到的资源项的值以及配置信息就保存在Res_value结构体bestValue
和ResTable_config结构体bestItem
,最后就可以分别将它们的内容拷贝到输出参数outValue
和outConfig
中去,以便可以返回给调用者。同时,ResTable类的成员函数getResource
还会将最终得到的资源项所在的Package的索引返回给调用者。通过这个Package的索引值,调用者就可以知道它所找到的资源项是在哪一个资源包中找到的,例如,是在系统资源包找到的,还是在应用程序本身的资源包找到的。
事实上,ResTable类的成员函数getResource
返回给调用者的还有一个很重要的信息,那就是参数resID
所描述的资源项的配置状况,也就是说,参数resID
所描述的资源项的配置差异性信息。这个配置差异性信息就保存在前面得到的Type结构体typeClass
的成员变量typeSpecFlags
所描述的一个uint32_t
数组中的第E
个元素中,这是因为参数resID
所描述的资源项的Entry ID
等于E`。关于资源项的配置差异性信息的详细描述,可以参考前面Android应用程序资源的编译和打包过程分析所提到的ResTable_typeSpec结构体。
在ResTable类的成员函数getResource
查找资源的过程中,还有两个地方是需要注意的。
-
第一个地方是ResTable类的成员函数
getResource
是从后往前遍历
Pakcage ID等于p
的PackageGroup中的每一个Package的。在一个PackageGroup中,第一个Package是一个Base Package,其它的Package都是属于Overlay Package,其中,在Overlay Package中定义的资源是用来覆盖在Base Package中定义的资源的。如果ResTable类的成员函数getResource
在调用另外一个成员函数getEntry
来在某一个package中找不到对应的资源项时,即调用ResTable类的成员函数getEntry
得到的返回值offset
小于等于0的时候,需要进一步检查该package是一个Base Package还是一个Overlay Package。如果是一个Overlay Package,那么这种情况是允许发生的,因为一个Overlay Package只需要定义它需要覆盖的资源。另一方面,如果是一个Base Package,那么就是这种情况就是异常的,因为一个Base Package无论如何都要保证在给定一个合法的资源ID的前提下,一定可以找到一个对应的资源项,在最坏的情况下,这个资源项就是一个default类型的。 -
第二个地方是只有当调用ResTable类的成员函数
getEntry
得到资源项是一个普通资源项时,即得到的ResTable_entry结构体entry
的成员变量flags
的值的FLAG_COMPLEX
位等于0
时,才可以将得到的ResTable_type结构体type
的偏移位置offse
t转换为一个Res_value结构体来访问。这是**因为如果找到的资源项不是一个普通资源项,而是一个Bag资源项时,得到的ResTable_type结构体type的偏移位置offset是一个ResTable_map结构体数组,而不是一个Res_value结构体,Android应用程序资源的编译和打包过程分析。
接下来,我们就继续分析ResTable类的成员函数getEntry
的实现,以便可以了解它是如何在一个Package中找到指定Type ID、Entry ID以及配置信息的资源项的,如下所示:
ssize_t ResTable::getEntry(
const Package* package, int typeIndex, int entryIndex,
const ResTable_config* config,
const ResTable_type** outType, const ResTable_entry** outEntry,
const Type** outTypeClass) const
{
......
const ResTable_package* const pkg = package->package;
const Type* allTypes = package->getType(typeIndex);
......
const ResTable_type* type = NULL;
uint32_t offset = ResTable_type::NO_ENTRY;
ResTable_config bestConfig;
memset(&bestConfig, 0, sizeof(bestConfig)); // make the compiler shut up
const size_t NT = allTypes->configs.size();
for (size_t i=0; i<NT; i++) {
const ResTable_type* const thisType = allTypes->configs[i];
if (thisType == NULL) continue;
ResTable_config thisConfig;
thisConfig.copyFromDtoH(thisType->config);
......
// Check to make sure this one is valid for the current parameters.
if (config && !thisConfig.match(*config)) {
......
continue;
}
// Check if there is the desired entry in this type.
const uint8_t* const end = ((const uint8_t*)thisType)
+ dtohl(thisType->header.size);
const uint32_t* const eindex = (const uint32_t*)
(((const uint8_t*)thisType) + dtohs(thisType->header.headerSize));
uint32_t thisOffset = dtohl(eindex[entryIndex]);
if (thisOffset == ResTable_type::NO_ENTRY) {
......
continue;
}
if (type != NULL) {
// Check if this one is less specific than the last found. If so,
// we will skip it. We check starting with things we most care
// about to those we least care about.
if (!thisConfig.isBetterThan(bestConfig, config)) {
......
continue;
}
}
type = thisType;
offset = thisOffset;
bestConfig = thisConfig;
......
if (!config) break;
}
if (type == NULL) {
......
return BAD_INDEX;
}
offset += dtohl(type->entriesStart);
......
const ResTable_entry* const entry = (const ResTable_entry*)
(((const uint8_t*)type) + offset);
......
*outType = type;
*outEntry = entry;
if (outTypeClass != NULL) {
*outTypeClass = allTypes;
}
return offset + dtohs(entry->size);
}
这个函数定义在文件frameworks/base/libs/utils/ResourceTypes.cpp中。
ResTable类的成员函数getEntry
首先是在参数pakcage
所描述的一个Package中找到与参数typeIndex
所对应的一个Type结构体allTypes
。这个Type结构体综合了我们在前面Android应用程序资源的编译和打包过程分析所描述的资源项的ResTable_typeSpec数据块和ResTable_type数据块,也就是,这个Type结构体的成员变量typeSpecFlags
所指向的一个uint32_t
数组描述了同一种类型的所有资源项的配置差异性性信息,即相当于一系列ResTable_typeSpec数据块,而成员变量configs
所指向的一个ResTable_type数组描述的同一种类型的资源项的具体内容,即相当于一系列ResTable_type数据块。
结合前面Android应用程序资源的编译和打包过程分析对ResTable_typeSpec数据块和ResTable_type数据块的描述,我们就可以比较容易理解ResTable类的成员函数getEntry
的实现了。由于Type结构体allTypes
的成员变量configs
所指向的一个ResTable_type数组描述的所有类型为typeIndex
的资源项的具体内容,因此,ResTable类的成员函数getEntry
只要遍历这个ResTable_type数组,并且找到与参数entryIdex
所对应的最匹配资源项即可,也就是最能匹配参数config
所描述的配置信息的资源项即可。
我们知道,在每一个ResTable_type结构体中,都包含有一个类型为uint32_t
的偏移数组。偏移数组中的第entryIndex
个元素的值就表示Entry ID等于entryIndex
的资源项的具体内容相对于ResTable_type结构体的偏移值。有了这个偏移值之后,我们就得到Entry ID
等于entryIndex
的资源项的具体内容了,也就是得到一个对应的ResTable_entry结构体。
注意,每一个ResTable_type结构体都有一个成员变量
config
,它描述的是一个ResTable_config对象,表示该ResTable_type结构体的配置信息。如果该配置信息要求的配置信息不匹配,也就是与参数config
所描述的设置配置信息不匹配,那么就不需要进一步检查该ResTable_type结构体是否存在Entry ID等于entryIndex
的资源项了。
如果一个ResTable_type结构体的配置信息与参数config
所描述的设置配置信息匹配,但是它不存在Entry ID等于entryIndex
的资源项,即它的偏移数组中的第entryIndex
个元素的值等于ResTable_type::NO_ENTRY
,那么也不需要进一步对该ResTable_type结构体进行处理了。
如果一个ResTable_type结构体的配置信息与参数config
所描述的设置配置信息匹配,并且它也存在Entry ID等于entryIndex的资源项,那么就需要比较前后两个ResTable_type结构体的配置信息与参数config
所描述的设置配置信息相比,哪一个更具体
一些。具有更具体的配置信息的资源项将作为最匹配的资源项返回给调用者。注意,前一个ResTable_type结构体及其配置信息分别保存在变量type
和bestConfig
中,并且在该ResTable_type结构体中Entry ID等于entryIndex的资源项相对它的偏移值保存在变量offset
中。
遍历完成上面所述的ResTable_type数组之后,最终得到的最区配资源项的相关信息就保存在变量type
、bestConfig
和offset
中。这时候将变量type
所描述的ResTable_type结构体的起始位置,加上它的成员变量entriesStart
的值,以及再加上变量offset
的值,就可以得到一个对应的ResTable_entry结构体entry
。这个ResTable_entry结构体entry
就是用来描述最终在参数package所描述的Package中,找到与参数typeIndex
、entryIndex
和config
最匹配的资源项的信息了。
最终,ResTable类的成员函数getEntry
就将前面所得到的ResTable_entry结构体entry
、ResTable_type结构体type
以及Type结构体allTypes
分别保存在输出参数outEntry
、outType
和outTypeClass
返回给调用者了。同时,ResTable类的成员函数getEntry
还会将变量offset
的值加上ResTable_entry结构体entry
的大小的结果返回给调用者。调用都得到这个结果之后,就可以将它作为输出参数outType
所指向的ResTable_type结构体的偏移量,从而可以得到参数outEntry
所指向的ResTable_entry结构体所描述的资源项的具体内容,也就是得到一个Res_value结构体,这个具体就可以参考前面对ResTable类的成员函数getResource
的分析。
这一步执行完成之后,回到前面的Step 8中,即AssetManager类的成员函数loadResourceValue
中,接下来它就会调用前面所获得的一个ResTable对象的成员函数resolveReference
来解析前面所获得的资源项的内容了。
Step 11. ResTable.resolveReference
ssize_t ResTable::resolveReference(Res_value* value, ssize_t blockIndex,
uint32_t* outLastRef, uint32_t* inoutTypeSpecFlags,
ResTable_config* outConfig) const
{
int count=0;
while (blockIndex >= 0 && value->dataType == value->TYPE_REFERENCE
&& value->data != 0 && count < 20) {
if (outLastRef) *outLastRef = value->data;
uint32_t lastRef = value->data;
uint32_t newFlags = 0;
const ssize_t newIndex = getResource(value->data, value, true, &newFlags,
outConfig);
if (newIndex == BAD_INDEX) {
return BAD_INDEX;
}
......
//printf("Getting reference 0x%08x: newIndex=%d\n", value->data, newIndex);
if (inoutTypeSpecFlags != NULL) *inoutTypeSpecFlags |= newFlags;
if (newIndex < 0) {
// This can fail if the resource being referenced is a style...
// in this case, just return the reference, and expect the
// caller to deal with.
return blockIndex;
}
blockIndex = newIndex;
count++;
}
return blockIndex;
}
这个函数定义在文件frameworks/base/libs/utils/ResourceTypes.cpp中。
ResTable类的成员函数resolveReference
的实现其实很简单,它就是对参数value
所描述的一个资源项值进行解析,前提是这个资源项值是一个引用,即value
所指向的一个Res_value结构体的成员变量dataType
的值等于TYPE_REFERENCE
,因为如果不是引用类型的资源项值,就没有必要解析了。
注意,一个资源项的值有可能是嵌套引用的,也就是可能是引用的引用,因此,ResTable类的成员函数
resolveReference
需要使用一个while循环来不断地对参数value所描述的一个资源项值进行解析,直到最后一次解析出来的结果不是引用为止。但是,为了防止无限地解析下去,该while循环最多只允许执行20次。每一次解析都是调用ResTable类的成员函数getResource
来实现的,这个成员函数我们在前面的Step 10中已经分析过了。
此外,上述while循环
还有两个条件需要满足。
- 第一个条件是每一次调用ResTable类的成员函数
getResource
来解析参数value
所描述的资源项值之后,得到的返回值blockIndex
都必须是大于等于0
的,这是因为它表示该次解析是成功的。由于参数blockIndex
最开始的值是由调用者传进来的,因此,也要保证它最开始的值大于等于0,才会执行代码中的while
循环对参数value
所描述的资源项值进行解析。 - 第二个条件是参数
value
所描述的资源项的值不能等于0
,即它所指向的一个Res_value结构体的成员变量data的值不能等于0
,这是因为引用者都是不可能等于0的。
这一步执行完成之后,沿着调用路径最终返回到前面的Step 5中,即Resources类的成员函数loadXmlResourceParser
中,我们就可以得到参数id
所描述的资源项的值了。在我们这个情景中,参数id
描述的是一个layout
资源ID,它所对应的资源项的值是一个字符串,这个字符串描述的便是一个UI布局文件,即一个经过编译的、以二进制格式保存的Xml资源文件。有了这个Xml资源文件的路径之后,Resources类的另外一个四个参数版本的成员函数loadXmlResourceParser
就会被调用来对该Xml资源文件进行解析,以便可以得到一个UI布局视图。
Step 12. Resources.loadXmlResourceParser
public class Resources {
......
private int mLastCachedXmlBlockIndex = -1;
private final int[] mCachedXmlBlockIds = { 0, 0, 0, 0 };
private final XmlBlock[] mCachedXmlBlocks = new XmlBlock[4];
......
/*package*/ XmlResourceParser loadXmlResourceParser(String file, int id,
int assetCookie, String type) throws NotFoundException {
if (id != 0) {
try {
// These may be compiled...
synchronized (mCachedXmlBlockIds) {
// First see if this block is in our cache.
final int num = mCachedXmlBlockIds.length;
for (int i=0; i<num; i++) {
if (mCachedXmlBlockIds[i] == id) {
......
return mCachedXmlBlocks[i].newParser();
}
}
// Not in the cache, create a new block and put it at
// the next slot in the cache.
XmlBlock block = mAssets.openXmlBlockAsset(
assetCookie, file);
if (block != null) {
int pos = mLastCachedXmlBlockIndex+1;
if (pos >= num) pos = 0;
mLastCachedXmlBlockIndex = pos;
XmlBlock oldBlock = mCachedXmlBlocks[pos];
if (oldBlock != null) {
oldBlock.close();
}
mCachedXmlBlockIds[pos] = id;
mCachedXmlBlocks[pos] = block;
......
return block.newParser();
}
}
} catch (Exception e) {
NotFoundException rnf = new NotFoundException(
"File " + file + " from xml type " + type + " resource ID #0x"
+ Integer.toHexString(id));
rnf.initCause(e);
throw rnf;
}
}
throw new NotFoundException(
"File " + file + " from xml type " + type + " resource ID #0x"
+ Integer.toHexString(id));
}
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/Resources.java中
Resources类有三个成员变量是与Xml资源文件缓存文件有关的,它们分别是mCachedXmlBlocks
、mCachedXmlBlockIds
和mLastCachedXmlBlockIndex
。其中,mCachedXmlBlocks
指向的是一个大小等于4的XmlBlock数组,mCachedXmlBlockIds
指向的也是一个大小等于4的资源ID数组,而mLastCachedXmlBlockIndex
表示上一次缓存的Xml资源文件在上述XmlBlock数组和资源ID数组中的索引。这就是说,Resources类最多可以缓存最近读取的四个Xml资源文件的内容,读取超过四个Xml资源文件之后 ,上述的XmlBlock数组和资源ID数组就会被循环利用。
理解了Resources类的上述三个成员变量的含义之后,Resources类的成员函数loadXmlResourceParser
的实现就容易理解了。它首先是在成员变量mCachedXmlBlockIds
所指向的资源ID数组中检查是否存在一个资源ID
与参数id
所描述的资源ID相等。如果存在的话,那么就会在成员变量mCachedXmlBlocks
所指向的XmlBlock数组中找到一个对应的XmlBlock对象,并且调用这个XmlBlock对象的成员函数newParser
来创建一个XmlResourceParser
对象返回给调用者。
如果在Resources类的成员变量mCachedXmlBlockIds
所指向的资源ID数组找到对应的资源ID的话,那么Resources类的成员函数loadXmlResourceParser
就会调用成员变量mAssets
所指向的一个Java层的AssetManager
对象的成员函数openXmlBlockAsset
来打开参数file
所指定的Xml资源文件,从而获得一个XmlBlock对象block
。这个XmlBlock对象block
以及参数id
所描述的资源ID同时也会被缓存在Resources类的成员变量mCachedXmlBlocks
和mCachedXmlBlockIds
所描述的XmlBlock数组和资源ID数组中。
最后,Resources类的成员函数loadXmlResourceParser
就可以调用前面得到的XmlBlock对象block
的成员函数newParser
来创建一个XmlResourceParser对象返回给调用者。
接下来,我们就首先分析Java层的AssetManager
类的成员函数openXmlBlockAsset
的实现,接着再分析XmlBlock类的成员函数newParser
的实现,以便可以了解一个Xml资源文件的打开过程。
Step 13. AssetManager.openXmlBlockAsset
public final class AssetManager {
......
/*package*/ final XmlBlock openXmlBlockAsset(int cookie, String fileName)
throws IOException {
synchronized (this) {
if (!mOpen) {
throw new RuntimeException("Assetmanager has been closed");
}
int xmlBlock = openXmlAssetNative(cookie, fileName);
if (xmlBlock != 0) {
XmlBlock res = new XmlBlock(this, xmlBlock);
incRefsLocked(res.hashCode());
return res;
}
}
throw new FileNotFoundException("Asset XML file: " + fileName);
}
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/AssetManager.java中。
AssetManager类的成员函数openXmlBlockAsset
首先检查成员变量mOpen
的值是否不等于true
。如果不等于true
的话,那么就说明当前正在处理的AssetManager对象还没有经过初始化,或者已经关闭了。
我们假设AssetManager类的成员变量mOpen
的值等于true
,那么接下来AssetManager类的成员函数openXmlBlockAsset
就会调用另外一个成员函数openXmlAssetNative
来打开参数fileName
所指定的Xml资源文件。
成功打开参数fileName
所指向的Xml资源文件之后,就会得到一个C++层的ResXMLTree对象
的地址值xmlBlock
,最后AssetManager类的成员函数openXmlBlockAsset
就将该C++层的ResXMLTree对象的地址值
封装在一个Java层的XmlBlock对象
中,并且将该XmlBlock对象返回给调用者。
接下来,我们就继续分析AssetManager类的成员函数openXmlAssetNative
的实现。
Step 14. AssetManager.openXmlAssetNative
public final class AssetManager {
......
private native final int openXmlAssetNative(int cookie, String fileName);
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/AssetManager.java中。
AssetManager类的成员函数openXmlAssetNative
是一个JNI方法,它是C++层的函数android_content_AssetManager_openXmlAssetNative
来实现的,如下所示:
static jint android_content_AssetManager_openXmlAssetNative(JNIEnv* env, jobject clazz,
jint cookie,
jstring fileName)
{
AssetManager* am = assetManagerForJavaObject(env, clazz);
......
const char* fileName8 = env->GetStringUTFChars(fileName, NULL);
Asset* a = cookie
? am->openNonAsset((void*)cookie, fileName8, Asset::ACCESS_BUFFER)
: am->openNonAsset(fileName8, Asset::ACCESS_BUFFER);
......
env->ReleaseStringUTFChars(fileName, fileName8);
ResXMLTree* block = new ResXMLTree();
status_t err = block->setTo(a->getBuffer(true), a->getLength(), true);
a->close();
delete a;
......
return (jint)block;
}
这个函数定义在文件frameworks/base/core/jni/android_util_AssetManager.cpp中。
函数android_content_AssetManager_openXmlAssetNative
首先调用另外一个函数assetManagerForJavaObject
来将参数clazz
所指向的一个Java层的AssetManager对象成员变量mObject
转换为一个C++层的AssetManager对象
。有了这个C++层的AssetManager对象
之后,就可以调用它的成员函数openNonAsset
来打开参数fileName
所指定的Xml资源文件。
调用C++层的AssetManager对象
的成员函数openNonAsset
来成功地打开参数fileName
所指定的Xml资源文件之后,函数android_content_AssetManager_openXmlAssetNative
接下来就会创建一个ResXMLTree对象
,并且将前面所打开的Xml资源文件的内容设置到该ResXMLTree对象中去,并且将该ResXMLTree对象的地址值
返回给调用者。
假设参数cookie的值大于0
,因此,函数android_content_AssetManager_openXmlAssetNative
实际调用的是C++层的AssetManager
类的三个参数版本的成员函数openNonAsset
来打开参数fileName
所指定的Xml资源文件,接下来,我们就分析这个函数的实现。
Step 15. AssetManager.openNonAsset
Asset* AssetManager::openNonAsset(void* cookie, const char* fileName, AccessMode mode)
{
const size_t which = ((size_t)cookie)-1;
AutoMutex _l(mLock);
......
if (which < mAssetPaths.size()) {
......
Asset* pAsset = openNonAssetInPathLocked(
fileName, mode, mAssetPaths.itemAt(which));
if (pAsset != NULL) {
return pAsset != kExcludedAsset ? pAsset : NULL;
}
}
return NULL;
}
这个函数定义在文件frameworks/base/libs/utils/AssetManager.cpp中。
参数cookie
是用来标识另外一个参数fileName
所指向的文件是属于哪个APK文件的。从前面Android应用程序资源管理器(Asset Manager)的创建过程分析一文可以知道,参数cookie
实际上是一个整数,将这个整数减去1之后,得到的结果就是参数fileName
所指向的文件所属于的APK文件在AssetManager类的成员变量mAssetPaths
所描述的一个数组的索引。AssetManager类的成员变量mAssetPaths
描述的是一个asset_path数组
,数组中的每一个元素都是表示一个APK文件路径,这些APK文件就包含了当前应用程序所要使用的资源包。
AssetManager类的成员函数openNonAsset
通过参数cookie
知道了当前要打开的文件是位于哪个APK文件之后,接着就继续调用另外一个成员函数openNonAssetInPathLocked
来打开该文件。
Step 16. AssetManager.openNonAssetInPathLocked
Asset* AssetManager::openNonAssetInPathLocked(const char* fileName, AccessMode mode,
const asset_path& ap)
{
Asset* pAsset = NULL;
/* look at the filesystem on disk */
if (ap.type == kFileTypeDirectory) {
String8 path(ap.path);
path.appendPath(fileName);
pAsset = openAssetFromFileLocked(path, mode);
if (pAsset == NULL) {
/* try again, this time with ".gz" */
path.append(".gz");
pAsset = openAssetFromFileLocked(path, mode);
}
if (pAsset != NULL) {
//printf("FOUND NA '%s' on disk\n", fileName);
pAsset->setAssetSource(path);
}
/* look inside the zip file */
} else {
String8 path(fileName);
/* check the appropriate Zip file */
ZipFileRO* pZip;
ZipEntryRO entry;
pZip = getZipFileLocked(ap);
if (pZip != NULL) {
//printf("GOT zip, checking NA '%s'\n", (const char*) path);
entry = pZip->findEntryByName(path.string());
if (entry != NULL) {
//printf("FOUND NA in Zip file for %s\n", appName ? appName : kAppCommon);
pAsset = openAssetFromZipLocked(pZip, entry, mode, path);
}
}
if (pAsset != NULL) {
/* create a "source" name, for debug/display */
pAsset->setAssetSource(
createZipSourceNameLocked(ZipSet::getPathName(ap.path.string()), String8(""),
String8(fileName)));
}
}
return pAsset;
}
这个函数定义在文件frameworks/base/libs/utils/AssetManager.cpp中。
AssetManager类的成员函数openNonAssetInPathLocked
的实现是比较简单的,它按照以下两种
方式来打开参数fileName
所指向的文件。
如果参数ap
描述的文件路径是一个目录,那么它就会直接将参数fileName
所指向的文件名称附加到该目录后面去,然后直接调用另外一个成员函数openAssetFromFileLocked
来打开参数fileName
所指向的文件。如果打开失败,那么就再假定参数ap
描述的文件路径是一个以“.gz“
为后缀的压缩包,然后再次调用成员函数openAssetFromFileLocked
来打开参数fileName
所指向的文件。
如果参数ap
描述的文件路径是一个普通文件,那么就意味着参数ap
描述的是一个压缩文件,因此,它就会先调用成员函数getZipFileLocked
来打开该压缩文件,然后再调用成员函数openAssetFromZipLocked
来在该压缩包将参数fileName
所指向的文件提取出来。
Android应用程序的资源一般都是打包在一个APK文件里面的,而APK文件就是一个Zip格式的压缩包,因此,AssetManager类的成员函数openNonAssetInPathLocked
一般就是按照第二种方式来打开参数fileName
所指向的文件。
接下来,我们就继续分析AssetManager类的成员函数openAssetFromZipLocked
的实现。
Step 17. AssetManager.openAssetFromZipLocked
Asset* AssetManager::openAssetFromZipLocked(const ZipFileRO* pZipFile,
const ZipEntryRO entry, AccessMode mode, const String8& entryName)
{
Asset* pAsset = NULL;
// TODO: look for previously-created shared memory slice?
int method;
size_t uncompressedLen;
......
if (!pZipFile->getEntryInfo(entry, &method, &uncompressedLen, NULL, NULL,
NULL, NULL))
{
LOGW("getEntryInfo failed\n");
return NULL;
}
FileMap* dataMap = pZipFile->createEntryFileMap(entry);
if (dataMap == NULL) {
LOGW("create map from entry failed\n");
return NULL;
}
if (method == ZipFileRO::kCompressStored) {
pAsset = Asset::createFromUncompressedMap(dataMap, mode);
LOGV("Opened uncompressed entry %s in zip %s mode %d: %p", entryName.string(),
dataMap->getFileName(), mode, pAsset);
} else {
pAsset = Asset::createFromCompressedMap(dataMap, method,
uncompressedLen, mode);
LOGV("Opened compressed entry %s in zip %s mode %d: %p", entryName.string(),
dataMap->getFileName(), mode, pAsset);
}
if (pAsset == NULL) {
/* unexpected */
LOGW("create from segment failed\n");
}
return pAsset;
}
这个函数定义在文件frameworks/base/libs/utils/AssetManager.cpp中。
AssetManager类的成员函数openAssetFromZipLocked
的实现也很简单,它无非就是从参数pZipFile
所描述的压缩包中将参数entryName
所描述的文件提取出来,并且根据提取出来的内容保存在一个Asset对象中返回给调用者。
注意,参数
entryName
所描述的文件有可能是经过是经过压缩后再打包到参数pZipFile
所描述的压缩包去的。在这种情况下,AssetManager类的成员函数openAssetFromZipLocked
就会调用Asset类的静态成员函数createFromCompressedMap
来对它进行解压,然后再将解压完成后得到的内容保存在一个Asset对象中。否则的话,AssetManager类的成员函数openAssetFromZipLocked
就会调用Asset类的静态成员函数createFromUncompressedMap
来将它的内容保存在一个Asset对象中。
这一步执行完成之后,返回到前面的Step 12中,即Resources类的成员函数loadXmlResourceParser中,接下来就会调用Java层的XmlBlock类的成员函数newParser
来创建一个XmlResourceParser对象,以便可以用来解析前面打开的Xml资源文件。
Step 18. XmlBlock.newParser
final class XmlBlock {
......
public XmlResourceParser newParser() {
synchronized (this) {
if (mNative != 0) {
return new Parser(nativeCreateParseState(mNative), this);
}
return null;
}
}
......
private final int mNative;
......
private static final native int nativeCreateParseState(int obj);
......
}
这个函数定义在文件frameworks/base/core/java/android/content/res/XmlBlock.java中。
XmlBlock类的成员函数mNative
指向的是C++层的一个ResXMLTree对象
。这个ResXMLTree对象是在前面的Step 14中创建的,用来描述前面所打开的一个Xml资源文件。
XmlBlock类的成员函数newParser
首先是以成员变量mNative
所描述的一个C++层的ResXMLTree对象的地址
为参数,来调用JNI方法nativeCreateParseState
,用来在C++层创建一个ResXMLParser对象
,最后再将该C++层的ResXMLParser对象
封装成Java层的一个Parser对象中
,并且将该Parser对象返回给调用者。
这一步执行完成之后,返回到前面的Step 3中,即LayoutInflater类的成员函数inflate
中,接下来它就会以前面所获得的一个Java层的Parser对象
来参数,来调用另外一个重载版本的成员函数inflate
,用来解析前面所打开的Xml资源文件,即一个UI布局文件,以便可以创建相应的UI布局出来。
Step 19. LayoutInflater.inflate
public abstract class LayoutInflater {
......
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 (TAG_MERGE.equals(name)) {
......
rInflate(parser, root, attrs);
} else {
// Temp is the root view that was found in the xml
View temp = createViewFromTag(name, attrs);
ViewGroup.LayoutParams params = null;
if (root != null) {
......
// 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);
}
}
......
// Inflate all children under temp
rInflate(parser, temp, attrs);
......
// 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) {
......
} catch (IOException e) {
......
} finally {
......
}
return result;
}
}
......
}
这个函数定义在文件frameworks/base/core/java/android/view/LayoutInflater.java中。
LayoutInflater类的成员函数inflate
主要负责处理前面所打开的Xml资源文件的根节点,然后再调用另外一个成员函数rInflate
来处理根节点的子节点。每一个节点都表示一个UI控件,这个UI控件是通过调用LayoutInflater类的成员函数createViewFromTag
来创建的。
LayoutInflater类的成员函数createViewFromTag
需要两个参数来创建一个UI控件。这两个参数分别对应于当前正在处理的Xml节点的名称以及属性集。注意,如果参数root的值不等于null,那么它所描述的一个ViewGroup就是调用LayoutInflater类的成员函数createViewFromTag
获得的UI控件的父控件。因此,调用LayoutInflater类的成员函数createViewFromTag
获得的UI控件及其对应的布局参数,最后都需要添加到参数root
所描述的一个ViewGroup中去。
有一种特殊情况,如果当前正在处理的Xml节点的名称等于TAG_MERGE
,即“merge”
,那么就表示不用处理的当前正在处理的Xml节点,而是直接去处理当前正在处理的Xml节点的子节点。这是Android系统为提供的一种UI布局优化机制,实际上就是减少了一层UI嵌套,具体可以参考官方文档:http://developer.android.com/training/improving-layouts/reusing-layouts.html。
LayoutInflater类的成员函数rInflate
在处理当前节点的子节点的时候,也是通过调用成员函数createViewFromTag
来创建相应的UI控件的,因此,接下来我们就主要分析LayoutInflater类的成员函数createViewFromTag
的实现。
Step 20. LayoutInflater.createViewFromTag
public abstract class LayoutInflater {
......
private Factory mFactory;
......
View createViewFromTag(String name, AttributeSet attrs) {
if (name.equals("view")) {
name = attrs.getAttributeValue(null, "class");
}
......
try {
View view = (mFactory == null) ? null : mFactory.onCreateView(name,
mContext, attrs);
if (view == null) {
if (-1 == name.indexOf('.')) {
view = onCreateView(name, attrs);
} else {
view = createView(name, null, attrs);
}
}
......
return view;
} catch (InflateException e) {
......
} catch (ClassNotFoundException e) {
......
} catch (Exception e) {
......
}
}
......
}
这个函数定义在文件frameworks/base/core/java/android/view/LayoutInflater.java中。
参数name
表示当前正在处理的Xml节点的名称。如果它的值等于“view
”的话,那么真正要创建的UI控件的类名记录在参数attrs
所描述的一个属性集中的一个名称为“class
”的属性中。因此,当参数name的值等于“view
”的时候,LayoutInflater类的成员函数createViewFromTag
首先要做的便是从参数attrs
所描述的一个属性集中获取接下来真正要创建的UI控件的类名。
LayoutInflater类的成员函数createViewFromTag
接下来检查成员变量mFactory
的值是否不等于null。如果不等于null的话,那么它就会指向一个Factory对象,该Factory对象描述的是一个UI控件创建工厂,专门用来负责创建UI控件。
如果LayoutInflater类的成员变量mFactory
的值等于null
,那么LayoutInflater类的成员函数createViewFromTag
就会调用成员函数onCreateView
或者createView
来创建由参数name
所指定的UI控件,取决于参数name
是否包含了一个“.”字符
。注意,如果参数name是否包含了一个“.”字符,那么就说明当前所创建的UI控件是一个用户自定义的UI控件,也就是不是Android提供的标准控件**。
我们假设LayoutInflater类的成员变量mFactory
的值等于null,并且参数name
没有包含有“.”字符,那么LayoutInflater类的成员函数createViewFromTag
最后就会调用成员函数onCreateView
来创建由参数name所指定的UI控件。
LayoutInflater类的成员函数onCreateView
是由其子类来重写的。从前面的Step 2可以知道,当前正在处理的实际上是一个PhoneLayoutInflater对象。PhoneLayoutInflater类继承了LayoutInflater类,并且重写了成员函数onCreateView
。因此,接下来我们就继续分析PhoneLayoutInflater类的成员函数onCreateView
的实现。
Step 21. PhoneLayoutInflater.onCreateView
public class PhoneLayoutInflater extends LayoutInflater {
private static final String[] sClassPrefixList = {
"android.widget.",
"android.webkit."
};
@Override protected View onCreateView(String name, AttributeSet attrs) throws ClassNotFoundException {
for (String prefix : sClassPrefixList) {
try {
View view = createView(name, prefix, attrs);
if (view != null) {
return view;
}
} catch (ClassNotFoundException e) {
// In this case we want to let the base class take a crack
// at it.
}
}
return super.onCreateView(name, attrs);
}
......
}
这个函数定义在文件frameworks/base/policy/src/com/android/internal/policy/impl/PhoneLayoutInflater.java中。
PhoneLayoutInflater类的成员函数onCreateView
只负责创建两类标准的UI控件,一种是属于android.widget
包的,另一种是属于android.webkit
包的,其中,优先创建android.widget
包的UI控件。
如果参数name所描述的UI控件既不属于android.widget包的,也不属于android.webkit包的,那么 PhoneLayoutInflater类的成员函数onCreateView
就会将创建UI控件的操作交给父类来处理,即通过调用父类的成员函数onCreateView
来创建。
如果参数name
所描述的UI控件是属于android.widget包或者android.webkit包的,那么PhoneLayoutInflater类的成员函数onCreateView
就会直接调用父类LayoutInflater的成员函数createView
来创建参数name所描述的UI控件,因此,接下来我们就继续分析LayoutInflater类的成员函数createView
的实现。
Step 22. LayoutInflater.createView
public abstract class LayoutInflater {
......
private Filter mFilter;
private final Object[] mConstructorArgs = new Object[2];
......
private static final HashMap<String, Constructor> sConstructorMap =
new HashMap<String, Constructor>();
private HashMap<String, Boolean> mFilterMap;
......
public final View createView(String name, String prefix, AttributeSet attrs)
throws ClassNotFoundException, InflateException {
Constructor constructor = sConstructorMap.get(name);
Class 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);
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);
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;
return (View) constructor.newInstance(args);
} catch (NoSuchMethodException e) {
......
} catch (ClassNotFoundException e) {
......
} catch (Exception e) {
......
}
}
......
}
这个函数定义在文件frameworks/base/core/java/android/view/LayoutInflater.java。
LayoutInflater类的静态成员变量sConstructorMap
指向的是一个HashMap。这个HashMap缓存了当前应用程序使用过的每一种类型的UI控件类的构造函数。这些构造函数必须具有两个参数,其中第一个参数是一个Context,第二个参数是一个AttributeSet。LayoutInflater类的成员函数createView
在创建一个UI控件的时候,就会将上述两个参数保存在LayoutInflater类的成员变量mConstructorArgs
所描述的一个大小为2的数组中,并且以这个数组为参数,来调用对应的UI控件类的构造函数。
LayoutInflater类的mFilter
指向的是一个Filter对象。这个Filter对象描述的是一个过滤器,LayoutInflater类的成员函数createView
在创建一个UI控件之前,会先调用该过滤器的成员函数onLoadClass
,来询问该过滤器是允许创建由参数name
所描述的UI控件。如果不允许的话,LayoutInflater类的成员函数createView
就会调用另外一个成员函数failNotAllowed
来抛出一个异常。
为了避免每次创建一个UI控件时,都去询问过滤器是否允许创建,LayoutInflater类的成员函数createView
只会在第一次创建一个名称为name的UI控件时,才会询问过滤器,并且将询问结果保存在成员变量mFilterMap所指向的一个HashMap。这样当LayoutInflater类的成员函数createView
以后再创建同名的UI控件时,就可以直接通过成员变量mFilterMap
所指向的一个HashMap来知道该UI控件是否是允许创建的。
理解了LayoutInflater类的静态成员变量sConstructorMap
以及成员变量mConstructorArgs
、mFilter
和mFilterMap
的含义之后,读者就可以自己去理解LayoutInflater类的成员函数createView
的实现了。这里需要重复强调的一点就是,我们在自定义一个UI控件的时候,一定要提供一个具有两个参数类型分别为Context和AttributeSet的构造函数,否则的话,该自定义控件就不可以在UI布局文件中使用。
至此,我们就以layout
资源为例,详细地分析了Android应用程序资源的查找过程了,并且也完整地分析完成Android系统的资源管理框架了。重新学习Android系统的资源管理框架,请参考前面Android资源管理框架(Asset Manager)。
————————————————
版权声明:本文为CSDN博主「罗升阳」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/luoshengyang/article/details/8806798
网友评论